blob: f7c420f1a285cd2f4af99a4af04f1a79a55acbd7 [file] [log] [blame]
/*
* Copyright (C) 2007 Apple Inc. All rights reserved.
* Copyright (C) 2006, 2007 Apple Inc. All rights reserved.
* Copyright (C) 2007 Trolltech ASA
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``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 APPLE COMPUTER, INC. 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 "ClipboardQt.h"
#include "CachedImage.h"
#include "CSSHelper.h"
#include "DeprecatedString.h"
#include "Document.h"
#include "Element.h"
#include "Frame.h"
#include "HTMLNames.h"
#include "Image.h"
#include "IntPoint.h"
#include "KURL.h"
#include "markup.h"
#include "PlatformString.h"
#include "Range.h"
#include "RenderImage.h"
#include "StringHash.h"
#include <QList>
#include <QMimeData>
#include <QStringList>
#include <QUrl>
#include <QApplication>
#include <QClipboard>
#include <qdebug.h>
#define methodDebug() qDebug("ClipboardQt: %s", __FUNCTION__)
namespace WebCore {
ClipboardQt::ClipboardQt(ClipboardAccessPolicy policy, const QMimeData* readableClipboard)
: Clipboard(policy, true)
, m_readableData(readableClipboard)
, m_writableData(0)
{
Q_ASSERT(policy == ClipboardReadable || policy == ClipboardTypesReadable);
}
ClipboardQt::ClipboardQt(ClipboardAccessPolicy policy, bool forDragging)
: Clipboard(policy, forDragging)
, m_readableData(0)
, m_writableData(0)
{
Q_ASSERT(policy == ClipboardReadable || policy == ClipboardWritable || policy == ClipboardNumb);
if (policy != ClipboardWritable) {
Q_ASSERT(!forDragging);
m_readableData = QApplication::clipboard()->mimeData();
}
}
ClipboardQt::~ClipboardQt()
{
if (m_writableData && !isForDragging())
m_writableData = 0;
else
delete m_writableData;
m_readableData = 0;
}
void ClipboardQt::clearData(const String& type)
{
if (policy() != ClipboardWritable)
return;
if (m_writableData) {
#if QT_VERSION >= 0x040400
m_writableData->removeFormat(type);
#else
const QString toClearType = type;
QMap<QString, QByteArray> formats;
foreach (QString format, m_writableData->formats()) {
if (format != toClearType)
formats[format] = m_writableData->data(format);
}
m_writableData->clear();
QMap<QString, QByteArray>::const_iterator it, end = formats.constEnd();
for (it = formats.begin(); it != end; ++it)
m_writableData->setData(it.key(), it.value());
#endif
if (m_writableData->formats().isEmpty()) {
if (isForDragging())
delete m_writableData;
m_writableData = 0;
}
}
if (!isForDragging())
QApplication::clipboard()->setMimeData(m_writableData);
}
void ClipboardQt::clearAllData()
{
if (policy() != ClipboardWritable)
return;
if (!isForDragging())
QApplication::clipboard()->setMimeData(0);
else
delete m_writableData;
m_writableData = 0;
}
String ClipboardQt::getData(const String& type, bool& success) const
{
if (policy() != ClipboardReadable) {
success = false;
return String();
}
ASSERT(m_readableData);
QByteArray data = m_readableData->data(QString(type));
success = !data.isEmpty();
return String(data.data(), data.size());
}
bool ClipboardQt::setData(const String& type, const String& data)
{
if (policy() != ClipboardWritable)
return false;
if (!m_writableData)
m_writableData = new QMimeData;
QByteArray array(reinterpret_cast<const char*>(data.characters()),
data.length()*2);
m_writableData->setData(QString(type), array);
if (!isForDragging())
QApplication::clipboard()->setMimeData(m_writableData);
return true;
}
// extensions beyond IE's API
HashSet<String> ClipboardQt::types() const
{
if (policy() != ClipboardReadable && policy() != ClipboardTypesReadable)
return HashSet<String>();
ASSERT(m_readableData);
HashSet<String> result;
QStringList formats = m_readableData->formats();
for (int i = 0; i < formats.count(); ++i)
result.add(formats.at(i));
return result;
}
void ClipboardQt::setDragImage(CachedImage* image, const IntPoint& point)
{
setDragImage(image, 0, point);
}
void ClipboardQt::setDragImageElement(Node* node, const IntPoint& point)
{
setDragImage(0, node, point);
}
void ClipboardQt::setDragImage(CachedImage* image, Node *node, const IntPoint &loc)
{
if (policy() != ClipboardImageWritable && policy() != ClipboardWritable)
return;
if (m_dragImage)
m_dragImage->deref(this);
m_dragImage = image;
if (m_dragImage)
m_dragImage->ref(this);
m_dragLoc = loc;
m_dragImageElement = node;
}
DragImageRef ClipboardQt::createDragImage(IntPoint& dragLoc) const
{
if (!m_dragImage)
return 0;
dragLoc = m_dragLoc;
return m_dragImage->image()->getPixmap();
}
static CachedImage* getCachedImage(Element* element)
{
// Attempt to pull CachedImage from element
ASSERT(element);
RenderObject* renderer = element->renderer();
if (!renderer || !renderer->isImage())
return 0;
RenderImage* image = static_cast<RenderImage*>(renderer);
if (image->cachedImage() && !image->cachedImage()->errorOccurred())
return image->cachedImage();
return 0;
}
void ClipboardQt::declareAndWriteDragImage(Element* element, const KURL& url, const String& title, Frame* frame)
{
ASSERT(frame);
Q_UNUSED(url);
Q_UNUSED(title);
//WebCore::writeURL(m_writableDataObject.get(), url, title, true, false);
if (!m_writableData)
m_writableData = new QMimeData;
CachedImage* cachedImage = getCachedImage(element);
if (!cachedImage || !cachedImage->image() || !cachedImage->isLoaded())
return;
QPixmap *pixmap = cachedImage->image()->getPixmap();
if (pixmap)
m_writableData->setImageData(pixmap);
AtomicString imageURL = element->getAttribute(HTMLNames::srcAttr);
if (imageURL.isEmpty())
return;
String fullURL = frame->document()->completeURL(parseURL(imageURL));
if (fullURL.isEmpty())
return;
QList<QUrl> urls;
urls.append(QUrl(fullURL));
m_writableData->setUrls(urls);
if (!isForDragging())
QApplication::clipboard()->setMimeData(m_writableData);
}
void ClipboardQt::writeURL(const KURL& url, const String&, Frame* frame)
{
ASSERT(frame);
QList<QUrl> urls;
urls.append(QUrl(frame->document()->completeURL(url.deprecatedString())));
if (!m_writableData)
m_writableData = new QMimeData;
m_writableData->setUrls(urls);
if (!isForDragging())
QApplication::clipboard()->setMimeData(m_writableData);
}
void ClipboardQt::writeRange(Range* range, Frame* frame)
{
ASSERT(range);
ASSERT(frame);
if (!m_writableData)
m_writableData = new QMimeData;
QString text = frame->selectedText();
text.replace(QChar(0xa0), QLatin1Char(' '));
m_writableData->setText(text);
m_writableData->setHtml(createMarkup(range, 0, AnnotateForInterchange));
if (!isForDragging())
QApplication::clipboard()->setMimeData(m_writableData);
}
bool ClipboardQt::hasData()
{
const QMimeData *data = m_readableData ? m_readableData : m_writableData;
if (!data)
return false;
return data->formats().count() > 0;
}
}