You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tqt3/src/kernel/qdnd_x11.cpp

1860 lines
51 KiB

/****************************************************************************
**
** XDND implementation for TQt. See http://www.cco.caltech.edu/~jafl/xdnd/
**
** Created : 980320
**
** Copyright (C) 1992-2008 Trolltech ASA. All rights reserved.
**
** This file is part of the kernel module of the TQt GUI Toolkit.
**
** This file may be used under the terms of the GNU General
** Public License versions 2.0 or 3.0 as published by the Free
** Software Foundation and appearing in the files LICENSE.GPL2
** and LICENSE.GPL3 included in the packaging of this file.
** Alternatively you may (at your option) use any later version
** of the GNU General Public License if such license has been
** publicly approved by Trolltech ASA (or its successors, if any)
** and the KDE Free TQt Foundation.
**
** Please review the following information to ensure GNU General
** Public Licensing requirements will be met:
** http://trolltech.com/products/qt/licenses/licensing/opensource/.
** If you are unsure which license is appropriate for your use, please
** review the following information:
** http://trolltech.com/products/qt/licenses/licensing/licensingoverview
** or contact the sales department at sales@trolltech.com.
**
** This file may be used under the terms of the Q Public License as
** defined by Trolltech ASA and appearing in the file LICENSE.TQPL
** included in the packaging of this file. Licensees holding valid TQt
** Commercial licenses may use this file in accordance with the TQt
** Commercial License Agreement provided with the Software.
**
** This file is provided "AS IS" with NO WARRANTY OF ANY KIND,
** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR
** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted
** herein.
**
**********************************************************************/
#include "qplatformdefs.h"
#include "ntqapplication.h"
#ifndef QT_NO_DRAGANDDROP
#include "ntqwidget.h"
#include "ntqintdict.h"
#include "ntqdatetime.h"
#include "ntqdict.h"
#include "ntqguardedptr.h"
#include "ntqdragobject.h"
#include "ntqobjectlist.h"
#include "ntqcursor.h"
#include "ntqbitmap.h"
#include "ntqpainter.h"
#include "qt_x11_p.h"
// conflict resolution
const int XKeyPress = KeyPress;
const int XKeyRelease = KeyRelease;
#undef KeyPress
#undef KeyRelease
// this stuff is copied from qapp_x11.cpp
extern void qt_x11_intern_atom( const char *, Atom * );
#if defined(Q_C_CALLBACKS)
extern "C" {
#endif
extern void qt_ignore_badwindow();
extern bool qt_badwindow();
extern void tqt_enter_modal( TQWidget *widget );
extern void tqt_leave_modal( TQWidget *widget );
#if defined(Q_C_CALLBACKS)
}
#endif
extern Window qt_x11_findClientWindow( Window, Atom, bool );
extern Atom tqt_wm_state;
extern Time tqt_x_time;
extern Time tqt_x_user_time;
// this stuff is copied from qclb_x11.cpp
extern bool qt_xclb_wait_for_event( Display *dpy, Window win, int type,
XEvent *event, int timeout );
extern bool qt_xclb_read_property( Display *dpy, Window win, Atom property,
bool deleteProperty,
TQByteArray *buffer, int *size, Atom *type,
int *format, bool nullterm );
extern TQByteArray qt_xclb_read_incremental_property( Display *dpy, Window win,
Atom property,
int nbytes, bool nullterm );
// and all this stuff is copied -into- qapp_x11.cpp
void qt_xdnd_setup();
void qt_handle_xdnd_enter( TQWidget *, const XEvent *, bool );
void qt_handle_xdnd_position( TQWidget *, const XEvent *, bool );
void qt_handle_xdnd_status( TQWidget *, const XEvent *, bool );
void qt_handle_xdnd_leave( TQWidget *, const XEvent *, bool );
void qt_handle_xdnd_drop( TQWidget *, const XEvent *, bool );
void qt_handle_xdnd_finished( TQWidget *, const XEvent *, bool );
void qt_xdnd_handle_selection_request( const XSelectionRequestEvent * );
bool qt_xdnd_handle_badwindow();
// client messages
Atom qt_xdnd_enter;
Atom qt_xdnd_position;
Atom qt_xdnd_status;
Atom qt_xdnd_leave;
Atom qt_xdnd_drop;
Atom qt_xdnd_finished;
Atom qt_xdnd_type_list;
const int qt_xdnd_version = 4;
extern int qt_x11_translateButtonState( int s );
// Actions
//
// The Xdnd spec allows for user-defined actions. This could be implemented
// with a registration process in TQt. WE SHOULD do that later.
//
Atom qt_xdnd_action_copy;
Atom qt_xdnd_action_link;
Atom qt_xdnd_action_move;
Atom qt_xdnd_action_private;
static
TQDropEvent::Action xdndaction_to_qtaction(Atom atom)
{
if ( atom == qt_xdnd_action_copy || atom == 0 )
return TQDropEvent::Copy;
if ( atom == qt_xdnd_action_link )
return TQDropEvent::Link;
if ( atom == qt_xdnd_action_move )
return TQDropEvent::Move;
return TQDropEvent::Private;
}
static
int qtaction_to_xdndaction(TQDropEvent::Action a)
{
switch ( a ) {
case TQDropEvent::Copy:
return qt_xdnd_action_copy;
case TQDropEvent::Link:
return qt_xdnd_action_link;
case TQDropEvent::Move:
return qt_xdnd_action_move;
case TQDropEvent::Private:
return qt_xdnd_action_private;
default:
return qt_xdnd_action_copy;
}
}
// clean up the stuff used.
static void qt_xdnd_cleanup();
static void qt_xdnd_send_leave();
// XDND selection
Atom qt_xdnd_selection;
// other selection
static Atom qt_selection_property;
// INCR
static Atom qt_incr_atom;
// properties for XDND drop sites
Atom qt_xdnd_aware;
Atom qt_xdnd_proxy;
// real variables:
// xid of current drag source
static Atom qt_xdnd_dragsource_xid = 0;
// the types in this drop. 100 is no good, but at least it's big.
const int qt_xdnd_max_type = 100;
static Atom qt_xdnd_types[qt_xdnd_max_type];
static TQIntDict<TQCString> * qt_xdnd_drag_types = 0;
static TQDict<Atom> * qt_xdnd_atom_numbers = 0;
// timer used when target wants "continuous" move messages (eg. scroll)
static int heartbeat = -1;
// rectangle in which the answer will be the same
static TQRect qt_xdnd_source_sameanswer;
//static TQRect qt_xdnd_target_sameanswer;
static bool qt_xdnd_target_answerwas;
// top-level window we sent position to last.
static Window qt_xdnd_current_target;
// window to send events to (always valid if qt_xdnd_current_target)
static Window qt_xdnd_current_proxy_target;
// widget we forwarded position to last, and local position
static TQGuardedPtr<TQWidget> qt_xdnd_current_widget;
static TQPoint qt_xdnd_current_position;
// time of this drop, as type Atom to save on casts
static Atom qt_xdnd_source_current_time;
// timestamp from the XdndPosition and XdndDrop
static Time qt_xdnd_target_current_time;
// screen number containing the pointer... -1 means default
static int qt_xdnd_current_screen = -1;
// state of dragging... true if dragging, false if not
bool qt_xdnd_dragging = FALSE;
// need to check state of keyboard modifiers
static bool need_modifiers_check = FALSE;
// dict of payload data, sorted by type atom
static TQIntDict<TQByteArray> * qt_xdnd_target_data = 0;
// first drag object, or 0
static TQDragObject * qt_xdnd_source_object = 0;
// Motif dnd
extern void qt_motifdnd_enable( TQWidget *, bool );
extern TQByteArray qt_motifdnd_obtain_data( const char *format );
extern const char *qt_motifdnd_format( int n );
bool qt_motifdnd_active = FALSE;
static bool dndCancelled = FALSE;
// Shift/Ctrl handling, and final drop status
static TQDragObject::DragMode drag_mode;
static TQDropEvent::Action global_requested_action = TQDropEvent::Copy;
static TQDropEvent::Action global_accepted_action = TQDropEvent::Copy;
// for embedding only
static TQWidget* current_embedding_widget = 0;
static XEvent last_enter_event;
// cursors
static TQCursor *noDropCursor = 0;
static TQCursor *moveCursor = 0;
static TQCursor *copyCursor = 0;
static TQCursor *linkCursor = 0;
static TQPixmap *defaultPm = 0;
static const int default_pm_hotx = -2;
static const int default_pm_hoty = -16;
static const char* const default_pm[] = {
"13 9 3 1",
". c None",
" c #000000",
"X c #FFFFFF",
"X X X X X X X",
" X X X X X X ",
"X ......... X",
" X.........X ",
"X ......... X",
" X.........X ",
"X ......... X",
" X X X X X X ",
"X X X X X X X"
};
class TQShapedPixmapWidget : public TQWidget {
public:
TQShapedPixmapWidget(int screen = -1) :
TQWidget(TQApplication::desktop()->screen( screen ),
0, WStyle_Customize | WStyle_Tool | WStyle_NoBorder | WX11BypassWM ), oldpmser( 0 ), oldbmser( 0 )
{
x11SetWindowType( X11WindowTypeDND );
}
void setPixmap(TQPixmap pm, TQPoint hot)
{
int bmser = pm.mask() ? pm.mask()->serialNumber() : 0;
if( oldpmser == pm.serialNumber() && oldbmser == bmser
&& oldhot == hot )
return;
oldpmser = pm.serialNumber();
oldbmser = bmser;
oldhot = hot;
bool hotspot_in = !(hot.x() < 0 || hot.y() < 0 || hot.x() >= pm.width() || hot.y() >= pm.height());
// if the pixmap has hotspot in its area, make a "hole" in it at that position
// this will allow XTranslateCoordinates() to find directly the window below the cursor instead
// of finding this pixmap, and therefore there won't be needed any (slow) search for the window
// using findRealWindow()
if( hotspot_in ) {
TQBitmap mask = pm.mask() ? *pm.mask() : TQBitmap( pm.width(), pm.height());
if( !pm.mask())
mask.fill( TQt::color1 );
TQPainter p( &mask );
p.setPen( TQt::color0 );
p.drawPoint( hot.x(), hot.y());
p.end();
pm.setMask( mask );
setMask( mask );
} else if ( pm.mask() ) {
setMask( *pm.mask() );
} else {
clearMask();
}
resize(pm.width(),pm.height());
setErasePixmap(pm);
erase();
}
private:
int oldpmser;
int oldbmser;
TQPoint oldhot;
};
static TQShapedPixmapWidget * qt_xdnd_deco = 0;
static TQWidget* desktop_proxy = 0;
class TQExtraWidget : public TQWidget
{
public:
TQWExtra* extraData() { return TQWidget::extraData(); }
TQTLWExtra* topData() { return TQWidget::topData(); }
};
static bool qt_xdnd_enable( TQWidget* w, bool on )
{
if ( on ) {
TQWidget * xdnd_widget = 0;
if ( w->isDesktop() ) {
if ( desktop_proxy ) // *WE* already have one.
return FALSE;
// As per Xdnd4, use XdndProxy
XGrabServer( w->x11Display() );
Atom type = None;
int f;
unsigned long n, a;
WId *proxy_id_ptr;
XGetWindowProperty( w->x11Display(), w->winId(),
qt_xdnd_proxy, 0, 1, False,
XA_WINDOW, &type, &f,&n,&a,(uchar**)&proxy_id_ptr );
WId proxy_id = 0;
if ( type == XA_WINDOW && proxy_id_ptr ) {
proxy_id = *proxy_id_ptr;
XFree(proxy_id_ptr);
proxy_id_ptr = 0;
// Already exists. Real?
qt_ignore_badwindow();
XGetWindowProperty( w->x11Display(), proxy_id,
qt_xdnd_proxy, 0, 1, False,
XA_WINDOW, &type, &f,&n,&a,(uchar**)&proxy_id_ptr );
if ( qt_badwindow() || type != XA_WINDOW || !proxy_id_ptr || *proxy_id_ptr != proxy_id ) {
// Bogus - we will overwrite.
proxy_id = 0;
}
}
if ( proxy_id_ptr )
XFree(proxy_id_ptr);
if ( !proxy_id ) {
xdnd_widget = desktop_proxy = new TQWidget;
proxy_id = desktop_proxy->winId();
XChangeProperty ( w->x11Display(),
w->winId(), qt_xdnd_proxy,
XA_WINDOW, 32, PropModeReplace,
(unsigned char *)&proxy_id, 1 );
XChangeProperty ( w->x11Display(),
proxy_id, qt_xdnd_proxy,
XA_WINDOW, 32, PropModeReplace,
(unsigned char *)&proxy_id, 1 );
}
XUngrabServer( w->x11Display() );
} else {
xdnd_widget = w->topLevelWidget();
}
if ( xdnd_widget ) {
Atom atm = (Atom)qt_xdnd_version;
XChangeProperty ( xdnd_widget->x11Display(), xdnd_widget->winId(),
qt_xdnd_aware, XA_ATOM, 32, PropModeReplace,
(unsigned char *)&atm, 1 );
return TRUE;
} else {
return FALSE;
}
} else {
if ( w->isDesktop() ) {
XDeleteProperty( w->x11Display(), w->winId(),
qt_xdnd_proxy );
delete desktop_proxy;
desktop_proxy = 0;
}
return TRUE;
}
}
const char* qt_xdnd_atom_to_str( Atom a )
{
if ( !a ) return 0;
if ( a == XA_STRING )
return "text/plain"; // some Xdnd clients are dumb
if ( !qt_xdnd_drag_types ) {
qt_xdnd_drag_types = new TQIntDict<TQCString>( 17 );
qt_xdnd_drag_types->setAutoDelete( TRUE );
}
TQCString* result;
if ( !(result=qt_xdnd_drag_types->find( a )) ) {
const char* mimeType = XGetAtomName( TQPaintDevice::x11AppDisplay(), a );
if ( !mimeType )
return 0; // only happens on protocol error
result = new TQCString( mimeType );
qt_xdnd_drag_types->insert( (long)a, result );
XFree((void*)mimeType);
}
return *result;
}
Atom* qt_xdnd_str_to_atom( const char *mimeType )
{
if ( !mimeType || !*mimeType )
return 0;
if ( !qt_xdnd_atom_numbers ) {
qt_xdnd_atom_numbers = new TQDict<Atom>( 17 );
qt_xdnd_atom_numbers->setAutoDelete( TRUE );
}
Atom * result;
if ( (result = qt_xdnd_atom_numbers->find( mimeType )) )
return result;
result = new Atom;
*result = 0;
qt_x11_intern_atom( mimeType, result );
qt_xdnd_atom_numbers->insert( mimeType, result );
qt_xdnd_atom_to_str( *result );
return result;
}
void qt_xdnd_setup() {
// set up protocol atoms
qt_x11_intern_atom( "XdndEnter", &qt_xdnd_enter );
qt_x11_intern_atom( "XdndPosition", &qt_xdnd_position );
qt_x11_intern_atom( "XdndStatus", &qt_xdnd_status );
qt_x11_intern_atom( "XdndLeave", &qt_xdnd_leave );
qt_x11_intern_atom( "XdndDrop", &qt_xdnd_drop );
qt_x11_intern_atom( "XdndFinished", &qt_xdnd_finished );
qt_x11_intern_atom( "XdndTypeList", &qt_xdnd_type_list );
qt_x11_intern_atom( "XdndSelection", &qt_xdnd_selection );
qt_x11_intern_atom( "XdndAware", &qt_xdnd_aware );
qt_x11_intern_atom( "XdndProxy", &qt_xdnd_proxy );
qt_x11_intern_atom( "XdndActionCopy", &qt_xdnd_action_copy );
qt_x11_intern_atom( "XdndActionLink", &qt_xdnd_action_link );
qt_x11_intern_atom( "XdndActionMove", &qt_xdnd_action_move );
qt_x11_intern_atom( "XdndActionPrivate", &qt_xdnd_action_private );
qt_x11_intern_atom( "QT_SELECTION", &qt_selection_property );
qt_x11_intern_atom( "INCR", &qt_incr_atom );
tqAddPostRoutine( qt_xdnd_cleanup );
}
void qt_xdnd_cleanup()
{
delete qt_xdnd_drag_types;
qt_xdnd_drag_types = 0;
delete qt_xdnd_atom_numbers;
qt_xdnd_atom_numbers = 0;
delete qt_xdnd_target_data;
qt_xdnd_target_data = 0;
delete noDropCursor;
noDropCursor = 0;
delete copyCursor;
copyCursor = 0;
delete moveCursor;
moveCursor = 0;
delete linkCursor;
linkCursor = 0;
delete defaultPm;
defaultPm = 0;
delete desktop_proxy;
desktop_proxy = 0;
}
static TQWidget * find_child( TQWidget * tlw, TQPoint & p )
{
TQWidget * w = tlw;
p = w->mapFromGlobal( p );
bool done = FALSE;
while ( !done ) {
done = TRUE;
if ( ((TQExtraWidget*)w)->extraData() &&
((TQExtraWidget*)w)->extraData()->xDndProxy != 0 )
break; // stop searching for widgets under the mouse cursor if found widget is a proxy.
if ( w->children() ) {
TQObjectListIt it( *w->children() );
it.toLast();
TQObject * o;
while( (o=it.current()) ) {
--it;
if ( o->isWidgetType() &&
((TQWidget*)o)->isVisible() &&
((TQWidget*)o)->geometry().contains( p ) &&
!((TQWidget*)o)->isTopLevel()) {
w = (TQWidget *)o;
done = FALSE;
p = w->mapFromParent( p );
break;
}
}
}
}
return w;
}
static bool checkEmbedded(TQWidget* w, const XEvent* xe)
{
if (!w)
return FALSE;
if (current_embedding_widget != 0 && current_embedding_widget != w) {
qt_xdnd_current_target = ((TQExtraWidget*)current_embedding_widget)->extraData()->xDndProxy;
qt_xdnd_current_proxy_target = qt_xdnd_current_target;
qt_xdnd_send_leave();
qt_xdnd_current_target = 0;
qt_xdnd_current_proxy_target = 0;
current_embedding_widget = 0;
}
TQWExtra* extra = ((TQExtraWidget*)w)->extraData();
if ( extra && extra->xDndProxy != 0 ) {
if (current_embedding_widget != w) {
last_enter_event.xany.window = extra->xDndProxy;
XSendEvent( TQPaintDevice::x11AppDisplay(), extra->xDndProxy, False, NoEventMask,
&last_enter_event );
current_embedding_widget = w;
}
((XEvent*)xe)->xany.window = extra->xDndProxy;
XSendEvent( TQPaintDevice::x11AppDisplay(), extra->xDndProxy, False, NoEventMask,
(XEvent*)xe );
qt_xdnd_current_widget = w;
return TRUE;
}
current_embedding_widget = 0;
return FALSE;
}
void qt_handle_xdnd_enter( TQWidget *, const XEvent * xe, bool /*passive*/ )
{
//if ( !w->neveHadAChildWithDropEventsOn() )
//return; // haven't been set up for dnd
qt_motifdnd_active = FALSE;
last_enter_event.xclient = xe->xclient;
qt_xdnd_target_answerwas = FALSE;
const long *l = xe->xclient.data.l;
int version = (int)(((unsigned long)(l[1])) >> 24);
if ( version > qt_xdnd_version )
return;
qt_xdnd_dragsource_xid = l[0];
int j = 0;
if ( l[1] & 1 ) {
// get the types from XdndTypeList
Atom type = None;
int f;
unsigned long n, a;
Atom *data;
XGetWindowProperty( TQPaintDevice::x11AppDisplay(), qt_xdnd_dragsource_xid,
qt_xdnd_type_list, 0,
qt_xdnd_max_type, False, XA_ATOM, &type, &f,&n,&a,(uchar**)&data );
for ( ; j<qt_xdnd_max_type && j < (int)n; j++ ) {
qt_xdnd_types[j] = data[j];
}
if ( data )
XFree( (uchar*)data );
} else {
// get the types from the message
int i;
for( i=2; i < 5; i++ ) {
qt_xdnd_types[j++] = l[i];
}
}
qt_xdnd_types[j] = 0;
}
void qt_handle_xdnd_position( TQWidget *w, const XEvent * xe, bool passive )
{
const unsigned long *l = (const unsigned long *)xe->xclient.data.l;
TQPoint p( (l[2] & 0xffff0000) >> 16, l[2] & 0x0000ffff );
TQWidget * c = find_child( w, p ); // changes p to to c-local coordinates
if (!passive && checkEmbedded(c, xe))
return;
if ( !c || !c->acceptDrops() && c->isDesktop() ) {
return;
}
if ( l[0] != qt_xdnd_dragsource_xid ) {
//tqDebug( "xdnd drag position from unexpected source (%08lx not %08lx)",
// l[0], qt_xdnd_dragsource_xid );
return;
}
if (l[3] != 0) {
// timestamp from the source
qt_xdnd_target_current_time = tqt_x_user_time = l[3];
}
XClientMessageEvent response;
response.type = ClientMessage;
response.window = qt_xdnd_dragsource_xid;
response.format = 32;
response.message_type = qt_xdnd_status;
response.data.l[0] = w->winId();
response.data.l[1] = 0; // flags
response.data.l[2] = 0; // x, y
response.data.l[3] = 0; // w, h
response.data.l[4] = 0; // action
if ( !passive ) { // otherwise just reject
while ( c && !c->acceptDrops() && !c->isTopLevel() ) {
p = c->mapToParent( p );
c = c->parentWidget();
}
TQRect answerRect( c->mapToGlobal( p ), TQSize( 1,1 ) );
TQDragMoveEvent me( p );
TQDropEvent::Action accepted_action = xdndaction_to_qtaction(l[4]);
me.setAction(accepted_action);
if ( c != qt_xdnd_current_widget ) {
qt_xdnd_target_answerwas = FALSE;
if ( qt_xdnd_current_widget ) {
TQDragLeaveEvent e;
TQApplication::sendEvent( qt_xdnd_current_widget, &e );
}
if ( c->acceptDrops() ) {
qt_xdnd_current_widget = c;
qt_xdnd_current_position = p;
TQDragEnterEvent de( p );
de.setAction(accepted_action);
TQApplication::sendEvent( c, &de );
if ( de.isAccepted() ) {
me.accept( de.answerRect() );
if ( !de.isActionAccepted() ) // only as a copy (move if we del)
accepted_action = TQDropEvent::Copy;
else
me.acceptAction(TRUE);
} else {
me.ignore( de.answerRect() );
}
}
} else {
if ( qt_xdnd_target_answerwas ) {
me.accept();
me.acceptAction(global_requested_action == global_accepted_action);
}
}
if ( !c->acceptDrops() ) {
qt_xdnd_current_widget = 0;
answerRect = TQRect( p, TQSize( 1, 1 ) );
} else if ( xdndaction_to_qtaction(l[4]) < TQDropEvent::Private ) {
qt_xdnd_current_widget = c;
qt_xdnd_current_position = p;
TQApplication::sendEvent( c, &me );
qt_xdnd_target_answerwas = me.isAccepted();
if ( me.isAccepted() ) {
response.data.l[1] = 1; // yes
if ( !me.isActionAccepted() ) // only as a copy (move if we del)
accepted_action = TQDropEvent::Copy;
} else {
response.data.l[0] = 0;
}
answerRect = me.answerRect().intersect( c->rect() );
} else {
response.data.l[0] = 0;
answerRect = TQRect( p, TQSize( 1, 1 ) );
}
answerRect = TQRect( c->mapToGlobal( answerRect.topLeft() ),
answerRect.size() );
if ( answerRect.left() < 0 )
answerRect.setLeft( 0 );
if ( answerRect.right() > 4096 )
answerRect.setRight( 4096 );
if ( answerRect.top() < 0 )
answerRect.setTop( 0 );
if ( answerRect.bottom() > 4096 )
answerRect.setBottom( 4096 );
if ( answerRect.width() < 0 )
answerRect.setWidth( 0 );
if ( answerRect.height() < 0 )
answerRect.setHeight( 0 );
response.data.l[2] = (answerRect.x() << 16) + answerRect.y();
response.data.l[3] = (answerRect.width() << 16) + answerRect.height();
response.data.l[4] = qtaction_to_xdndaction(accepted_action);
global_accepted_action = accepted_action;
}
// reset
qt_xdnd_target_current_time = CurrentTime;
TQWidget * source = TQWidget::find( qt_xdnd_dragsource_xid );
if ( source && source->isDesktop() && !source->acceptDrops() )
source = 0;
if ( source )
qt_handle_xdnd_status( source, (const XEvent *)&response, passive );
else
XSendEvent( TQPaintDevice::x11AppDisplay(), qt_xdnd_dragsource_xid, False,
NoEventMask, (XEvent*)&response );
}
void qt_handle_xdnd_status( TQWidget * w, const XEvent * xe, bool /*passive*/ )
{
const unsigned long *l = (const unsigned long *)xe->xclient.data.l;
// Messy: TQDragResponseEvent is just a call to TQDragManager function
global_accepted_action = xdndaction_to_qtaction(l[4]);
TQDragResponseEvent e( (int)(l[1] & 1) );
TQApplication::sendEvent( w, &e );
if ( (int)(l[1] & 2) == 0 ) {
TQPoint p( (l[2] & 0xffff0000) >> 16, l[2] & 0x0000ffff );
TQSize s( (l[3] & 0xffff0000) >> 16, l[3] & 0x0000ffff );
qt_xdnd_source_sameanswer = TQRect( p, s );
if ( qt_xdnd_source_sameanswer.isNull() ) {
// Application wants "coninutous" move events
}
} else {
qt_xdnd_source_sameanswer = TQRect();
}
}
void qt_handle_xdnd_leave( TQWidget *w, const XEvent * xe, bool /*passive*/ )
{
//tqDebug( "xdnd leave" );
if ( !qt_xdnd_current_widget ||
w->topLevelWidget() != qt_xdnd_current_widget->topLevelWidget() ) {
return; // sanity
}
if (checkEmbedded(current_embedding_widget, xe)) {
current_embedding_widget = 0;
qt_xdnd_current_widget = 0;
return;
}
const unsigned long *l = (const unsigned long *)xe->xclient.data.l;
TQDragLeaveEvent e;
TQApplication::sendEvent( qt_xdnd_current_widget, &e );
if ( l[0] != qt_xdnd_dragsource_xid ) {
// This often happens - leave other-process window tquickly
//tqDebug( "xdnd drag leave from unexpected source (%08lx not %08lx",
//l[0], qt_xdnd_dragsource_xid );
qt_xdnd_current_widget = 0;
return;
}
qt_xdnd_dragsource_xid = 0;
qt_xdnd_types[0] = 0;
qt_xdnd_current_widget = 0;
}
void qt_xdnd_send_leave()
{
if ( !qt_xdnd_current_target )
return;
XClientMessageEvent leave;
leave.type = ClientMessage;
leave.window = qt_xdnd_current_target;
leave.format = 32;
leave.message_type = qt_xdnd_leave;
leave.data.l[0] = qt_xdnd_dragsource_xid;
leave.data.l[1] = 0; // flags
leave.data.l[2] = 0; // x, y
leave.data.l[3] = 0; // w, h
leave.data.l[4] = 0; // just null
TQWidget * w = TQWidget::find( qt_xdnd_current_proxy_target );
if ( w && w->isDesktop() && !w->acceptDrops() )
w = 0;
if ( w )
qt_handle_xdnd_leave( w, (const XEvent *)&leave, FALSE );
else
XSendEvent( TQPaintDevice::x11AppDisplay(), qt_xdnd_current_proxy_target, False,
NoEventMask, (XEvent*)&leave );
qt_xdnd_current_target = 0;
qt_xdnd_current_proxy_target = 0;
}
void qt_handle_xdnd_drop( TQWidget *, const XEvent * xe, bool passive )
{
if ( !qt_xdnd_current_widget ) {
qt_xdnd_dragsource_xid = 0;
return; // sanity
}
if (!passive && checkEmbedded(qt_xdnd_current_widget, xe)){
current_embedding_widget = 0;
qt_xdnd_dragsource_xid = 0;
qt_xdnd_current_widget = 0;
return;
}
const unsigned long *l = (const unsigned long *)xe->xclient.data.l;
//tqDebug( "xdnd drop" );
if ( l[0] != qt_xdnd_dragsource_xid ) {
//tqDebug( "xdnd drop from unexpected source (%08lx not %08lx",
// l[0], qt_xdnd_dragsource_xid );
return;
}
if (l[2] != 0) {
// update the "user time" from the timestamp in the event.
qt_xdnd_target_current_time = tqt_x_user_time = l[2];
}
if ( qt_xdnd_source_object )
qt_xdnd_source_object->setTarget( qt_xdnd_current_widget );
if ( !passive ) {
TQDropEvent de( qt_xdnd_current_position );
de.setAction( global_accepted_action );
TQApplication::sendEvent( qt_xdnd_current_widget, &de );
if ( !de.isAccepted() ) {
// Ignore a failed drag
global_accepted_action = TQDropEvent::Copy;
dndCancelled = TRUE;
}
XClientMessageEvent finished;
finished.type = ClientMessage;
finished.window = qt_xdnd_dragsource_xid;
finished.format = 32;
finished.message_type = qt_xdnd_finished;
finished.data.l[0] = qt_xdnd_current_widget?qt_xdnd_current_widget->topLevelWidget()->winId():0;
finished.data.l[1] = 0; // flags
XSendEvent( TQPaintDevice::x11AppDisplay(), qt_xdnd_dragsource_xid, False,
NoEventMask, (XEvent*)&finished );
} else {
TQDragLeaveEvent e;
TQApplication::sendEvent( qt_xdnd_current_widget, &e );
}
qt_xdnd_dragsource_xid = 0;
qt_xdnd_current_widget = 0;
// reset
qt_xdnd_target_current_time = CurrentTime;
}
void qt_handle_xdnd_finished( TQWidget *, const XEvent * xe, bool passive )
{
const unsigned long *l = (const unsigned long *)xe->xclient.data.l;
if ( l[0] && (l[0] == qt_xdnd_current_target
|| l[0] == qt_xdnd_current_proxy_target) ) {
//
if ( !passive )
(void ) checkEmbedded( qt_xdnd_current_widget, xe);
current_embedding_widget = 0;
qt_xdnd_current_target = 0;
qt_xdnd_current_proxy_target = 0;
delete qt_xdnd_source_object;
qt_xdnd_source_object = 0;
}
}
void TQDragManager::timerEvent( TQTimerEvent* e )
{
if ( e->timerId() == heartbeat ) {
if( need_modifiers_check ) {
Window root, child;
int root_x, root_y, win_x, win_y;
unsigned int mask;
XQueryPointer( tqt_xdisplay(), tqt_xrootwin( qt_xdnd_current_screen ),
&root, &child, &root_x, &root_y, &win_x, &win_y, &mask );
if( updateMode( (ButtonState)qt_x11_translateButtonState( mask )))
qt_xdnd_source_sameanswer = TQRect(); // force move
}
need_modifiers_check = TRUE;
if( qt_xdnd_source_sameanswer.isNull() )
move( TQCursor::pos() );
}
}
static bool qt_xdnd_was_move = false;
static bool qt_xdnd_found = false;
// check whole incoming X queue for move events
// checking whole queue is done by always returning False in the predicate
// if there's another move event in the queue, and there's not a mouse button
// or keyboard or ClientMessage event before it, the current move event
// may be safely discarded
// this helps avoiding being overloaded by being flooded from many events
// from the XServer
static
Bool qt_xdnd_predicate( Display*, XEvent* ev, XPointer )
{
if( qt_xdnd_found )
return False;
if( ev->type == MotionNotify )
{
qt_xdnd_was_move = true;
qt_xdnd_found = true;
}
if( ev->type == ButtonPress || ev->type == ButtonRelease
|| ev->type == XKeyPress || ev->type == XKeyRelease
|| ev->type == ClientMessage )
{
qt_xdnd_was_move = false;
qt_xdnd_found = true;
}
return False;
}
static
bool qt_xdnd_another_movement()
{
qt_xdnd_was_move = false;
qt_xdnd_found = false;
XEvent dummy;
XCheckIfEvent( tqt_xdisplay(), &dummy, qt_xdnd_predicate, NULL );
return qt_xdnd_was_move;
}
bool TQDragManager::eventFilter( TQObject * o, TQEvent * e)
{
if ( beingCancelled ) {
if ( e->type() == TQEvent::KeyRelease &&
((TQKeyEvent*)e)->key() == Key_Escape ) {
tqApp->removeEventFilter( this );
object = 0;
dragSource = 0;
beingCancelled = FALSE;
tqApp->exit_loop();
return TRUE; // block the key release
}
return FALSE;
}
Q_ASSERT( object != 0 );
if ( !o->isWidgetType() )
return FALSE;
if ( e->type() == TQEvent::MouseMove ) {
TQMouseEvent* me = (TQMouseEvent *)e;
if( !qt_xdnd_another_movement()) {
updateMode(me->stateAfter());
move( me->globalPos() );
}
need_modifiers_check = FALSE;
return TRUE;
} else if ( e->type() == TQEvent::MouseButtonRelease ) {
tqApp->removeEventFilter( this );
if ( willDrop )
drop();
else
cancel();
object = 0;
dragSource = 0;
beingCancelled = FALSE;
tqApp->exit_loop();
return TRUE;
} else if ( e->type() == TQEvent::DragResponse ) {
if ( ((TQDragResponseEvent *)e)->dragAccepted() ) {
if ( !willDrop ) {
willDrop = TRUE;
}
} else {
if ( willDrop ) {
willDrop = FALSE;
}
}
updateCursor();
return TRUE;
}
if ( e->type() == TQEvent::KeyPress
|| e->type() == TQEvent::KeyRelease )
{
TQKeyEvent *ke = ((TQKeyEvent*)e);
if ( ke->key() == Key_Escape && e->type() == TQEvent::KeyPress ) {
cancel();
tqApp->removeEventFilter( this );
object = 0;
dragSource = 0;
beingCancelled = FALSE;
tqApp->exit_loop();
} else {
if( updateMode(ke->stateAfter())) {
qt_xdnd_source_sameanswer = TQRect(); // force move
move( TQCursor::pos() );
}
need_modifiers_check = FALSE;
}
return TRUE; // Eat all key events
}
// ### We bind modality to widgets, so we have to do this
// ### "manually".
// DnD is modal - eat all other interactive events
switch ( e->type() ) {
case TQEvent::MouseButtonPress:
case TQEvent::MouseButtonRelease:
case TQEvent::MouseButtonDblClick:
case TQEvent::MouseMove:
case TQEvent::KeyPress:
case TQEvent::KeyRelease:
case TQEvent::Wheel:
case TQEvent::Accel:
case TQEvent::AccelAvailable:
case TQEvent::AccelOverride:
return TRUE;
default:
return FALSE;
}
}
static TQt::ButtonState oldstate;
bool TQDragManager::updateMode( ButtonState newstate )
{
if ( newstate == oldstate )
return false;
const int both = ShiftButton|ControlButton;
if ( (newstate & both) == both ) {
global_requested_action = TQDropEvent::Link;
} else {
bool local = qt_xdnd_source_object != 0;
if ( drag_mode == TQDragObject::DragMove )
global_requested_action = TQDropEvent::Move;
else if ( drag_mode == TQDragObject::DragCopy )
global_requested_action = TQDropEvent::Copy;
else if ( drag_mode == TQDragObject::DragLink )
global_requested_action = TQDropEvent::Link;
else {
if ( drag_mode == TQDragObject::DragDefault && local )
global_requested_action = TQDropEvent::Move;
else
global_requested_action = TQDropEvent::Copy;
if ( newstate & ShiftButton )
global_requested_action = TQDropEvent::Move;
else if ( newstate & ControlButton )
global_requested_action = TQDropEvent::Copy;
}
}
oldstate = newstate;
return true;
}
void TQDragManager::createCursors()
{
if ( !noDropCursor ) {
noDropCursor = new TQCursor( ForbiddenCursor );
if ( !pm_cursor[0].isNull() )
moveCursor = new TQCursor(pm_cursor[0], 0,0);
if ( !pm_cursor[1].isNull() )
copyCursor = new TQCursor(pm_cursor[1], 0,0);
if ( !pm_cursor[2].isNull() )
linkCursor = new TQCursor(pm_cursor[2], 0,0);
}
}
void TQDragManager::updateCursor()
{
TQCursor *c;
if ( willDrop ) {
if ( global_accepted_action == TQDropEvent::Copy ) {
if ( global_requested_action == TQDropEvent::Move )
c = moveCursor; // (source can delete)
else
c = copyCursor;
} else if ( global_accepted_action == TQDropEvent::Link ) {
c = linkCursor;
} else {
c = moveCursor;
}
if ( qt_xdnd_deco ) {
qt_xdnd_deco->show();
qt_xdnd_deco->raise();
}
} else {
c = noDropCursor;
//if ( qt_xdnd_deco )
// qt_xdnd_deco->hide();
}
#ifndef QT_NO_CURSOR
if ( c )
tqApp->setOverrideCursor( *c, TRUE );
#endif
}
void TQDragManager::cancel( bool deleteSource )
{
killTimer( heartbeat );
heartbeat = -1;
if ( object ) {
beingCancelled = TRUE;
object = 0;
}
if ( qt_xdnd_current_target ) {
qt_xdnd_send_leave();
}
#ifndef QT_NO_CURSOR
if ( restoreCursor ) {
TQApplication::restoreOverrideCursor();
restoreCursor = FALSE;
}
#endif
if ( deleteSource )
delete qt_xdnd_source_object;
qt_xdnd_source_object = 0;
delete qt_xdnd_deco;
qt_xdnd_deco = 0;
dndCancelled = TRUE;
}
static
Window findRealWindow( const TQPoint & pos, Window w, int md )
{
if ( qt_xdnd_deco && w == qt_xdnd_deco->winId() )
return 0;
if ( md ) {
qt_ignore_badwindow();
XWindowAttributes attr;
XGetWindowAttributes( TQPaintDevice::x11AppDisplay(), w, &attr );
if (qt_badwindow())
return 0;
if ( attr.map_state == IsViewable
&& TQRect(attr.x,attr.y,attr.width,attr.height)
.contains(pos) )
{
{
Atom type = None;
int f;
unsigned long n, a;
unsigned char *data;
XGetWindowProperty( TQPaintDevice::x11AppDisplay(), w, qt_xdnd_aware, 0,
0, False, AnyPropertyType, &type, &f,&n,&a,&data );
if ( data ) XFree(data);
if ( type ) return w;
}
Window r, p;
Window* c;
uint nc;
if ( XQueryTree( TQPaintDevice::x11AppDisplay(), w, &r, &p, &c, &nc ) ) {
r=0;
for (uint i=nc; !r && i--; ) {
r = findRealWindow( pos-TQPoint(attr.x,attr.y),
c[i], md-1 );
}
XFree(c);
if ( r )
return r;
// We didn't find a client window! Just use the
// innermost window.
}
// No children!
return w;
}
}
return 0;
}
void TQDragManager::move( const TQPoint & globalPos )
{
if (!object) {
// perhaps the target crashed?
return;
}
int screen = TQCursor::x11Screen();
if ( ( qt_xdnd_current_screen == -1 && screen != TQPaintDevice::x11AppScreen() ) ||
( screen != qt_xdnd_current_screen ) ) {
// recreate the pixmap on the new screen...
delete qt_xdnd_deco;
qt_xdnd_deco = new TQShapedPixmapWidget( screen );
qt_xdnd_deco->x11SetWindowTransient( dragSource->topLevelWidget());
if (!TQWidget::mouseGrabber()) {
updatePixmap();
qt_xdnd_deco->grabMouse();
}
}
updatePixmap( globalPos );
if ( qt_xdnd_source_sameanswer.contains( globalPos ) &&
qt_xdnd_source_sameanswer.isValid() ) {
return;
}
qt_xdnd_current_screen = screen;
Window rootwin = TQPaintDevice::x11AppRootWindow( qt_xdnd_current_screen );
Window target = 0;
int lx = 0, ly = 0;
if ( !XTranslateCoordinates( TQPaintDevice::x11AppDisplay(), rootwin, rootwin,
globalPos.x(), globalPos.y(),
&lx, &ly, &target) )
// some wierd error...
return;
if ( target == rootwin ) {
// Ok.
} else if ( target ) {
//me
Window src = rootwin;
while (target != 0) {
int lx2, ly2;
Window t;
// translate coordinates
if (!XTranslateCoordinates(TQPaintDevice::x11AppDisplay(), src, target,
lx, ly, &lx2, &ly2, &t)) {
target = 0;
break;
}
lx = lx2;
ly = ly2;
src = target;
// check if it has XdndAware
Atom type = None;
int f;
unsigned long n, a;
unsigned char *data = 0;
XGetWindowProperty(TQPaintDevice::x11AppDisplay(), target, qt_xdnd_aware, 0, 0, False,
AnyPropertyType, &type, &f,&n,&a,&data);
if (data)
XFree(data);
if (type)
break;
// find child at the coordinates
if (!XTranslateCoordinates( TQPaintDevice::x11AppDisplay(), src, src,
lx, ly, &lx2, &ly2, &target)) {
target = 0;
break;
}
}
if ( qt_xdnd_deco && (!target || target == qt_xdnd_deco->winId()) ) {
target = findRealWindow(globalPos,rootwin,6);
}
}
TQWidget* w;
if ( target ) {
w = TQWidget::find( (WId)target );
if ( w && w->isDesktop() && !w->acceptDrops() )
w = 0;
} else {
w = 0;
target = rootwin;
}
WId proxy_target = target;
int target_version = 1;
{
Atom type = None;
int r, f;
unsigned long n, a;
WId *proxy_id;
qt_ignore_badwindow();
r = XGetWindowProperty( tqt_xdisplay(), target, qt_xdnd_proxy, 0,
1, False, XA_WINDOW, &type, &f,&n,&a,(uchar**)&proxy_id );
if ( ( r != Success ) || qt_badwindow() ) {
proxy_target = target = 0;
} else if ( type == XA_WINDOW && proxy_id ) {
proxy_target = *proxy_id;
XFree(proxy_id);
proxy_id = 0;
r = XGetWindowProperty( tqt_xdisplay(), proxy_target, qt_xdnd_proxy, 0,
1, False, XA_WINDOW, &type, &f,&n,&a,(uchar**)&proxy_id );
if ( ( r != Success ) || qt_badwindow() || !type || !proxy_id || *proxy_id != proxy_target ) {
// Bogus
proxy_target = 0;
target = 0;
}
if ( proxy_id )
XFree(proxy_id);
}
if ( proxy_target ) {
int *tv;
qt_ignore_badwindow();
r = XGetWindowProperty( tqt_xdisplay(), proxy_target, qt_xdnd_aware, 0,
1, False, AnyPropertyType, &type, &f,&n,&a,(uchar**)&tv );
if ( r != Success ) {
target = 0;
} else {
target_version = TQMIN(qt_xdnd_version,tv ? *tv : 1);
if ( tv )
XFree( tv );
if (!(!qt_badwindow() && type))
target = 0;
}
}
}
if ( target != qt_xdnd_current_target ) {
if ( qt_xdnd_current_target )
qt_xdnd_send_leave();
qt_xdnd_current_target = target;
qt_xdnd_current_proxy_target = proxy_target;
if ( target ) {
TQMemArray<Atom> type;
int flags = target_version << 24;
const char* fmt;
int nfmt=0;
for (nfmt=0; (fmt=object->format(nfmt)); nfmt++) {
type.resize(nfmt+1);
type[nfmt] = *qt_xdnd_str_to_atom( fmt );
}
if ( nfmt >= 3 ) {
XChangeProperty( TQPaintDevice::x11AppDisplay(),
object->source()->winId(), qt_xdnd_type_list,
XA_ATOM, 32, PropModeReplace,
(unsigned char *)type.data(),
type.size() );
flags |= 0x0001;
}
XClientMessageEvent enter;
enter.type = ClientMessage;
enter.window = target;
enter.format = 32;
enter.message_type = qt_xdnd_enter;
enter.data.l[0] = object->source()->winId();
enter.data.l[1] = flags;
enter.data.l[2] = type.size()>0 ? type[0] : 0;
enter.data.l[3] = type.size()>1 ? type[1] : 0;
enter.data.l[4] = type.size()>2 ? type[2] : 0;
// provisionally set the rectangle to 5x5 pixels...
qt_xdnd_source_sameanswer = TQRect( globalPos.x() - 2,
globalPos.y() -2 , 5, 5 );
if ( w ) {
qt_handle_xdnd_enter( w, (const XEvent *)&enter, FALSE );
} else if ( target ) {
XSendEvent( TQPaintDevice::x11AppDisplay(), proxy_target, False, NoEventMask,
(XEvent*)&enter );
}
}
}
if ( target ) {
XClientMessageEvent move;
move.type = ClientMessage;
move.window = target;
move.format = 32;
move.message_type = qt_xdnd_position;
move.window = target;
move.data.l[0] = object->source()->winId();
move.data.l[1] = 0; // flags
move.data.l[2] = (globalPos.x() << 16) + globalPos.y();
move.data.l[3] = tqt_x_time;
move.data.l[4] = qtaction_to_xdndaction( global_requested_action );
if ( w )
qt_handle_xdnd_position( w, (const XEvent *)&move, FALSE );
else
XSendEvent( TQPaintDevice::x11AppDisplay(), proxy_target, False, NoEventMask,
(XEvent*)&move );
} else {
if ( willDrop ) {
willDrop = FALSE;
updateCursor();
}
}
}
void TQDragManager::drop()
{
killTimer( heartbeat );
heartbeat = -1;
if ( !qt_xdnd_current_target )
return;
delete qt_xdnd_deco;
qt_xdnd_deco = 0;
XClientMessageEvent drop;
drop.type = ClientMessage;
drop.window = qt_xdnd_current_target;
drop.format = 32;
drop.message_type = qt_xdnd_drop;
drop.data.l[0] = object->source()->winId();
drop.data.l[1] = 0; // flags
drop.data.l[2] = tqt_x_time;
drop.data.l[3] = 0;
drop.data.l[4] = 0;
TQWidget * w = TQWidget::find( qt_xdnd_current_proxy_target );
if ( w && w->isDesktop() && !w->acceptDrops() )
w = 0;
if ( w )
qt_handle_xdnd_drop( w, (const XEvent *)&drop, FALSE );
else
XSendEvent( TQPaintDevice::x11AppDisplay(), qt_xdnd_current_proxy_target, False,
NoEventMask, (XEvent*)&drop );
#ifndef QT_NO_CURSOR
if ( restoreCursor ) {
TQApplication::restoreOverrideCursor();
restoreCursor = FALSE;
}
#endif
}
bool qt_xdnd_handle_badwindow()
{
if ( qt_xdnd_source_object && qt_xdnd_current_target ) {
qt_xdnd_current_target = 0;
qt_xdnd_current_proxy_target = 0;
delete qt_xdnd_source_object;
qt_xdnd_source_object = 0;
delete qt_xdnd_deco;
qt_xdnd_deco = 0;
return TRUE;
}
if ( qt_xdnd_dragsource_xid ) {
qt_xdnd_dragsource_xid = 0;
if ( qt_xdnd_current_widget ) {
TQDragLeaveEvent e;
TQApplication::sendEvent( qt_xdnd_current_widget, &e );
qt_xdnd_current_widget = 0;
}
return TRUE;
}
return FALSE;
}
/*!
\class TQDragMoveEvent ntqevent.h
\ingroup events
\ingroup draganddrop
\brief The TQDragMoveEvent class provides an event which is sent while a drag and drop is in progress.
When a widget \link TQWidget::setAcceptDrops() accepts drop
events\endlink, it will receive this event repeatedly while the
drag is within the widget's boundaries. The widget should examine
the event to see what data it \link TQDragMoveEvent::provides()
provides\endlink, and accept() the drop if appropriate.
Note that this class inherits most of its functionality from
TQDropEvent.
*/
/*!
Returns TRUE if this event provides format \a mimeType; otherwise
returns FALSE.
\sa data()
*/
bool TQDropEvent::provides( const char *mimeType ) const
{
if ( qt_motifdnd_active && tqstrnicmp( mimeType, "text/", 5 ) == 0 )
return TRUE;
int n=0;
const char* f;
do {
f = format( n );
if ( !f )
return FALSE;
n++;
} while( tqstricmp( mimeType, f ) );
return TRUE;
}
void qt_xdnd_handle_selection_request( const XSelectionRequestEvent * req )
{
if ( !req )
return;
XEvent evt;
evt.xselection.type = SelectionNotify;
evt.xselection.display = req->display;
evt.xselection.requestor = req->requestor;
evt.xselection.selection = req->selection;
evt.xselection.target = req->target;
evt.xselection.property = None;
evt.xselection.time = req->time;
const char* format = qt_xdnd_atom_to_str( req->target );
if ( format && qt_xdnd_source_object &&
qt_xdnd_source_object->provides( format ) ) {
TQByteArray a = qt_xdnd_source_object->encodedData(format);
XChangeProperty ( TQPaintDevice::x11AppDisplay(), req->requestor, req->property,
req->target, 8, PropModeReplace,
(unsigned char *)a.data(), a.size() );
evt.xselection.property = req->property;
}
// ### this can die if req->requestor crashes at the wrong
// ### moment
XSendEvent( TQPaintDevice::x11AppDisplay(), req->requestor, False, 0, &evt );
}
/*
XChangeProperty ( TQPaintDevice::x11AppDisplay(), req->requestor, req->property,
XA_STRING, 8,
PropModeReplace,
(uchar *)d->text(), strlen(d->text()) );
evt.xselection.property = req->property;
*/
static TQByteArray qt_xdnd_obtain_data( const char *format )
{
TQByteArray result;
TQWidget* w;
if ( qt_xdnd_dragsource_xid && qt_xdnd_source_object &&
(w=TQWidget::find( qt_xdnd_dragsource_xid ))
&& (!w->isDesktop() || w->acceptDrops()) )
{
TQDragObject * o = qt_xdnd_source_object;
if ( o->provides( format ) )
result = o->encodedData(format);
return result;
}
Atom * a = qt_xdnd_str_to_atom( format );
if ( !a || !*a )
return result;
if ( !qt_xdnd_target_data )
qt_xdnd_target_data = new TQIntDict<TQByteArray>( 17 );
if ( qt_xdnd_target_data->find( (int)*a ) ) {
result = *(qt_xdnd_target_data->find( (int)*a ));
} else {
if ( XGetSelectionOwner( TQPaintDevice::x11AppDisplay(),
qt_xdnd_selection ) == None )
return result; // should never happen?
TQWidget* tw = qt_xdnd_current_widget;
if ( !qt_xdnd_current_widget ||
qt_xdnd_current_widget->isDesktop() ) {
tw = new TQWidget;
}
XConvertSelection( TQPaintDevice::x11AppDisplay(),
qt_xdnd_selection,
*a,
qt_xdnd_selection,
tw->winId(),
qt_xdnd_target_current_time );
XFlush( TQPaintDevice::x11AppDisplay() );
XEvent xevent;
bool got=qt_xclb_wait_for_event( TQPaintDevice::x11AppDisplay(),
tw->winId(),
SelectionNotify, &xevent, 5000);
if ( got ) {
Atom type;
if ( qt_xclb_read_property( TQPaintDevice::x11AppDisplay(),
tw->winId(),
qt_xdnd_selection, TRUE,
&result, 0, &type, 0, FALSE ) ) {
if ( type == qt_incr_atom ) {
int nbytes = result.size() >= 4 ? *((int*)result.data()) : 0;
result = qt_xclb_read_incremental_property( TQPaintDevice::x11AppDisplay(),
tw->winId(),
qt_xdnd_selection,
nbytes, FALSE );
} else if ( type != *a ) {
// (includes None) tqDebug( "TQt clipboard: unknown atom %ld", type);
}
#if 0
// this needs to be matched by a qt_xdnd_target_data->clear()
// when each drag is finished. for 2.0, we do the safe thing
// and disable the entire caching.
if ( type != None )
qt_xdnd_target_data->insert( (int)((long)a), new TQByteArray(result) );
#endif
}
}
if ( !qt_xdnd_current_widget ||
qt_xdnd_current_widget->isDesktop() ) {
delete tw;
}
}
return result;
}
/*
Enable drag and drop for widget w by installing the proper
properties on w's toplevel widget.
*/
bool qt_dnd_enable( TQWidget* w, bool on )
{
w = w->topLevelWidget();
if ( on ) {
if ( ( (TQExtraWidget*)w)->topData()->dnd )
return TRUE; // been there, done that
((TQExtraWidget*)w)->topData()->dnd = 1;
}
qt_motifdnd_enable( w, on );
return qt_xdnd_enable( w, on );
}
/*!
\class TQDropEvent ntqevent.h
\ingroup events
\ingroup draganddrop
\brief The TQDropEvent class provides an event which is sent when a drag and drop is completed.
When a widget \link TQWidget::setAcceptDrops() accepts drop
events\endlink, it will receive this event if it has accepted the
most recent TQDragEnterEvent or TQDragMoveEvent sent to it.
The widget should use data() to extract the data in an appropriate
format.
*/
/*!
\fn TQDropEvent::TQDropEvent (const TQPoint & pos, Type typ)
Constructs a drop event that drops a drop of type \a typ on point
\a pos.
*/ // ### pos is in which coordinate system?
/*!
Returns a byte array containing the drag's data, in \a format.
data() normally needs to get the data from the drag source, which
is potentially very slow, so it's advisable to call this function
only if you're sure that you will need the data in \a format.
The resulting data will have a size of 0 if the format was not
available.
\sa format() TQByteArray::size()
*/
TQByteArray TQDropEvent::encodedData( const char *format ) const
{
if ( qt_motifdnd_active )
return qt_motifdnd_obtain_data( format );
return qt_xdnd_obtain_data( format );
}
/*!
Returns a string describing one of the available data types for
this drag. Common examples are "text/plain" and "image/gif". If \a
n is less than zero or greater than the number of available data
types, format() returns 0.
This function is provided mainly for debugging. Most drop targets
will use provides().
\sa data() provides()
*/
const char* TQDropEvent::format( int n ) const
{
if ( qt_motifdnd_active )
return qt_motifdnd_format( n );
int i = 0;
while( i<n && qt_xdnd_types[i] )
i++;
if ( i < n )
return 0;
const char* name = qt_xdnd_atom_to_str( qt_xdnd_types[i] );
if ( !name )
return 0; // should never happen
return name;
}
bool TQDragManager::drag( TQDragObject * o, TQDragObject::DragMode mode )
{
if ( object == o || !o || !o->parent() )
return FALSE;
if ( object ) {
cancel();
tqApp->removeEventFilter( this );
beingCancelled = FALSE;
}
if ( qt_xdnd_source_object ) {
// the last drag and drop operation hasn't finished, so we are going to wait
// for one second to see if it does... if the finish message comes after this,
// then we could still have problems, but this is highly unlikely
TQApplication::flushX();
TQTime started = TQTime::currentTime();
TQTime now = started;
do {
XEvent event;
if ( XCheckTypedEvent( TQPaintDevice::x11AppDisplay(),
ClientMessage, &event ) )
tqApp->x11ProcessEvent( &event );
now = TQTime::currentTime();
if ( started > now ) // crossed midnight
started = now;
// sleep 50ms, so we don't use up CPU cycles all the time.
struct timeval usleep_tv;
usleep_tv.tv_sec = 0;
usleep_tv.tv_usec = 50000;
select(0, 0, 0, 0, &usleep_tv);
} while ( qt_xdnd_source_object && started.msecsTo(now) < 1000 );
}
qt_xdnd_source_object = o;
qt_xdnd_source_object->setTarget( 0 );
qt_xdnd_deco = new TQShapedPixmapWidget();
willDrop = FALSE;
object = o;
updatePixmap();
dragSource = (TQWidget *)(object->parent());
qt_xdnd_deco->x11SetWindowTransient( dragSource->topLevelWidget());
tqApp->installEventFilter( this );
qt_xdnd_source_current_time = tqt_x_time;
XSetSelectionOwner( TQPaintDevice::x11AppDisplay(), qt_xdnd_selection,
dragSource->topLevelWidget()->winId(),
qt_xdnd_source_current_time );
oldstate = ButtonState(-1); // #### Should use state that caused the drag
drag_mode = mode;
global_accepted_action = TQDropEvent::Copy;
updateMode(ButtonState(0));
qt_xdnd_source_sameanswer = TQRect();
move(TQCursor::pos());
heartbeat = startTimer(200);
need_modifiers_check = FALSE;
#ifndef QT_NO_CURSOR
tqApp->setOverrideCursor( arrowCursor );
restoreCursor = TRUE;
updateCursor();
#endif
dndCancelled = FALSE;
qt_xdnd_dragging = TRUE;
if (!TQWidget::mouseGrabber())
qt_xdnd_deco->grabMouse();
tqApp->enter_loop(); // Do the DND.
#ifndef QT_NO_CURSOR
tqApp->restoreOverrideCursor();
#endif
delete qt_xdnd_deco;
qt_xdnd_deco = 0;
killTimer( heartbeat );
heartbeat = -1;
qt_xdnd_current_screen = -1;
qt_xdnd_dragging = FALSE;
return ((! dndCancelled) && // source del?
(global_accepted_action == TQDropEvent::Copy &&
global_requested_action == TQDropEvent::Move));
// qt_xdnd_source_object persists until we get an xdnd_finish message
}
void TQDragManager::updatePixmap( const TQPoint& cursorPos )
{
if ( qt_xdnd_deco ) {
TQPixmap pm;
TQPoint pm_hot(default_pm_hotx,default_pm_hoty);
if ( object ) {
pm = object->pixmap();
if ( !pm.isNull() )
pm_hot = object->pixmapHotSpot();
}
if ( pm.isNull() ) {
if ( !defaultPm )
defaultPm = new TQPixmap(default_pm);
pm = *defaultPm;
}
qt_xdnd_deco->setPixmap(pm, pm_hot);
qt_xdnd_deco->move(cursorPos-pm_hot);
//if ( willDrop ) {
qt_xdnd_deco->show();
//} else {
// qt_xdnd_deco->hide();
//}
}
}
void TQDragManager::updatePixmap()
{
updatePixmap( TQCursor::pos());
}
#endif // QT_NO_DRAGANDDROP