// Copyright 2014 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "mojo/services/html_viewer/webclipboard_impl.h" #include "base/bind.h" #include "mojo/services/html_viewer/blink_basic_type_converters.h" using mojo::Array; using mojo::Clipboard; using mojo::Map; using mojo::String; namespace html_viewer { namespace { void CopyUint64(uint64_t* output, uint64_t input) { *output = input; } void CopyWebString(blink::WebString* output, const Array& input) { // blink does not differentiate between the requested data type not existing // and the empty string. if (input.is_null()) { output->reset(); } else { *output = blink::WebString::fromUTF8( reinterpret_cast(&input.front()), input.size()); } } void CopyURL(blink::WebURL* pageURL, const Array& input) { if (input.is_null()) { *pageURL = blink::WebURL(); } else { *pageURL = GURL(std::string(reinterpret_cast(&input.front()), input.size())); } } void CopyVectorString(std::vector* output, const Array& input) { *output = input.To >(); } template bool Contains(const std::vector& v, const U& item) { return std::find(v.begin(), v.end(), item) != v.end(); } const char kMimeTypeWebkitSmartPaste[] = "chromium/x-webkit-paste"; } // namespace WebClipboardImpl::WebClipboardImpl(mojo::ClipboardPtr clipboard) : clipboard_(clipboard.Pass()) { } WebClipboardImpl::~WebClipboardImpl() { } uint64_t WebClipboardImpl::sequenceNumber(Buffer buffer) { mojo::Clipboard::Type clipboard_type = ConvertBufferType(buffer); uint64_t number = 0; clipboard_->GetSequenceNumber(clipboard_type, base::Bind(&CopyUint64, &number)); // Force this to be synchronous. clipboard_.WaitForIncomingMethodCall(); return number; } bool WebClipboardImpl::isFormatAvailable(Format format, Buffer buffer) { Clipboard::Type clipboard_type = ConvertBufferType(buffer); std::vector types; clipboard_->GetAvailableMimeTypes( clipboard_type, base::Bind(&CopyVectorString, &types)); // Force this to be synchronous. clipboard_.WaitForIncomingMethodCall(); switch (format) { case FormatPlainText: return Contains(types, Clipboard::MIME_TYPE_TEXT); case FormatHTML: return Contains(types, Clipboard::MIME_TYPE_HTML); case FormatSmartPaste: return Contains(types, kMimeTypeWebkitSmartPaste); case FormatBookmark: // This might be difficult. return false; } return false; } blink::WebVector WebClipboardImpl::readAvailableTypes( Buffer buffer, bool* containsFilenames) { Clipboard::Type clipboard_type = ConvertBufferType(buffer); std::vector types; clipboard_->GetAvailableMimeTypes( clipboard_type, base::Bind(&CopyVectorString, &types)); // Force this to be synchronous. clipboard_.WaitForIncomingMethodCall(); // AFAICT, every instance of setting containsFilenames is false. *containsFilenames = false; blink::WebVector output(types.size()); for (size_t i = 0; i < types.size(); ++i) { output[i] = blink::WebString::fromUTF8(types[i]); } return output; } blink::WebString WebClipboardImpl::readPlainText(Buffer buffer) { Clipboard::Type type = ConvertBufferType(buffer); blink::WebString text; clipboard_->ReadMimeType(type, Clipboard::MIME_TYPE_TEXT, base::Bind(&CopyWebString, &text)); // Force this to be synchronous. clipboard_.WaitForIncomingMethodCall(); return text; } blink::WebString WebClipboardImpl::readHTML(Buffer buffer, blink::WebURL* pageURL, unsigned* fragmentStart, unsigned* fragmentEnd) { Clipboard::Type type = ConvertBufferType(buffer); blink::WebString html; clipboard_->ReadMimeType(type, Clipboard::MIME_TYPE_HTML, base::Bind(&CopyWebString, &html)); clipboard_.WaitForIncomingMethodCall(); *fragmentStart = 0; *fragmentEnd = static_cast(html.length()); clipboard_->ReadMimeType(type, Clipboard::MIME_TYPE_URL, base::Bind(&CopyURL, pageURL)); clipboard_.WaitForIncomingMethodCall(); return html; } blink::WebString WebClipboardImpl::readCustomData( Buffer buffer, const blink::WebString& mime_type) { Clipboard::Type clipboard_type = ConvertBufferType(buffer); blink::WebString data; clipboard_->ReadMimeType( clipboard_type, mime_type.utf8(), base::Bind(&CopyWebString, &data)); // Force this to be synchronous. clipboard_.WaitForIncomingMethodCall(); return data; } void WebClipboardImpl::writePlainText(const blink::WebString& text) { Map> data; data[Clipboard::MIME_TYPE_TEXT] = Array::From(text); clipboard_->WriteClipboardData(Clipboard::TYPE_COPY_PASTE, data.Pass()); } void WebClipboardImpl::writeHTML(const blink::WebString& htmlText, const blink::WebURL& url, const blink::WebString& plainText, bool writeSmartPaste) { Map> data; data[Clipboard::MIME_TYPE_TEXT] = Array::From(plainText); data[Clipboard::MIME_TYPE_HTML] = Array::From(htmlText); data[Clipboard::MIME_TYPE_URL] = Array::From(url.string()); if (writeSmartPaste) data[kMimeTypeWebkitSmartPaste] = Array::From(blink::WebString()); clipboard_->WriteClipboardData(Clipboard::TYPE_COPY_PASTE, data.Pass()); } Clipboard::Type WebClipboardImpl::ConvertBufferType(Buffer buffer) { switch (buffer) { case BufferStandard: return Clipboard::TYPE_COPY_PASTE; case BufferSelection: return Clipboard::TYPE_SELECTION; } NOTREACHED(); return Clipboard::TYPE_COPY_PASTE; } } // namespace html_viewer