katie/src/gui/kernel/qclipboard_x11.cpp
Ivailo Monev 5b459cf0ec update copyright and add script for doing so
Signed-off-by: Ivailo Monev <xakepa10@laimg.moc>
2019-12-29 23:21:34 +00:00

1422 lines
46 KiB
C++

/****************************************************************************
**
** Copyright (C) 2015 The Qt Company Ltd.
** Copyright (C) 2016-2020 Ivailo Monev
**
** This file is part of the QtGui module of the Katie Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL$
**
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** As a special exception, The Qt Company gives you certain additional
** rights. These rights are described in The Qt Company LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3.0 as published by the Free Software
** Foundation and appearing in the file LICENSE.GPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU General Public License version 3.0 requirements will be
** met: http://www.gnu.org/copyleft/gpl.html.
**
** $QT_END_LICENSE$
**
****************************************************************************/
// #define QCLIPBOARD_DEBUG
// #define QCLIPBOARD_DEBUG_VERBOSE
#ifdef QCLIPBOARD_DEBUG
# define DEBUG qDebug
#else
# define DEBUG if (false) qDebug
#endif
#ifdef QCLIPBOARD_DEBUG_VERBOSE
# define VDEBUG qDebug
#else
# define VDEBUG if (false) qDebug
#endif
#include "qplatformdefs.h"
#include "qclipboard.h"
#ifndef QT_NO_CLIPBOARD
#include "qabstracteventdispatcher.h"
#include "qapplication.h"
#include "qdesktopwidget.h"
#include "qbitmap.h"
#include "qiodevice.h"
#include "qbuffer.h"
#include "qtextcodec.h"
#include "qlist.h"
#include "qmap.h"
#include "qapplication_p.h"
#include "qevent.h"
#include "qt_x11_p.h"
#include "qx11info_x11.h"
#include "qimagewriter.h"
#include "qelapsedtimer.h"
#include "qvariant.h"
#include "qdnd_p.h"
#include "qwidget_p.h"
#include "qevent_p.h"
#include "qthread.h"
QT_BEGIN_NAMESPACE
/*****************************************************************************
Internal QClipboard functions for X11.
*****************************************************************************/
static int clipboard_timeout = 5000; // 5s timeout on clipboard operations
static QWidget * owner = 0;
static QWidget *requestor = 0;
static bool timer_event_clear = false;
static int timer_id = 0;
static int pending_timer_id = 0;
static bool pending_clipboard_changed = false;
static bool pending_selection_changed = false;
class QClipboardWatcher; // forward decl
static QClipboardWatcher *selection_watcher = 0;
static QClipboardWatcher *clipboard_watcher = 0;
static void cleanup()
{
delete owner;
delete requestor;
owner = 0;
requestor = 0;
}
static
void setupOwner()
{
if (owner)
return;
owner = new QWidget(0);
owner->setObjectName(QLatin1String("internal clipboard owner"));
owner->createWinId();
requestor = new QWidget(0);
requestor->createWinId();
requestor->setObjectName(QLatin1String("internal clipboard requestor"));
// We don't need this internal widgets to appear in QApplication::topLevelWidgets()
if (QWidgetPrivate::allWidgets) {
QWidgetPrivate::allWidgets->remove(owner);
QWidgetPrivate::allWidgets->remove(requestor);
}
qAddPostRoutine(cleanup);
}
class QClipboardWatcher : public QInternalMimeData {
public:
QClipboardWatcher(QClipboard::Mode mode);
~QClipboardWatcher();
bool empty() const;
virtual bool hasFormat_sys(const QString &mimetype) const;
virtual QStringList formats_sys() const;
QVariant retrieveData_sys(const QString &mimetype, QVariant::Type type) const;
QByteArray getDataInFormat(Atom fmtatom) const;
Atom atom;
mutable QStringList formatList;
mutable QByteArray format_atoms;
};
class QClipboardData
{
private:
QMimeData *&mimeDataRef() const
{
if(mode == QClipboard::Selection)
return selectionData;
return clipboardData;
}
public:
QClipboardData(QClipboard::Mode mode);
~QClipboardData();
void setSource(QMimeData* s)
{
if ((mode == QClipboard::Selection && selectionData == s)
|| clipboardData == s) {
return;
}
if (selectionData != clipboardData) {
delete mimeDataRef();
}
mimeDataRef() = s;
}
QMimeData *source() const
{
return mimeDataRef();
}
void clear()
{
timestamp = CurrentTime;
if (selectionData == clipboardData) {
mimeDataRef() = 0;
} else {
QMimeData *&src = mimeDataRef();
delete src;
src = 0;
}
}
static QMimeData *selectionData;
static QMimeData *clipboardData;
Time timestamp;
QClipboard::Mode mode;
};
QMimeData *QClipboardData::selectionData = 0;
QMimeData *QClipboardData::clipboardData = 0;
QClipboardData::QClipboardData(QClipboard::Mode clipboardMode)
{
timestamp = CurrentTime;
mode = clipboardMode;
}
QClipboardData::~QClipboardData()
{ clear(); }
static QClipboardData *internalCbData = 0;
static QClipboardData *internalSelData = 0;
static void cleanupClipboardData()
{
delete internalCbData;
internalCbData = 0;
}
static QClipboardData *clipboardData()
{
if (internalCbData == 0) {
internalCbData = new QClipboardData(QClipboard::Clipboard);
qAddPostRoutine(cleanupClipboardData);
}
return internalCbData;
}
static void cleanupSelectionData()
{
delete internalSelData;
internalSelData = 0;
}
static QClipboardData *selectionData()
{
if (internalSelData == 0) {
internalSelData = new QClipboardData(QClipboard::Selection);
qAddPostRoutine(cleanupSelectionData);
}
return internalSelData;
}
class QClipboardINCRTransaction
{
public:
QClipboardINCRTransaction(Window w, Atom p, Atom t, int f, QByteArray d, unsigned int i);
~QClipboardINCRTransaction(void);
int x11Event(XEvent *event);
Window window;
Atom property, target;
int format;
QByteArray data;
unsigned int increment;
unsigned int offset;
};
typedef QMap<Window,QClipboardINCRTransaction*> TransactionMap;
static TransactionMap *transactions = 0;
static QApplication::EventFilter prev_event_filter = 0;
static int incr_timer_id = 0;
static bool qt_x11_incr_event_filter(void *message, long *result)
{
XEvent *event = reinterpret_cast<XEvent *>(message);
TransactionMap::Iterator it = transactions->find(event->xany.window);
if (it != transactions->end()) {
if ((*it)->x11Event(event) != 0)
return true;
}
if (prev_event_filter)
return prev_event_filter(event, result);
return false;
}
QClipboardINCRTransaction::QClipboardINCRTransaction(Window w, Atom p, Atom t, int f,
QByteArray d, unsigned int i)
: window(w), property(p), target(t), format(f), data(d), increment(i), offset(0u)
{
DEBUG("QClipboard: sending %d bytes (INCR transaction %p)", d.size(), this);
XSelectInput(qt_x11Data->display, window, PropertyChangeMask);
if (! transactions) {
VDEBUG("QClipboard: created INCR transaction map");
transactions = new TransactionMap;
prev_event_filter = qApp->setEventFilter(qt_x11_incr_event_filter);
incr_timer_id = QApplication::clipboard()->startTimer(clipboard_timeout);
}
transactions->insert(window, this);
}
QClipboardINCRTransaction::~QClipboardINCRTransaction(void)
{
VDEBUG("QClipboard: destroyed INCR transacton %p", this);
XSelectInput(qt_x11Data->display, window, NoEventMask);
transactions->remove(window);
if (transactions->isEmpty()) {
VDEBUG("QClipboard: no more INCR transactions");
delete transactions;
transactions = 0;
(void)qApp->setEventFilter(prev_event_filter);
if (incr_timer_id != 0) {
QApplication::clipboard()->killTimer(incr_timer_id);
incr_timer_id = 0;
}
}
}
int QClipboardINCRTransaction::x11Event(XEvent *event)
{
if (event->type != PropertyNotify
|| (event->xproperty.state != PropertyDelete
|| event->xproperty.atom != property))
return 0;
// restart the INCR timer
if (incr_timer_id) QApplication::clipboard()->killTimer(incr_timer_id);
incr_timer_id = QApplication::clipboard()->startTimer(clipboard_timeout);
unsigned int bytes_left = data.size() - offset;
if (bytes_left > 0) {
unsigned int xfer = qMin(increment, bytes_left);
VDEBUG("QClipboard: sending %d bytes, %d remaining (INCR transaction %p)",
xfer, bytes_left - xfer, this);
XChangeProperty(qt_x11Data->display, window, property, target, format,
PropModeReplace, (uchar *) data.data() + offset, xfer);
offset += xfer;
} else {
// INCR transaction finished...
XChangeProperty(qt_x11Data->display, window, property, target, format,
PropModeReplace, (uchar *) data.data(), 0);
delete this;
}
return 1;
}
/*****************************************************************************
QClipboard member functions for X11.
*****************************************************************************/
struct qt_init_timestamp_data
{
Time timestamp;
};
#if defined(Q_C_CALLBACKS)
extern "C" {
#endif
static Bool qt_init_timestamp_scanner(Display*, XEvent *event, XPointer arg)
{
qt_init_timestamp_data *data =
reinterpret_cast<qt_init_timestamp_data*>(arg);
switch(event->type)
{
case XButtonPress:
case XButtonRelease:
data->timestamp = event->xbutton.time;
break;
case MotionNotify:
data->timestamp = event->xmotion.time;
break;
case XKeyPress:
case XKeyRelease:
data->timestamp = event->xkey.time;
break;
case PropertyNotify:
data->timestamp = event->xproperty.time;
break;
case EnterNotify:
case LeaveNotify:
data->timestamp = event->xcrossing.time;
break;
case SelectionClear:
data->timestamp = event->xselectionclear.time;
break;
default:
break;
}
#ifndef QT_NO_XFIXES
if (qt_x11Data->use_xfixes && event->type == (qt_x11Data->xfixes_eventbase + XFixesSelectionNotify)) {
XFixesSelectionNotifyEvent *req =
reinterpret_cast<XFixesSelectionNotifyEvent *>(event);
data->timestamp = req->selection_timestamp;
}
#endif
return false;
}
#if defined(Q_C_CALLBACKS)
}
#endif
QClipboard::QClipboard(QObject *parent)
: QObject(parent)
{
// create desktop widget since we need it to get PropertyNotify or
// XFixesSelectionNotify events when someone changes the
// clipboard.
(void)QApplication::desktop();
#ifndef QT_NO_XFIXES
if (qt_x11Data->use_xfixes) {
const unsigned long eventMask =
XFixesSetSelectionOwnerNotifyMask | XFixesSelectionWindowDestroyNotifyMask | XFixesSelectionClientCloseNotifyMask;
for (int i = 0; i < qt_x11Data->screenCount; ++i) {
XFixesSelectSelectionInput(qt_x11Data->display, QX11Info::appRootWindow(i),
XA_PRIMARY, eventMask);
XFixesSelectSelectionInput(qt_x11Data->display, QX11Info::appRootWindow(i),
ATOM(CLIPBOARD), eventMask);
}
}
#endif // QT_NO_XFIXES
if (qt_x11Data->time == CurrentTime) {
// send a dummy event to myself to get the timestamp from X11.
qt_init_timestamp_data data;
data.timestamp = CurrentTime;
XEvent ev;
XCheckIfEvent(qt_x11Data->display, &ev, &qt_init_timestamp_scanner, (XPointer)&data);
if (data.timestamp == CurrentTime) {
setupOwner();
// We need this value just for completeness, we don't use it.
long dummy = 0;
Window ownerId = owner->internalWinId();
XChangeProperty(qt_x11Data->display, ownerId,
ATOM(CLIP_TEMPORARY), XA_INTEGER, 32,
PropModeReplace, (uchar*)&dummy, 1);
XWindowEvent(qt_x11Data->display, ownerId, PropertyChangeMask, &ev);
data.timestamp = ev.xproperty.time;
XDeleteProperty(qt_x11Data->display, ownerId, ATOM(CLIP_TEMPORARY));
}
qt_x11Data->time = data.timestamp;
}
}
void QClipboard::clear(Mode mode)
{
setMimeData(0, mode);
}
bool QClipboard::supportsMode(Mode mode) const
{
return (mode == Clipboard || mode == Selection);
}
bool QClipboard::ownsMode(Mode mode) const
{
if (mode == Clipboard)
return clipboardData()->timestamp != CurrentTime;
else if(mode == Selection)
return selectionData()->timestamp != CurrentTime;
else
return false;
}
static Bool checkForClipboardEvents(Display *, XEvent *e, XPointer)
{
return ((e->type == SelectionRequest && (e->xselectionrequest.selection == XA_PRIMARY
|| e->xselectionrequest.selection == ATOM(CLIPBOARD)))
|| (e->type == SelectionClear && (e->xselectionclear.selection == XA_PRIMARY
|| e->xselectionclear.selection == ATOM(CLIPBOARD))));
}
bool QX11Data::clipboardWaitForEvent(Window win, int type, XEvent *event, int timeout, bool checkManager)
{
QElapsedTimer started;
started.start();
QElapsedTimer now = started;
do {
if (XCheckTypedWindowEvent(qt_x11Data->display,win,type,event))
return true;
if (checkManager && XGetSelectionOwner(qt_x11Data->display, ATOM(CLIPBOARD_MANAGER)) == XNone)
return false;
// process other clipboard events, since someone is probably requesting data from us
XEvent e;
// Pass the event through the event dispatcher filter so that applications
// which install an event filter on the dispatcher get to handle it first.
if (XCheckIfEvent(qt_x11Data->display, &e, checkForClipboardEvents, 0) &&
!QAbstractEventDispatcher::instance()->filterEvent(&e))
qApp->x11ProcessEvent(&e);
now.start();
XFlush(qt_x11Data->display);
// sleep 50 ms, so we don't use up CPU cycles all the time.
QThread::usleep(50000);
} while (started.msecsTo(now) < timeout);
return false;
}
static inline int maxSelectionIncr(Display *dpy)
{ return XMaxRequestSize(dpy) > 65536 ? 65536*4 : XMaxRequestSize(dpy)*4 - 100; }
bool QX11Data::clipboardReadProperty(Window win, Atom property, bool deleteProperty,
QByteArray *buffer, int *size, Atom *type, int *format)
{
int maxsize = maxSelectionIncr(display);
ulong bytes_left; // bytes_after
ulong length; // nitems
uchar *data;
Atom dummy_type;
int dummy_format;
int r;
if (!type) // allow null args
type = &dummy_type;
if (!format)
format = &dummy_format;
// Don't read anything, just get the size of the property data
r = XGetWindowProperty(display, win, property, 0, 0, False,
AnyPropertyType, type, format,
&length, &bytes_left, &data);
if (r != Success || (type && *type == XNone)) {
buffer->resize(0);
return false;
}
XFree((char*)data);
int offset = 0, buffer_offset = 0, format_inc = 1, proplen = bytes_left;
VDEBUG("QClipboard: read_property(): initial property length: %d", proplen);
switch (*format) {
case 8:
default:
format_inc = sizeof(char) / 1;
break;
case 16:
format_inc = sizeof(short) / 2;
proplen *= sizeof(short) / 2;
break;
case 32:
format_inc = sizeof(long) / 4;
proplen *= sizeof(long) / 4;
break;
}
int newSize = proplen;
buffer->resize(newSize);
bool ok = (buffer->size() == newSize);
VDEBUG("QClipboard: read_property(): buffer resized to %d", buffer->size());
if (ok && newSize) {
// could allocate buffer
while (bytes_left) {
// more to read...
r = XGetWindowProperty(display, win, property, offset, maxsize/4,
False, AnyPropertyType, type, format,
&length, &bytes_left, &data);
if (r != Success || (type && *type == XNone))
break;
offset += length / (32 / *format);
length *= format_inc * (*format) / 8;
// Here we check if we get a buffer overflow and tries to
// recover -- this shouldn't normally happen, but it doesn't
// hurt to be defensive
if ((int)(buffer_offset + length) > buffer->size()) {
length = buffer->size() - buffer_offset;
// escape loop
bytes_left = 0;
}
memcpy(buffer->data() + buffer_offset, data, length);
buffer_offset += length;
XFree((char*)data);
}
if (*format == 8 && *type == ATOM(COMPOUND_TEXT)) {
// convert COMPOUND_TEXT to a multibyte string
XTextProperty textprop;
textprop.encoding = *type;
textprop.format = *format;
textprop.nitems = buffer_offset;
textprop.value = (unsigned char *) buffer->data();
char **list_ret = 0;
int count;
if (XmbTextPropertyToTextList(display, &textprop, &list_ret,
&count) == Success && count && list_ret) {
offset = buffer_offset = strlen(list_ret[0]);
buffer->resize(offset);
memcpy(buffer->data(), list_ret[0], offset);
}
if (list_ret) XFreeStringList(list_ret);
}
}
// correct size, not 0-term.
if (size)
*size = buffer_offset;
VDEBUG("QClipboard: read_property(): buffer size %d, buffer offset %d, offset %d",
buffer->size(), buffer_offset, offset);
if (deleteProperty)
XDeleteProperty(display, win, property);
XFlush(display);
return ok;
}
QByteArray QX11Data::clipboardReadIncrementalProperty(Window win, Atom property, int nbytes, bool nullterm)
{
XEvent event;
QByteArray buf;
QByteArray tmp_buf;
bool alloc_error = false;
int length;
int offset = 0;
if (nbytes > 0) {
// Reserve buffer + zero-terminator (for text data)
// We want to complete the INCR transfer even if we cannot
// allocate more memory
buf.resize(nbytes+1);
alloc_error = buf.size() != nbytes+1;
}
for (;;) {
XFlush(display);
if (!clipboardWaitForEvent(win,PropertyNotify,&event,clipboard_timeout))
break;
if (event.xproperty.atom != property ||
event.xproperty.state != PropertyNewValue)
continue;
if (qt_x11Data->clipboardReadProperty(win, property, true, &tmp_buf, &length, 0, 0)) {
if (length == 0) { // no more data, we're done
if (nullterm) {
buf.resize(offset+1);
buf[offset] = '\0';
} else {
buf.resize(offset);
}
return buf;
} else if (!alloc_error) {
if (offset+length > (int)buf.size()) {
buf.resize(offset+length+65535);
if (buf.size() != offset+length+65535) {
alloc_error = true;
length = buf.size() - offset;
}
}
memcpy(buf.data()+offset, tmp_buf.constData(), length);
tmp_buf.resize(0);
offset += length;
}
} else {
break;
}
}
// timed out ... create a new requestor window, otherwise the requestor
// could consider next request to be still part of this timed out request
delete requestor;
requestor = new QWidget(0);
requestor->setObjectName(QLatin1String("internal clipboard requestor"));
// We don't need this internal widget to appear in QApplication::topLevelWidgets()
if (QWidgetPrivate::allWidgets)
QWidgetPrivate::allWidgets->remove(requestor);
return QByteArray();
}
static Atom send_targets_selection(QClipboardData *d, Window window, Atom property)
{
QVector<Atom> types;
QStringList formats = QInternalMimeData::formatsHelper(d->source());
for (int i = 0; i < formats.size(); ++i) {
QList<Atom> atoms = qt_x11Data->xdndMimeAtomsForFormat(formats.at(i));
for (int j = 0; j < atoms.size(); ++j) {
if (!types.contains(atoms.at(j)))
types.append(atoms.at(j));
}
}
types.append(ATOM(TARGETS));
types.append(ATOM(MULTIPLE));
types.append(ATOM(TIMESTAMP));
types.append(ATOM(SAVE_TARGETS));
XChangeProperty(qt_x11Data->display, window, property, XA_ATOM, 32,
PropModeReplace, (uchar *) types.data(), types.size());
return property;
}
static Atom send_selection(QClipboardData *d, Atom target, Window window, Atom property)
{
Atom atomFormat = target;
int dataFormat = 0;
QByteArray data;
QByteArray fmt = qt_x11Data->xdndAtomToString(target);
if (fmt.isEmpty()) { // Not a MIME type we have
DEBUG("QClipboard: send_selection(): converting to type '%s' is not supported", fmt.data());
return XNone;
}
DEBUG("QClipboard: send_selection(): converting to type '%s'", fmt.data());
if (qt_x11Data->xdndMimeDataForAtom(target, d->source(), &data, &atomFormat, &dataFormat)) {
VDEBUG("QClipboard: send_selection():\n"
" property type %lx\n"
" property name '%s'\n"
" format %d\n"
" %d bytes\n",
target, qt_x11Data->xdndMimeAtomToString(atomFormat).toLatin1().data(), dataFormat, data.size());
// don't allow INCR transfers when using MULTIPLE or to
// Motif clients (since Motif doesn't support INCR)
static Atom motif_clip_temporary = ATOM(CLIP_TEMPORARY);
bool allow_incr = property != motif_clip_temporary;
// X_ChangeProperty protocol request is 24 bytes
const int increment = (XMaxRequestSize(qt_x11Data->display) * 4) - 24;
if (data.size() > increment && allow_incr) {
long bytes = data.size();
XChangeProperty(qt_x11Data->display, window, property,
ATOM(INCR), 32, PropModeReplace, (uchar *) &bytes, 1);
(void)new QClipboardINCRTransaction(window, property, atomFormat, dataFormat, data, increment);
return property;
}
// make sure we can perform the XChangeProperty in a single request
if (data.size() > increment)
return XNone; // ### perhaps use several XChangeProperty calls w/ PropModeAppend?
int dataSize = data.size() / (dataFormat / 8);
// use a single request to transfer data
XChangeProperty(qt_x11Data->display, window, property, atomFormat,
dataFormat, PropModeReplace, (uchar *) data.data(),
dataSize);
}
return property;
}
/*! \internal
Internal optimization for Windows.
*/
void QClipboard::connectNotify(const char *)
{ }
bool QClipboard::event(QEvent *e)
{
if (e->type() == QEvent::Timer) {
QTimerEvent *te = static_cast<QTimerEvent *>(e);
if (te->timerId() == timer_id) {
killTimer(timer_id);
timer_id = 0;
timer_event_clear = true;
if (selection_watcher) // clear selection
selectionData()->clear();
if (clipboard_watcher) // clear clipboard
clipboardData()->clear();
timer_event_clear = false;
return true;
} else if (te->timerId() == pending_timer_id) {
// I hate klipper
killTimer(pending_timer_id);
pending_timer_id = 0;
if (pending_clipboard_changed) {
pending_clipboard_changed = false;
clipboardData()->clear();
emitChanged(QClipboard::Clipboard);
}
if (pending_selection_changed) {
pending_selection_changed = false;
selectionData()->clear();
emitChanged(QClipboard::Selection);
}
return true;
} else if (te->timerId() == incr_timer_id) {
killTimer(incr_timer_id);
incr_timer_id = 0;
/*
no INCR activity has happened for 'clipboard_timeout'
milliseconds... we assume that all unfinished transactions have
timed out and remove everything from the transaction map
*/
qWarning("QClipboard: Timed out while sending data");
while (transactions)
delete *transactions->begin();
return true;
} else {
return QObject::event(e);
}
} else if (e->type() != QEvent::Clipboard) {
return QObject::event(e);
}
XEvent *xevent = static_cast<QClipboardEvent*>(e)->m_event;
Display *dpy = qt_x11Data->display;
if (!xevent) {
// That means application exits and we need to give clipboard
// content to the clipboard manager.
// First we check if there is a clipboard manager.
if (XGetSelectionOwner(qt_x11Data->display, ATOM(CLIPBOARD_MANAGER)) == XNone
|| !owner)
return true;
Window ownerId = owner->internalWinId();
Q_ASSERT(ownerId);
// we delete the property so the manager saves all TARGETS.
XDeleteProperty(qt_x11Data->display, ownerId, ATOM(_QT_SELECTION));
XConvertSelection(qt_x11Data->display, ATOM(CLIPBOARD_MANAGER), ATOM(SAVE_TARGETS),
ATOM(_QT_SELECTION), ownerId, qt_x11Data->time);
XSync(dpy, false);
XEvent waitevent;
// waiting until the clipboard manager fetches the content.
if (!qt_x11Data->clipboardWaitForEvent(ownerId, SelectionNotify, &waitevent, 10000, true)) {
qWarning("QClipboard: Unable to receive an event from the "
"clipboard manager in a reasonable time");
}
return true;
}
switch (xevent->type) {
case SelectionClear:
// new selection owner
if (xevent->xselectionclear.selection == XA_PRIMARY) {
QClipboardData *d = selectionData();
// ignore the event if it was generated before we gained selection ownership
if (d->timestamp != CurrentTime && xevent->xselectionclear.time <= d->timestamp)
break;
DEBUG("QClipboard: new selection owner 0x%lx at time %lx (ours %lx)",
XGetSelectionOwner(dpy, XA_PRIMARY),
xevent->xselectionclear.time, d->timestamp);
d->clear();
emitChanged(QClipboard::Selection);
} else if (xevent->xselectionclear.selection == ATOM(CLIPBOARD)) {
QClipboardData *d = clipboardData();
// ignore the event if it was generated before we gained selection ownership
if (d->timestamp != CurrentTime && xevent->xselectionclear.time <= d->timestamp)
break;
DEBUG("QClipboard: new clipboard owner 0x%lx at time %lx (%lx)",
XGetSelectionOwner(dpy, ATOM(CLIPBOARD)),
xevent->xselectionclear.time, d->timestamp);
d->clear();
emitChanged(QClipboard::Clipboard);
} else {
qWarning("QClipboard: Unknown SelectionClear event received");
return false;
}
break;
case SelectionNotify:
/*
Something has delivered data to us, but this was not caught
by QClipboardWatcher::getDataInFormat()
Just skip the event to prevent Bad Things (tm) from
happening later on...
*/
break;
case SelectionRequest:
{
// someone wants our data
XSelectionRequestEvent *req = &xevent->xselectionrequest;
if (requestor && req->requestor == requestor->internalWinId())
break;
XEvent event;
event.xselection.type = SelectionNotify;
event.xselection.display = req->display;
event.xselection.requestor = req->requestor;
event.xselection.selection = req->selection;
event.xselection.target = req->target;
event.xselection.property = XNone;
event.xselection.time = req->time;
DEBUG("QClipboard: SelectionRequest from %lx\n"
" selection 0x%lx (%s) target 0x%lx (%s)",
req->requestor,
req->selection,
qt_x11Data->xdndAtomToString(req->selection).data(),
req->target,
qt_x11Data->xdndAtomToString(req->target).data());
QClipboardData *d;
if (req->selection == XA_PRIMARY) {
d = selectionData();
} else if (req->selection == ATOM(CLIPBOARD)) {
d = clipboardData();
} else {
qWarning("QClipboard: Unknown selection '%lx'", req->selection);
XSendEvent(dpy, req->requestor, False, NoEventMask, &event);
break;
}
if (! d->source()) {
qWarning("QClipboard: Cannot transfer data, no data available");
XSendEvent(dpy, req->requestor, False, NoEventMask, &event);
break;
}
DEBUG("QClipboard: SelectionRequest at time %lx (ours %lx)",
req->time, d->timestamp);
if (d->timestamp == CurrentTime // we don't own the selection anymore
|| (req->time != CurrentTime && req->time < d->timestamp)) {
DEBUG("QClipboard: SelectionRequest too old");
XSendEvent(dpy, req->requestor, False, NoEventMask, &event);
break;
}
Atom xa_targets = ATOM(TARGETS);
Atom xa_multiple = ATOM(MULTIPLE);
Atom xa_timestamp = ATOM(TIMESTAMP);
struct AtomPair { Atom target; Atom property; } *multi = 0;
Atom multi_type = XNone;
int multi_format = 0;
int nmulti = 0;
int imulti = -1;
bool multi_writeback = false;
if (req->target == xa_multiple) {
QByteArray multi_data;
if (req->property == XNone
|| !qt_x11Data->clipboardReadProperty(req->requestor, req->property, false, &multi_data,
0, &multi_type, &multi_format)
|| multi_format != 32) {
// MULTIPLE property not formatted correctly
XSendEvent(dpy, req->requestor, False, NoEventMask, &event);
break;
}
nmulti = multi_data.size()/sizeof(*multi);
multi = new AtomPair[nmulti];
memcpy(multi,multi_data.data(),multi_data.size());
imulti = 0;
}
for (; imulti < nmulti; ++imulti) {
Atom target;
Atom property;
if (multi) {
target = multi[imulti].target;
property = multi[imulti].property;
} else {
target = req->target;
property = req->property;
if (property == XNone) // obsolete client
property = target;
}
Atom ret = XNone;
if (target == XNone || property == XNone) {
;
} else if (target == xa_timestamp) {
if (d->timestamp != CurrentTime) {
XChangeProperty(dpy, req->requestor, property, XA_INTEGER, 32,
PropModeReplace, (uchar *) &d->timestamp, 1);
ret = property;
} else {
qWarning("QClipboard: Invalid data timestamp");
}
} else if (target == xa_targets) {
ret = send_targets_selection(d, req->requestor, property);
} else {
ret = send_selection(d, target, req->requestor, property);
}
if (nmulti > 0) {
if (ret == XNone) {
multi[imulti].property = XNone;
multi_writeback = true;
}
} else {
event.xselection.property = ret;
break;
}
}
if (nmulti > 0) {
if (multi_writeback) {
// according to ICCCM 2.6.2 says to put None back
// into the original property on the requestor window
XChangeProperty(dpy, req->requestor, req->property, multi_type, 32,
PropModeReplace, (uchar *) multi, nmulti * 2);
}
delete [] multi;
event.xselection.property = req->property;
}
// send selection notify to requestor
XSendEvent(dpy, req->requestor, False, NoEventMask, &event);
DEBUG("QClipboard: SelectionNotify to 0x%lx\n"
" property 0x%lx (%s)",
req->requestor, event.xselection.property,
qt_x11Data->xdndAtomToString(event.xselection.property).data());
}
break;
}
return true;
}
QClipboardWatcher::QClipboardWatcher(QClipboard::Mode mode)
: QInternalMimeData()
{
switch (mode) {
case QClipboard::Selection:
atom = XA_PRIMARY;
break;
case QClipboard::Clipboard:
atom = ATOM(CLIPBOARD);
break;
default:
qWarning("QClipboardWatcher: Internal error: Unsupported clipboard mode");
break;
}
setupOwner();
}
QClipboardWatcher::~QClipboardWatcher()
{
if(selection_watcher == this)
selection_watcher = 0;
if(clipboard_watcher == this)
clipboard_watcher = 0;
}
bool QClipboardWatcher::empty() const
{
Display *dpy = qt_x11Data->display;
Window win = XGetSelectionOwner(dpy, atom);
if(win == requestor->internalWinId()) {
qWarning("QClipboardWatcher::empty: Internal error: Application owns the selection");
return true;
}
return win == XNone;
}
QStringList QClipboardWatcher::formats_sys() const
{
if (empty())
return QStringList();
if (!formatList.count()) {
// get the list of targets from the current clipboard owner - we do this
// once so that multiple calls to this function don't require multiple
// server round trips...
format_atoms = getDataInFormat(ATOM(TARGETS));
if (format_atoms.size() > 0) {
Atom *targets = (Atom *) format_atoms.data();
int size = format_atoms.size() / sizeof(Atom);
for (int i = 0; i < size; ++i) {
if (targets[i] == 0)
continue;
QStringList formatsForAtom = qt_x11Data->xdndMimeFormatsForAtom(targets[i]);
for (int j = 0; j < formatsForAtom.size(); ++j) {
if (!formatList.contains(formatsForAtom.at(j)))
formatList.append(formatsForAtom.at(j));
}
VDEBUG(" format: %s", qt_x11Data->xdndAtomToString(targets[i]).data());
VDEBUG(" data:\n%s\n", getDataInFormat(targets[i]).data());
}
DEBUG("QClipboardWatcher::format: %d formats available", formatList.count());
}
}
return formatList;
}
bool QClipboardWatcher::hasFormat_sys(const QString &format) const
{
QStringList list = formats();
return list.contains(format);
}
QVariant QClipboardWatcher::retrieveData_sys(const QString &fmt, QVariant::Type requestedType) const
{
if (fmt.isEmpty() || empty())
return QByteArray();
(void)formats(); // trigger update of format list
DEBUG("QClipboardWatcher::data: fetching format '%s'", fmt.toLatin1().data());
QList<Atom> atoms;
Atom *targets = (Atom *) format_atoms.data();
int size = format_atoms.size() / sizeof(Atom);
for (int i = 0; i < size; ++i)
atoms.append(targets[i]);
QByteArray encoding;
Atom fmtatom = qt_x11Data->xdndMimeAtomForFormat(fmt, requestedType, atoms, &encoding);
if (fmtatom == 0)
return QVariant();
return qt_x11Data->xdndMimeConvertToFormat(fmtatom, getDataInFormat(fmtatom), fmt, requestedType, encoding);
}
QByteArray QClipboardWatcher::getDataInFormat(Atom fmtatom) const
{
QByteArray buf;
Display *dpy = qt_x11Data->display;
requestor->createWinId();
Window win = requestor->internalWinId();
Q_ASSERT(requestor->testAttribute(Qt::WA_WState_Created));
DEBUG("QClipboardWatcher::getDataInFormat: selection '%s' format '%s'",
qt_x11Data->xdndAtomToString(atom).data(), qt_x11Data->xdndAtomToString(fmtatom).data());
XSelectInput(dpy, win, NoEventMask); // don't listen for any events
XDeleteProperty(dpy, win, ATOM(_QT_SELECTION));
XConvertSelection(dpy, atom, fmtatom, ATOM(_QT_SELECTION), win, qt_x11Data->time);
XSync(dpy, false);
VDEBUG("QClipboardWatcher::getDataInFormat: waiting for SelectionNotify event");
XEvent xevent;
if (!qt_x11Data->clipboardWaitForEvent(win,SelectionNotify,&xevent,clipboard_timeout) ||
xevent.xselection.property == XNone) {
DEBUG("QClipboardWatcher::getDataInFormat: format not available");
return buf;
}
VDEBUG("QClipboardWatcher::getDataInFormat: fetching data...");
Atom type;
XSelectInput(dpy, win, PropertyChangeMask);
if (qt_x11Data->clipboardReadProperty(win, ATOM(_QT_SELECTION), true, &buf, 0, &type, 0)) {
if (type == ATOM(INCR)) {
int nbytes = buf.size() >= 4 ? *((int*)buf.data()) : 0;
buf = qt_x11Data->clipboardReadIncrementalProperty(win, ATOM(_QT_SELECTION), nbytes, false);
}
}
XSelectInput(dpy, win, NoEventMask);
DEBUG("QClipboardWatcher::getDataInFormat: %d bytes received", buf.size());
return buf;
}
const QMimeData* QClipboard::mimeData(Mode mode) const
{
QClipboardData *d = 0;
switch (mode) {
case Selection:
d = selectionData();
break;
case Clipboard:
d = clipboardData();
break;
default:
qWarning("QClipboard::mimeData: unsupported mode '%d'", mode);
return 0;
}
if (! d->source() && ! timer_event_clear) {
if (mode == Selection) {
if (! selection_watcher)
selection_watcher = new QClipboardWatcher(mode);
d->setSource(selection_watcher);
} else {
if (! clipboard_watcher)
clipboard_watcher = new QClipboardWatcher(mode);
d->setSource(clipboard_watcher);
}
if (! timer_id) {
// start a zero timer - we will clear cached data when the timer
// times out, which will be the next time we hit the event loop...
// that way, the data is cached long enough for calls within a single
// loop/function, but the data doesn't linger around in case the selection
// changes
QClipboard *that = ((QClipboard *) this);
timer_id = that->startTimer(0);
}
}
return d->source();
}
void QClipboard::setMimeData(QMimeData* src, Mode mode)
{
Atom atom, sentinel_atom;
QClipboardData *d;
switch (mode) {
case Selection:
atom = XA_PRIMARY;
sentinel_atom = ATOM(_QT_SELECTION_SENTINEL);
d = selectionData();
break;
case Clipboard:
atom = ATOM(CLIPBOARD);
sentinel_atom = ATOM(_QT_CLIPBOARD_SENTINEL);
d = clipboardData();
break;
default:
qWarning("QClipboard::setMimeData: unsupported mode '%d'", mode);
return;
}
Display *dpy = qt_x11Data->display;
Window newOwner;
if (! src) { // no data, clear clipboard contents
newOwner = XNone;
d->clear();
} else {
setupOwner();
newOwner = owner->internalWinId();
d->setSource(src);
d->timestamp = qt_x11Data->time;
}
Window prevOwner = XGetSelectionOwner(dpy, atom);
// use qt_x11Data->time, since d->timestamp == CurrentTime when clearing
XSetSelectionOwner(dpy, atom, newOwner, qt_x11Data->time);
if (mode == Selection)
emitChanged(QClipboard::Selection);
else
emitChanged(QClipboard::Clipboard);
if (XGetSelectionOwner(dpy, atom) != newOwner) {
qWarning("QClipboard::setData: Cannot set X11 selection owner for %s",
qt_x11Data->xdndAtomToString(atom).data());
d->clear();
return;
}
// Signal to other Qt processes that the selection has changed
Window owners[2];
owners[0] = newOwner;
owners[1] = prevOwner;
XChangeProperty(dpy, QApplication::desktop()->screen(0)->internalWinId(),
sentinel_atom, XA_WINDOW, 32, PropModeReplace,
(unsigned char*)&owners, 2);
}
/*
Called by the main event loop in qapplication_x11.cpp when either
the _QT_SELECTION_SENTINEL property has been changed (i.e. when some
Qt process has performed QClipboard::setData()) or when Xfixes told
us that an other application changed the selection. If it returns
true, the QClipBoard dataChanged() signal should be emitted.
*/
bool qt_check_selection_sentinel()
{
bool doIt = true;
if (owner && !qt_x11Data->use_xfixes) {
/*
Since the X selection mechanism cannot give any signal when
the selection has changed, we emulate it (for Qt processes) here.
The notification should be ignored in case of either
a) This is the process that did setData (because setData()
then has already emitted dataChanged())
b) This is the process that owned the selection when dataChanged()
was called (because we have then received a SelectionClear event,
and have already emitted dataChanged() as a result of that)
*/
unsigned char *retval;
Atom actualType;
int actualFormat;
ulong nitems;
ulong bytesLeft;
if (XGetWindowProperty(qt_x11Data->display,
QApplication::desktop()->screen(0)->internalWinId(),
ATOM(_QT_SELECTION_SENTINEL), 0, 2, False, XA_WINDOW,
&actualType, &actualFormat, &nitems,
&bytesLeft, &retval) == Success) {
Window *owners = (Window *)retval;
if (actualType == XA_WINDOW && actualFormat == 32 && nitems == 2) {
Window win = owner->internalWinId();
if (owners[0] == win || owners[1] == win)
doIt = false;
}
XFree(owners);
}
}
if (doIt) {
selectionData()->clear();
}
return doIt;
}
bool qt_check_clipboard_sentinel()
{
bool doIt = true;
if (owner && !qt_x11Data->use_xfixes) {
unsigned char *retval;
Atom actualType;
int actualFormat;
unsigned long nitems, bytesLeft;
if (XGetWindowProperty(qt_x11Data->display,
QApplication::desktop()->screen(0)->internalWinId(),
ATOM(_QT_CLIPBOARD_SENTINEL), 0, 2, False, XA_WINDOW,
&actualType, &actualFormat, &nitems, &bytesLeft,
&retval) == Success) {
Window *owners = (Window *)retval;
if (actualType == XA_WINDOW && actualFormat == 32 && nitems == 2) {
Window win = owner->internalWinId();
if (owners[0] == win || owners[1] == win)
doIt = false;
}
XFree(owners);
}
}
if (doIt) {
clipboardData()->clear();
}
return doIt;
}
bool qt_xfixes_selection_changed(Window selectionOwner, Time timestamp)
{
QClipboardData *d = selectionData();
#ifdef QCLIPBOARD_DEBUG
DEBUG("qt_xfixes_selection_changed: owner = %u; selectionOwner = %u; internal timestamp = %u; external timestamp = %u",
(unsigned int)(owner ? (int)owner->internalWinId() : 0), (unsigned int)selectionOwner,
(unsigned int)(d ? d->timestamp : 0), (unsigned int)timestamp);
#endif
if (!owner || (selectionOwner && selectionOwner != owner->internalWinId()) ||
(!selectionOwner && (d->timestamp == CurrentTime || d->timestamp < timestamp)))
return qt_check_selection_sentinel();
return false;
}
bool qt_xfixes_clipboard_changed(Window clipboardOwner, Time timestamp)
{
QClipboardData *d = clipboardData();
#ifdef QCLIPBOARD_DEBUG
DEBUG("qt_xfixes_clipboard_changed: owner = %u; clipboardOwner = %u; internal timestamp = %u; external timestamp = %u",
(unsigned int)(owner ? (int)owner->internalWinId() : 0), (unsigned int)clipboardOwner,
(unsigned int)(d ? d->timestamp : 0), (unsigned int)timestamp);
#endif
if (!owner || (clipboardOwner && clipboardOwner != owner->internalWinId()) ||
(!clipboardOwner && (d->timestamp == CurrentTime || d->timestamp < timestamp)))
return qt_check_clipboard_sentinel();
return false;
}
QT_END_NAMESPACE
#endif // QT_NO_CLIPBOARD
#undef DEBUG
#undef VDEBUG