blob: e12314983e68ccfc0a3f0362d206951f4d1eb883 [file] [log] [blame]
/*
* Copyright (C) 2011 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "core/platform/chromium/ChromiumDataObjectItem.h"
#include "core/dom/DataTransferItem.h"
#include "core/dom/StringCallback.h"
#include "core/fileapi/Blob.h"
#include "core/fileapi/File.h"
#include "core/platform/SharedBuffer.h"
#include "core/platform/chromium/ClipboardMimeTypes.h"
#include "core/platform/chromium/ClipboardUtilitiesChromium.h"
#include "public/platform/Platform.h"
#include "public/platform/WebClipboard.h"
namespace WebCore {
PassRefPtr<ChromiumDataObjectItem> ChromiumDataObjectItem::createFromString(const String& type, const String& data)
{
RefPtr<ChromiumDataObjectItem> item = adoptRef(new ChromiumDataObjectItem(DataTransferItem::kindString, type));
item->m_data = data;
return item.release();
}
PassRefPtr<ChromiumDataObjectItem> ChromiumDataObjectItem::createFromFile(PassRefPtr<File> file)
{
RefPtr<ChromiumDataObjectItem> item = adoptRef(new ChromiumDataObjectItem(DataTransferItem::kindFile, file->type()));
item->m_file = file;
return item.release();
}
PassRefPtr<ChromiumDataObjectItem> ChromiumDataObjectItem::createFromURL(const String& url, const String& title)
{
RefPtr<ChromiumDataObjectItem> item = adoptRef(new ChromiumDataObjectItem(DataTransferItem::kindString, mimeTypeTextURIList));
item->m_data = url;
item->m_title = title;
return item.release();
}
PassRefPtr<ChromiumDataObjectItem> ChromiumDataObjectItem::createFromHTML(const String& html, const KURL& baseURL)
{
RefPtr<ChromiumDataObjectItem> item = adoptRef(new ChromiumDataObjectItem(DataTransferItem::kindString, mimeTypeTextHTML));
item->m_data = html;
item->m_baseURL = baseURL;
return item.release();
}
PassRefPtr<ChromiumDataObjectItem> ChromiumDataObjectItem::createFromSharedBuffer(const String& name, PassRefPtr<SharedBuffer> buffer)
{
RefPtr<ChromiumDataObjectItem> item = adoptRef(new ChromiumDataObjectItem(DataTransferItem::kindFile, String()));
item->m_sharedBuffer = buffer;
item->m_title = name;
return item.release();
}
PassRefPtr<ChromiumDataObjectItem> ChromiumDataObjectItem::createFromPasteboard(const String& type, uint64_t sequenceNumber)
{
if (type == mimeTypeImagePng)
return adoptRef(new ChromiumDataObjectItem(DataTransferItem::kindFile, type, sequenceNumber));
return adoptRef(new ChromiumDataObjectItem(DataTransferItem::kindString, type, sequenceNumber));
}
ChromiumDataObjectItem::ChromiumDataObjectItem(const String& kind, const String& type)
: m_source(InternalSource)
, m_kind(kind)
, m_type(type)
, m_sequenceNumber(0)
{
}
ChromiumDataObjectItem::ChromiumDataObjectItem(const String& kind, const String& type, uint64_t sequenceNumber)
: m_source(PasteboardSource)
, m_kind(kind)
, m_type(type)
, m_sequenceNumber(sequenceNumber)
{
}
void ChromiumDataObjectItem::getAsString(PassRefPtr<StringCallback> callback, ScriptExecutionContext* context) const
{
if (!callback || kind() != DataTransferItem::kindString)
return;
callback->scheduleCallback(context, internalGetAsString());
}
PassRefPtr<Blob> ChromiumDataObjectItem::getAsFile() const
{
if (kind() != DataTransferItem::kindFile)
return 0;
if (m_source == InternalSource) {
if (m_file)
return m_file;
ASSERT(m_sharedBuffer);
// FIXME: This code is currently impossible--we never populate m_sharedBuffer when dragging
// in. At some point though, we may need to support correctly converting a shared buffer
// into a file.
return 0;
}
ASSERT(m_source == PasteboardSource);
if (type() == mimeTypeImagePng) {
// FIXME: This is pretty inefficient. We copy the data from the browser
// to the renderer. We then place it in a blob in WebKit, which
// registers it and copies it *back* to the browser. When a consumer
// wants to read the data, we then copy the data back into the renderer.
// https://bugs.webkit.org/show_bug.cgi?id=58107 has been filed to track
// improvements to this code (in particular, add a registerClipboardBlob
// method to the blob registry; that way the data is only copied over
// into the renderer when it's actually read, not when the blob is
// initially constructed).
RefPtr<SharedBuffer> data = static_cast<PassRefPtr<SharedBuffer> >(WebKit::Platform::current()->clipboard()->readImage(WebKit::WebClipboard::BufferStandard));
RefPtr<RawData> rawData = RawData::create();
rawData->mutableData()->append(data->data(), data->size());
OwnPtr<BlobData> blobData = BlobData::create();
blobData->appendData(rawData, 0, -1);
blobData->setContentType(mimeTypeImagePng);
return Blob::create(blobData.release(), data->size());
}
return 0;
}
String ChromiumDataObjectItem::internalGetAsString() const
{
ASSERT(m_kind == DataTransferItem::kindString);
if (m_source == InternalSource)
return m_data;
ASSERT(m_source == PasteboardSource);
String data;
// This is ugly but there's no real alternative.
if (m_type == mimeTypeTextPlain)
data = WebKit::Platform::current()->clipboard()->readPlainText(currentPasteboardBuffer());
else if (m_type == mimeTypeTextHTML) {
WebKit::WebURL ignoredSourceURL;
unsigned ignored;
data = WebKit::Platform::current()->clipboard()->readHTML(currentPasteboardBuffer(), &ignoredSourceURL, &ignored, &ignored);
} else
data = WebKit::Platform::current()->clipboard()->readCustomData(currentPasteboardBuffer(), m_type);
return WebKit::Platform::current()->clipboard()->sequenceNumber(currentPasteboardBuffer()) == m_sequenceNumber ? data : String();
}
bool ChromiumDataObjectItem::isFilename() const
{
// FIXME: https://bugs.webkit.org/show_bug.cgi?id=81261: When we properly support File dragout,
// we'll need to make sure this works as expected for DragDataChromium.
return m_kind == DataTransferItem::kindFile && m_file;
}
} // namespace WebCore