forked from AuroraMiddleware/gtk
4172138154
Commit 1c96b703
changed the way icon
information is given to DnD. Previously an icon helper was kept at
the drag source site. Now an image definition is stored there.
The difference is that icon helper is an object that changes its
state in response to an icon being set, thus the object survived
multiple icon changes. Whereas image definition is destroyed and
re-created from scratch every time a drag icon is changed.
This created a problem where gtk_drag_begin_internal() would receive
the value of site->image_def when a drag just began, then it emits
"drag-begin" signal, in response to which an application can
set drag icon, changing the value of site->image_def. However,
gtk_drag_begin_internal() is unable to know about that change and
continues to use the old value it received from up the stack.
Not only does it prevent drag icon from being set from "drag-begin",
it also can induce a crash, since the old image_def value used
by gtk_drag_begin_internal() points to a freed memory region.
Fix this by only setting a default icon (which is created in-place)
in gtk_drag_begin_internal() if the caller does not care about icons.
Otherwise gtk_drag_begin_internal() will return a boolean that indicates
whether an icon needs to be set. Then the caller can invoke
gtk_drag_set_icon_definition() to set the icon, if needed.
Fixes #1407.
3335 lines
105 KiB
C
3335 lines
105 KiB
C
/* GTK - The GIMP Toolkit
|
||
* Copyright (C) 1995-1999 Peter Mattis, Spencer Kimball and Josh MacDonald
|
||
*
|
||
* This library is free software; you can redistribute it and/or
|
||
* modify it under the terms of the GNU Lesser General Public
|
||
* License as published by the Free Software Foundation; either
|
||
* version 2 of the License, or (at your option) any later version.
|
||
*
|
||
* This library is distributed in the hope that it will be useful,
|
||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
* Lesser General Public License for more details.
|
||
*
|
||
* You should have received a copy of the GNU Lesser General Public
|
||
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
|
||
/*
|
||
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
||
* file for a list of people on the GTK+ Team. See the ChangeLog
|
||
* files for a list of changes. These files are distributed with
|
||
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
||
*/
|
||
|
||
#include "config.h"
|
||
|
||
#include "gtkdnd.h"
|
||
#include "gtkdndprivate.h"
|
||
#include "gtksettingsprivate.h"
|
||
|
||
#include <math.h>
|
||
#include <stdlib.h>
|
||
#include <string.h>
|
||
|
||
#include "gdk/gdk.h"
|
||
|
||
#ifdef GDK_WINDOWING_X11
|
||
#include <X11/Xlib.h>
|
||
#include <X11/keysym.h>
|
||
#include "gdk/x11/gdkx.h"
|
||
#ifdef XINPUT_2
|
||
#include <X11/extensions/XInput2.h>
|
||
#endif
|
||
#endif
|
||
|
||
#ifdef GDK_WINDOWING_WIN32
|
||
#include <gdk/win32/gdkwin32.h>
|
||
#endif
|
||
|
||
#ifdef GDK_WINDOWING_WAYLAND
|
||
#include <gdk/wayland/gdkwayland.h>
|
||
#endif
|
||
|
||
#include "gtkdragdest.h"
|
||
#include "gtkgesturedrag.h"
|
||
#include "gtkgesturesingle.h"
|
||
#include "gtkicontheme.h"
|
||
#include "gtkimageprivate.h"
|
||
#include "gtkintl.h"
|
||
#include "gtkmain.h"
|
||
#include "gtkplug.h"
|
||
#include "gtktooltipprivate.h"
|
||
#include "gtkwindow.h"
|
||
#include "gtkrender.h"
|
||
#include "gtkselectionprivate.h"
|
||
#include "gtkwindowgroup.h"
|
||
#include "gtkwindowprivate.h"
|
||
#include "gtkwidgetprivate.h"
|
||
|
||
|
||
/**
|
||
* SECTION:gtkdnd
|
||
* @Short_description: Functions for controlling drag and drop handling
|
||
* @Title: Drag and Drop
|
||
*
|
||
* GTK+ has a rich set of functions for doing inter-process communication
|
||
* via the drag-and-drop metaphor.
|
||
*
|
||
* As well as the functions listed here, applications may need to use some
|
||
* facilities provided for [Selections][gtk3-Selections]. Also, the Drag and
|
||
* Drop API makes use of signals in the #GtkWidget class.
|
||
*/
|
||
|
||
|
||
static GSList *source_widgets = NULL;
|
||
|
||
typedef struct _GtkDragSourceInfo GtkDragSourceInfo;
|
||
typedef struct _GtkDragDestInfo GtkDragDestInfo;
|
||
|
||
|
||
typedef enum
|
||
{
|
||
GTK_DRAG_STATUS_DRAG,
|
||
GTK_DRAG_STATUS_WAIT,
|
||
GTK_DRAG_STATUS_DROP
|
||
} GtkDragStatus;
|
||
|
||
struct _GtkDragSourceInfo
|
||
{
|
||
GtkWidget *widget;
|
||
GtkTargetList *target_list; /* Targets for drag data */
|
||
GdkDragAction possible_actions; /* Actions allowed by source */
|
||
GdkDragContext *context; /* drag context */
|
||
GtkWidget *icon_window; /* Window for drag */
|
||
GtkWidget *icon_widget; /* Widget for drag */
|
||
GtkWidget *ipc_widget; /* GtkInvisible for grab, message passing */
|
||
GdkCursor *cursor; /* Cursor for drag */
|
||
gint hot_x, hot_y; /* Hot spot for drag */
|
||
gint button; /* mouse button starting drag */
|
||
|
||
GtkDragStatus status; /* drag status */
|
||
GdkEvent *last_event; /* pending event */
|
||
|
||
gint start_x, start_y; /* Initial position */
|
||
gint cur_x, cur_y; /* Current Position */
|
||
GdkScreen *cur_screen; /* Current screen for pointer */
|
||
|
||
guint32 grab_time; /* timestamp for initial grab */
|
||
GList *selections; /* selections we've claimed */
|
||
|
||
GtkDragDestInfo *proxy_dest; /* Set if this is a proxy drag */
|
||
|
||
guint update_idle; /* Idle function to update the drag */
|
||
guint drop_timeout; /* Timeout for aborting drop */
|
||
guint destroy_icon : 1; /* If true, destroy icon_widget */
|
||
guint have_grab : 1; /* Do we still have the pointer grab */
|
||
};
|
||
|
||
struct _GtkDragDestInfo
|
||
{
|
||
GtkWidget *widget; /* Widget in which drag is in */
|
||
GdkDragContext *context; /* Drag context */
|
||
GtkDragSourceInfo *proxy_source; /* Set if this is a proxy drag */
|
||
GtkSelectionData *proxy_data; /* Set while retrieving proxied data */
|
||
guint32 proxy_drop_time; /* Timestamp for proxied drop */
|
||
guint proxy_drop_wait : 1; /* Set if we are waiting for a
|
||
* status reply before sending
|
||
* a proxied drop on.
|
||
*/
|
||
guint dropped : 1; /* Set after we receive a drop */
|
||
gint drop_x, drop_y; /* Position of drop */
|
||
};
|
||
|
||
#define DROP_ABORT_TIME 300000
|
||
|
||
typedef gboolean (* GtkDragDestCallback) (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
gint x,
|
||
gint y,
|
||
guint32 time);
|
||
|
||
/* Enumeration for some targets we handle internally */
|
||
|
||
enum {
|
||
TARGET_DELETE = 0x40000002
|
||
};
|
||
|
||
/* Forward declarations */
|
||
static void gtk_drag_get_event_actions (const GdkEvent *event,
|
||
gint button,
|
||
GdkDragAction actions,
|
||
GdkDragAction *suggested_action,
|
||
GdkDragAction *possible_actions);
|
||
static GdkCursor * gtk_drag_get_cursor (GtkWidget *widget,
|
||
GdkDisplay *display,
|
||
GdkDragAction action,
|
||
GtkDragSourceInfo *info);
|
||
static void gtk_drag_update_cursor (GtkDragSourceInfo *info);
|
||
static GtkWidget *gtk_drag_get_ipc_widget (GtkWidget *widget);
|
||
static GtkWidget *gtk_drag_get_ipc_widget_for_screen (GdkScreen *screen);
|
||
static void gtk_drag_release_ipc_widget (GtkWidget *widget);
|
||
|
||
static void gtk_drag_selection_received (GtkWidget *widget,
|
||
GtkSelectionData *selection_data,
|
||
guint time,
|
||
gpointer data);
|
||
static gboolean gtk_drag_find_widget (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
GtkDragDestInfo *info,
|
||
gint x,
|
||
gint y,
|
||
guint32 time,
|
||
GtkDragDestCallback callback);
|
||
static void gtk_drag_proxy_begin (GtkWidget *widget,
|
||
GtkDragDestInfo *dest_info,
|
||
guint32 time);
|
||
static void gtk_drag_dest_leave (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
guint time);
|
||
static gboolean gtk_drag_dest_motion (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
gint x,
|
||
gint y,
|
||
guint time);
|
||
static gboolean gtk_drag_dest_drop (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
gint x,
|
||
gint y,
|
||
guint time);
|
||
static void gtk_drag_dest_set_widget (GtkDragDestInfo *info,
|
||
GtkWidget *widget);
|
||
|
||
static GtkDragDestInfo * gtk_drag_get_dest_info (GdkDragContext *context,
|
||
gboolean create);
|
||
static GtkDragSourceInfo *gtk_drag_get_source_info (GdkDragContext *context,
|
||
gboolean create);
|
||
static void gtk_drag_clear_source_info (GdkDragContext *context);
|
||
|
||
static void gtk_drag_source_check_selection (GtkDragSourceInfo *info,
|
||
GdkAtom selection,
|
||
guint32 time);
|
||
static void gtk_drag_source_release_selections (GtkDragSourceInfo *info,
|
||
guint32 time);
|
||
static void gtk_drag_drop (GtkDragSourceInfo *info,
|
||
guint32 time);
|
||
static void gtk_drag_drop_finished (GtkDragSourceInfo *info,
|
||
GtkDragResult result,
|
||
guint time);
|
||
static void gtk_drag_cancel_internal (GtkDragSourceInfo *info,
|
||
GtkDragResult result,
|
||
guint32 time);
|
||
|
||
static void gtk_drag_selection_get (GtkWidget *widget,
|
||
GtkSelectionData *selection_data,
|
||
guint sel_info,
|
||
guint32 time,
|
||
gpointer data);
|
||
static void gtk_drag_remove_icon (GtkDragSourceInfo *info);
|
||
static void gtk_drag_source_info_destroy (GtkDragSourceInfo *info);
|
||
|
||
static void gtk_drag_context_drop_performed_cb (GdkDragContext *context,
|
||
guint time,
|
||
GtkDragSourceInfo *info);
|
||
static void gtk_drag_context_cancel_cb (GdkDragContext *context,
|
||
GdkDragCancelReason reason,
|
||
GtkDragSourceInfo *info);
|
||
static void gtk_drag_context_action_cb (GdkDragContext *context,
|
||
GdkDragAction action,
|
||
GtkDragSourceInfo *info);
|
||
static void gtk_drag_context_dnd_finished_cb (GdkDragContext *context,
|
||
GtkDragSourceInfo *info);
|
||
static void gtk_drag_add_update_idle (GtkDragSourceInfo *info);
|
||
|
||
static void gtk_drag_update (GtkDragSourceInfo *info,
|
||
GdkScreen *screen,
|
||
gint x_root,
|
||
gint y_root,
|
||
const GdkEvent *event);
|
||
static gboolean gtk_drag_motion_cb (GtkWidget *widget,
|
||
GdkEventMotion *event,
|
||
gpointer data);
|
||
static gboolean gtk_drag_key_cb (GtkWidget *widget,
|
||
GdkEventKey *event,
|
||
gpointer data);
|
||
static gboolean gtk_drag_grab_broken_event_cb (GtkWidget *widget,
|
||
GdkEventGrabBroken *event,
|
||
gpointer data);
|
||
static void gtk_drag_grab_notify_cb (GtkWidget *widget,
|
||
gboolean was_grabbed,
|
||
gpointer data);
|
||
static gboolean gtk_drag_button_release_cb (GtkWidget *widget,
|
||
GdkEventButton *event,
|
||
gpointer data);
|
||
static gboolean gtk_drag_abort_timeout (gpointer data);
|
||
|
||
static void set_icon_helper (GdkDragContext *context,
|
||
GtkImageDefinition*def,
|
||
gint hot_x,
|
||
gint hot_y);
|
||
|
||
/************************
|
||
* Cursor and Icon data *
|
||
************************/
|
||
|
||
static struct {
|
||
GdkDragAction action;
|
||
const gchar *name;
|
||
GdkPixbuf *pixbuf;
|
||
GdkCursor *cursor;
|
||
} drag_cursors[] = {
|
||
{ GDK_ACTION_DEFAULT, NULL },
|
||
{ GDK_ACTION_ASK, "dnd-ask", NULL, NULL },
|
||
{ GDK_ACTION_COPY, "copy", NULL, NULL },
|
||
{ GDK_ACTION_MOVE, "move", NULL, NULL },
|
||
{ GDK_ACTION_LINK, "alias", NULL, NULL },
|
||
{ 0 , "no-drop", NULL, NULL },
|
||
};
|
||
|
||
/*********************
|
||
* Utility functions *
|
||
*********************/
|
||
|
||
static GtkWidget *
|
||
gtk_drag_get_ipc_widget_for_screen (GdkScreen *screen)
|
||
{
|
||
GtkWidget *result;
|
||
GSList *drag_widgets = g_object_get_data (G_OBJECT (screen),
|
||
"gtk-dnd-ipc-widgets");
|
||
|
||
if (drag_widgets)
|
||
{
|
||
GSList *tmp = drag_widgets;
|
||
result = drag_widgets->data;
|
||
drag_widgets = drag_widgets->next;
|
||
g_object_set_data (G_OBJECT (screen),
|
||
I_("gtk-dnd-ipc-widgets"),
|
||
drag_widgets);
|
||
g_slist_free_1 (tmp);
|
||
}
|
||
else
|
||
{
|
||
result = gtk_window_new (GTK_WINDOW_POPUP);
|
||
gtk_window_set_screen (GTK_WINDOW (result), screen);
|
||
gtk_window_resize (GTK_WINDOW (result), 1, 1);
|
||
gtk_window_move (GTK_WINDOW (result), -99, -99);
|
||
gtk_widget_show (result);
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
static GtkWidget *
|
||
gtk_drag_get_ipc_widget (GtkWidget *widget)
|
||
{
|
||
GtkWidget *result;
|
||
GtkWidget *toplevel;
|
||
|
||
result = gtk_drag_get_ipc_widget_for_screen (gtk_widget_get_screen (widget));
|
||
|
||
toplevel = gtk_widget_get_toplevel (widget);
|
||
|
||
if (GTK_IS_WINDOW (toplevel))
|
||
{
|
||
if (gtk_window_has_group (GTK_WINDOW (toplevel)))
|
||
gtk_window_group_add_window (gtk_window_get_group (GTK_WINDOW (toplevel)),
|
||
GTK_WINDOW (result));
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
#if defined (GDK_WINDOWING_X11)
|
||
|
||
/*
|
||
* We want to handle a handful of keys during DND, e.g. Escape to abort.
|
||
* Grabbing the keyboard has the unfortunate side-effect of preventing
|
||
* useful things such as using Alt-Tab to cycle between windows or
|
||
* switching workspaces. Therefore, we just grab the few keys we are
|
||
* interested in. Note that we need to put the grabs on the root window
|
||
* in order for them to still work when the focus is moved to another
|
||
* app/workspace.
|
||
*
|
||
* GDK needs a little help to successfully deliver root key events...
|
||
*/
|
||
|
||
static GdkFilterReturn
|
||
root_key_filter (GdkXEvent *xevent,
|
||
GdkEvent *event,
|
||
gpointer data)
|
||
{
|
||
XEvent *ev = (XEvent *) xevent;
|
||
|
||
if ((ev->type == KeyPress || ev->type == KeyRelease) &&
|
||
ev->xkey.root == ev->xkey.window)
|
||
ev->xkey.window = (Window)data;
|
||
else if (ev->type == GenericEvent)
|
||
{
|
||
XGenericEventCookie *cookie;
|
||
XIDeviceEvent *dev;
|
||
|
||
cookie = &ev->xcookie;
|
||
dev = (XIDeviceEvent *) cookie->data;
|
||
|
||
if (dev->evtype == XI_KeyPress ||
|
||
dev->evtype == XI_KeyRelease)
|
||
dev->event = (Window)data;
|
||
}
|
||
|
||
return GDK_FILTER_CONTINUE;
|
||
}
|
||
|
||
typedef struct {
|
||
gint keysym;
|
||
gint modifiers;
|
||
} GrabKey;
|
||
|
||
static GrabKey grab_keys[] = {
|
||
{ XK_Escape, 0 },
|
||
{ XK_space, 0 },
|
||
{ XK_KP_Space, 0 },
|
||
{ XK_Return, 0 },
|
||
{ XK_KP_Enter, 0 },
|
||
{ XK_Up, 0 },
|
||
{ XK_Up, Mod1Mask },
|
||
{ XK_Down, 0 },
|
||
{ XK_Down, Mod1Mask },
|
||
{ XK_Left, 0 },
|
||
{ XK_Left, Mod1Mask },
|
||
{ XK_Right, 0 },
|
||
{ XK_Right, Mod1Mask },
|
||
{ XK_KP_Up, 0 },
|
||
{ XK_KP_Up, Mod1Mask },
|
||
{ XK_KP_Down, 0 },
|
||
{ XK_KP_Down, Mod1Mask },
|
||
{ XK_KP_Left, 0 },
|
||
{ XK_KP_Left, Mod1Mask },
|
||
{ XK_KP_Right, 0 },
|
||
{ XK_KP_Right, Mod1Mask }
|
||
};
|
||
|
||
static void
|
||
grab_dnd_keys (GtkWidget *widget,
|
||
GdkDevice *device,
|
||
guint32 time)
|
||
{
|
||
guint i;
|
||
GdkDisplay *display;
|
||
GdkWindow *window, *root;
|
||
gint keycode;
|
||
#ifdef XINPUT_2
|
||
gint deviceid;
|
||
XIGrabModifiers mods;
|
||
gint num_mods;
|
||
XIEventMask evmask;
|
||
unsigned char mask[(XI_LASTEVENT + 7)/8];
|
||
gboolean using_xi2;
|
||
|
||
window = gtk_widget_get_window (widget);
|
||
if (!GDK_IS_X11_WINDOW (window))
|
||
{
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
gdk_device_grab (device,
|
||
gtk_widget_get_window (widget),
|
||
GDK_OWNERSHIP_APPLICATION, FALSE,
|
||
GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK,
|
||
NULL, time);
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
return;
|
||
}
|
||
|
||
deviceid = gdk_x11_device_get_id (device);
|
||
|
||
if (GDK_IS_X11_DEVICE_XI2 (device))
|
||
using_xi2 = TRUE;
|
||
else
|
||
using_xi2 = FALSE;
|
||
#endif
|
||
|
||
display = gtk_widget_get_display (widget);
|
||
root = gdk_screen_get_root_window (gtk_widget_get_screen (widget));
|
||
|
||
gdk_x11_display_error_trap_push (display);
|
||
|
||
for (i = 0; i < G_N_ELEMENTS (grab_keys); ++i)
|
||
{
|
||
keycode = XKeysymToKeycode (GDK_WINDOW_XDISPLAY (window), grab_keys[i].keysym);
|
||
if (keycode == NoSymbol)
|
||
continue;
|
||
|
||
#ifdef XINPUT_2
|
||
if (using_xi2)
|
||
{
|
||
memset (mask, 0, sizeof (mask));
|
||
XISetMask (mask, XI_KeyPress);
|
||
XISetMask (mask, XI_KeyRelease);
|
||
|
||
evmask.deviceid = deviceid;
|
||
evmask.mask_len = sizeof (mask);
|
||
evmask.mask = mask;
|
||
|
||
num_mods = 1;
|
||
mods.modifiers = grab_keys[i].modifiers;
|
||
|
||
XIGrabKeycode (GDK_WINDOW_XDISPLAY (window),
|
||
deviceid,
|
||
keycode,
|
||
GDK_WINDOW_XID (root),
|
||
GrabModeAsync,
|
||
GrabModeAsync,
|
||
False,
|
||
&evmask,
|
||
num_mods,
|
||
&mods);
|
||
}
|
||
else
|
||
#endif
|
||
XGrabKey (GDK_WINDOW_XDISPLAY (window),
|
||
keycode, grab_keys[i].modifiers,
|
||
GDK_WINDOW_XID (root),
|
||
FALSE,
|
||
GrabModeAsync,
|
||
GrabModeAsync);
|
||
}
|
||
|
||
gdk_display_flush (display);
|
||
gdk_x11_display_error_trap_pop_ignored (display);
|
||
|
||
gdk_window_add_filter (NULL, root_key_filter, (gpointer) GDK_WINDOW_XID (window));
|
||
}
|
||
|
||
static void
|
||
ungrab_dnd_keys (GtkWidget *widget,
|
||
GdkDevice *device,
|
||
guint32 time)
|
||
{
|
||
guint i;
|
||
GdkWindow *window, *root;
|
||
GdkDisplay *display;
|
||
gint keycode;
|
||
#ifdef XINPUT_2
|
||
XIGrabModifiers mods;
|
||
gint num_mods;
|
||
gint deviceid;
|
||
gboolean using_xi2;
|
||
|
||
window = gtk_widget_get_window (widget);
|
||
if (!GDK_IS_X11_WINDOW (window))
|
||
{
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
gdk_device_ungrab (device, time);
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
return;
|
||
}
|
||
|
||
deviceid = gdk_x11_device_get_id (device);
|
||
|
||
if (GDK_IS_X11_DEVICE_XI2 (device))
|
||
using_xi2 = TRUE;
|
||
else
|
||
using_xi2 = FALSE;
|
||
#endif
|
||
|
||
display = gtk_widget_get_display (widget);
|
||
root = gdk_screen_get_root_window (gtk_widget_get_screen (widget));
|
||
|
||
gdk_window_remove_filter (NULL, root_key_filter, (gpointer) GDK_WINDOW_XID (window));
|
||
|
||
gdk_x11_display_error_trap_push (display);
|
||
|
||
for (i = 0; i < G_N_ELEMENTS (grab_keys); ++i)
|
||
{
|
||
keycode = XKeysymToKeycode (GDK_WINDOW_XDISPLAY (window), grab_keys[i].keysym);
|
||
if (keycode == NoSymbol)
|
||
continue;
|
||
|
||
#ifdef XINPUT_2
|
||
if (using_xi2)
|
||
{
|
||
num_mods = 1;
|
||
mods.modifiers = grab_keys[i].modifiers;
|
||
|
||
XIUngrabKeycode (GDK_WINDOW_XDISPLAY (window),
|
||
deviceid,
|
||
keycode,
|
||
GDK_WINDOW_XID (root),
|
||
num_mods,
|
||
&mods);
|
||
}
|
||
else
|
||
#endif
|
||
XUngrabKey (GDK_WINDOW_XDISPLAY (window),
|
||
keycode, grab_keys[i].modifiers,
|
||
GDK_WINDOW_XID (root));
|
||
}
|
||
|
||
gdk_display_flush (display);
|
||
gdk_x11_display_error_trap_pop_ignored (display);
|
||
}
|
||
|
||
#else /* !GDK_WINDOWING_X11 */
|
||
|
||
static void
|
||
grab_dnd_keys (GtkWidget *widget,
|
||
GdkDevice *device,
|
||
guint32 time)
|
||
{
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
gdk_device_grab (device,
|
||
gtk_widget_get_window (widget),
|
||
GDK_OWNERSHIP_APPLICATION, FALSE,
|
||
GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK,
|
||
NULL, time);
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
}
|
||
|
||
static void
|
||
ungrab_dnd_keys (GtkWidget *widget,
|
||
GdkDevice *device,
|
||
guint32 time)
|
||
{
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
gdk_device_ungrab (device, time);
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
}
|
||
|
||
#endif /* GDK_WINDOWING_X11 */
|
||
|
||
/*
|
||
* gtk_drag_release_ipc_widget:
|
||
* @widget: the widget to release
|
||
*
|
||
* Releases widget retrieved with gtk_drag_get_ipc_widget().
|
||
*/
|
||
static void
|
||
gtk_drag_release_ipc_widget (GtkWidget *widget)
|
||
{
|
||
GtkWindow *window = GTK_WINDOW (widget);
|
||
GdkScreen *screen = gtk_widget_get_screen (widget);
|
||
GdkDragContext *context = g_object_get_data (G_OBJECT (widget), "drag-context");
|
||
GSList *drag_widgets = g_object_get_data (G_OBJECT (screen),
|
||
"gtk-dnd-ipc-widgets");
|
||
GdkDevice *pointer, *keyboard;
|
||
|
||
if (context)
|
||
{
|
||
pointer = gdk_drag_context_get_device (context);
|
||
keyboard = gdk_device_get_associated_device (pointer);
|
||
|
||
if (keyboard)
|
||
ungrab_dnd_keys (widget, keyboard, GDK_CURRENT_TIME);
|
||
}
|
||
|
||
if (gtk_window_has_group (window))
|
||
gtk_window_group_remove_window (gtk_window_get_group (window),
|
||
window);
|
||
drag_widgets = g_slist_prepend (drag_widgets, widget);
|
||
g_object_set_data (G_OBJECT (screen),
|
||
I_("gtk-dnd-ipc-widgets"),
|
||
drag_widgets);
|
||
}
|
||
|
||
static guint32
|
||
gtk_drag_get_event_time (GdkEvent *event)
|
||
{
|
||
guint32 tm = GDK_CURRENT_TIME;
|
||
|
||
if (event)
|
||
switch (event->type)
|
||
{
|
||
case GDK_MOTION_NOTIFY:
|
||
tm = event->motion.time; break;
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_2BUTTON_PRESS:
|
||
case GDK_3BUTTON_PRESS:
|
||
case GDK_BUTTON_RELEASE:
|
||
tm = event->button.time; break;
|
||
case GDK_KEY_PRESS:
|
||
case GDK_KEY_RELEASE:
|
||
tm = event->key.time; break;
|
||
case GDK_ENTER_NOTIFY:
|
||
case GDK_LEAVE_NOTIFY:
|
||
tm = event->crossing.time; break;
|
||
case GDK_PROPERTY_NOTIFY:
|
||
tm = event->property.time; break;
|
||
case GDK_SELECTION_CLEAR:
|
||
case GDK_SELECTION_REQUEST:
|
||
case GDK_SELECTION_NOTIFY:
|
||
tm = event->selection.time; break;
|
||
case GDK_PROXIMITY_IN:
|
||
case GDK_PROXIMITY_OUT:
|
||
tm = event->proximity.time; break;
|
||
default: /* use current time */
|
||
break;
|
||
}
|
||
|
||
return tm;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_get_event_actions (const GdkEvent *event,
|
||
gint button,
|
||
GdkDragAction actions,
|
||
GdkDragAction *suggested_action,
|
||
GdkDragAction *possible_actions)
|
||
{
|
||
*suggested_action = 0;
|
||
*possible_actions = 0;
|
||
|
||
if (event)
|
||
{
|
||
GdkModifierType state = 0;
|
||
|
||
switch (event->type)
|
||
{
|
||
case GDK_MOTION_NOTIFY:
|
||
state = event->motion.state;
|
||
break;
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_2BUTTON_PRESS:
|
||
case GDK_3BUTTON_PRESS:
|
||
case GDK_BUTTON_RELEASE:
|
||
state = event->button.state;
|
||
break;
|
||
case GDK_KEY_PRESS:
|
||
case GDK_KEY_RELEASE:
|
||
state = event->key.state;
|
||
break;
|
||
case GDK_ENTER_NOTIFY:
|
||
case GDK_LEAVE_NOTIFY:
|
||
state = event->crossing.state;
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
|
||
if ((button == GDK_BUTTON_MIDDLE || button == GDK_BUTTON_SECONDARY) && (actions & GDK_ACTION_ASK))
|
||
{
|
||
*suggested_action = GDK_ACTION_ASK;
|
||
*possible_actions = actions;
|
||
}
|
||
else if (state & (GDK_SHIFT_MASK | GDK_CONTROL_MASK))
|
||
{
|
||
if ((state & GDK_SHIFT_MASK) && (state & GDK_CONTROL_MASK))
|
||
{
|
||
if (actions & GDK_ACTION_LINK)
|
||
{
|
||
*suggested_action = GDK_ACTION_LINK;
|
||
*possible_actions = GDK_ACTION_LINK;
|
||
}
|
||
}
|
||
else if (state & GDK_CONTROL_MASK)
|
||
{
|
||
if (actions & GDK_ACTION_COPY)
|
||
{
|
||
*suggested_action = GDK_ACTION_COPY;
|
||
*possible_actions = GDK_ACTION_COPY;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
if (actions & GDK_ACTION_MOVE)
|
||
{
|
||
*suggested_action = GDK_ACTION_MOVE;
|
||
*possible_actions = GDK_ACTION_MOVE;
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
*possible_actions = actions;
|
||
|
||
if ((state & (GDK_MOD1_MASK)) && (actions & GDK_ACTION_ASK))
|
||
*suggested_action = GDK_ACTION_ASK;
|
||
else if (actions & GDK_ACTION_COPY)
|
||
*suggested_action = GDK_ACTION_COPY;
|
||
else if (actions & GDK_ACTION_MOVE)
|
||
*suggested_action = GDK_ACTION_MOVE;
|
||
else if (actions & GDK_ACTION_LINK)
|
||
*suggested_action = GDK_ACTION_LINK;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
*possible_actions = actions;
|
||
|
||
if (actions & GDK_ACTION_COPY)
|
||
*suggested_action = GDK_ACTION_COPY;
|
||
else if (actions & GDK_ACTION_MOVE)
|
||
*suggested_action = GDK_ACTION_MOVE;
|
||
else if (actions & GDK_ACTION_LINK)
|
||
*suggested_action = GDK_ACTION_LINK;
|
||
}
|
||
}
|
||
|
||
static void
|
||
ensure_drag_cursor_pixbuf (int i)
|
||
{
|
||
if (drag_cursors[i].pixbuf == NULL)
|
||
{
|
||
char *path = g_strconcat ("/org/gtk/libgtk/cursor/", drag_cursors[i].name, ".png", NULL);
|
||
GInputStream *stream = g_resources_open_stream (path, 0, NULL);
|
||
if (stream != NULL)
|
||
{
|
||
drag_cursors[i].pixbuf = gdk_pixbuf_new_from_stream (stream, NULL, NULL);
|
||
g_object_unref (stream);
|
||
}
|
||
g_free (path);
|
||
}
|
||
}
|
||
|
||
static GdkCursor *
|
||
gtk_drag_get_cursor (GtkWidget *widget,
|
||
GdkDisplay *display,
|
||
GdkDragAction action,
|
||
GtkDragSourceInfo *info)
|
||
{
|
||
gint i;
|
||
|
||
/* reconstruct the cursors for each new drag (thus !info),
|
||
* to catch cursor theme changes
|
||
*/
|
||
if (!info)
|
||
{
|
||
for (i = 0 ; i < G_N_ELEMENTS (drag_cursors) - 1; i++)
|
||
g_clear_object (&drag_cursors[i].cursor);
|
||
}
|
||
|
||
for (i = 0 ; i < G_N_ELEMENTS (drag_cursors) - 1; i++)
|
||
if (drag_cursors[i].action == action)
|
||
break;
|
||
|
||
if (drag_cursors[i].cursor != NULL)
|
||
{
|
||
if (display != gdk_cursor_get_display (drag_cursors[i].cursor))
|
||
g_clear_object (&drag_cursors[i].cursor);
|
||
}
|
||
|
||
if (drag_cursors[i].cursor == NULL)
|
||
drag_cursors[i].cursor = gdk_cursor_new_from_name (display, drag_cursors[i].name);
|
||
|
||
if (drag_cursors[i].cursor == NULL)
|
||
{
|
||
ensure_drag_cursor_pixbuf (i);
|
||
drag_cursors[i].cursor = gdk_cursor_new_from_pixbuf (display, drag_cursors[i].pixbuf, 0, 0);
|
||
}
|
||
|
||
return drag_cursors[i].cursor;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_update_cursor (GtkDragSourceInfo *info)
|
||
{
|
||
GdkCursor *cursor;
|
||
gint i;
|
||
|
||
if (!info->have_grab)
|
||
return;
|
||
|
||
for (i = 0 ; i < G_N_ELEMENTS (drag_cursors) - 1; i++)
|
||
if (info->cursor == drag_cursors[i].cursor)
|
||
break;
|
||
|
||
if (i == G_N_ELEMENTS (drag_cursors))
|
||
return;
|
||
|
||
cursor = gtk_drag_get_cursor (info->widget,
|
||
gdk_cursor_get_display (info->cursor),
|
||
drag_cursors[i].action, info);
|
||
|
||
if (cursor != info->cursor)
|
||
{
|
||
GdkDevice *pointer;
|
||
|
||
pointer = gdk_drag_context_get_device (info->context);
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
gdk_device_grab (pointer,
|
||
gtk_widget_get_window (info->ipc_widget),
|
||
GDK_OWNERSHIP_APPLICATION, FALSE,
|
||
GDK_POINTER_MOTION_MASK | GDK_BUTTON_RELEASE_MASK,
|
||
cursor, info->grab_time);
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
info->cursor = cursor;
|
||
}
|
||
}
|
||
|
||
/********************
|
||
* Destination side *
|
||
********************/
|
||
|
||
/**
|
||
* gtk_drag_get_data: (method)
|
||
* @widget: the widget that will receive the
|
||
* #GtkWidget::drag-data-received signal
|
||
* @context: the drag context
|
||
* @target: the target (form of the data) to retrieve
|
||
* @time_: a timestamp for retrieving the data. This will
|
||
* generally be the time received in a #GtkWidget::drag-motion
|
||
* or #GtkWidget::drag-drop signal
|
||
*
|
||
* Gets the data associated with a drag. When the data
|
||
* is received or the retrieval fails, GTK+ will emit a
|
||
* #GtkWidget::drag-data-received signal. Failure of the retrieval
|
||
* is indicated by the length field of the @selection_data
|
||
* signal parameter being negative. However, when gtk_drag_get_data()
|
||
* is called implicitely because the %GTK_DEST_DEFAULT_DROP was set,
|
||
* then the widget will not receive notification of failed
|
||
* drops.
|
||
*/
|
||
void
|
||
gtk_drag_get_data (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
GdkAtom target,
|
||
guint32 time_)
|
||
{
|
||
GtkWidget *selection_widget;
|
||
|
||
g_return_if_fail (GTK_IS_WIDGET (widget));
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
|
||
selection_widget = gtk_drag_get_ipc_widget (widget);
|
||
|
||
g_object_ref (context);
|
||
g_object_ref (widget);
|
||
|
||
g_signal_connect (selection_widget, "selection-received",
|
||
G_CALLBACK (gtk_drag_selection_received), widget);
|
||
|
||
g_object_set_data (G_OBJECT (selection_widget), I_("drag-context"), context);
|
||
|
||
gtk_selection_convert (selection_widget,
|
||
gdk_drag_get_selection (context),
|
||
target,
|
||
time_);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_get_source_widget:
|
||
* @context: a (destination side) drag context
|
||
*
|
||
* Determines the source widget for a drag.
|
||
*
|
||
* Returns: (nullable) (transfer none): if the drag is occurring
|
||
* within a single application, a pointer to the source widget.
|
||
* Otherwise, %NULL.
|
||
*/
|
||
GtkWidget *
|
||
gtk_drag_get_source_widget (GdkDragContext *context)
|
||
{
|
||
GSList *tmp_list;
|
||
|
||
g_return_val_if_fail (GDK_IS_DRAG_CONTEXT (context), NULL);
|
||
|
||
tmp_list = source_widgets;
|
||
while (tmp_list)
|
||
{
|
||
GtkWidget *ipc_widget = tmp_list->data;
|
||
|
||
if (gtk_widget_get_window (ipc_widget) == gdk_drag_context_get_source_window (context))
|
||
{
|
||
GtkDragSourceInfo *info;
|
||
info = g_object_get_data (G_OBJECT (ipc_widget), "gtk-info");
|
||
|
||
return info ? info->widget : NULL;
|
||
}
|
||
|
||
tmp_list = tmp_list->next;
|
||
}
|
||
|
||
return NULL;
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_finish:
|
||
* @context: the drag context
|
||
* @success: a flag indicating whether the drop was successful
|
||
* @del: a flag indicating whether the source should delete the
|
||
* original data. (This should be %TRUE for a move)
|
||
* @time_: the timestamp from the #GtkWidget::drag-drop signal
|
||
*
|
||
* Informs the drag source that the drop is finished, and
|
||
* that the data of the drag will no longer be required.
|
||
*/
|
||
void
|
||
gtk_drag_finish (GdkDragContext *context,
|
||
gboolean success,
|
||
gboolean del,
|
||
guint32 time)
|
||
{
|
||
GdkAtom target = GDK_NONE;
|
||
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
|
||
if (success && del)
|
||
{
|
||
target = gdk_atom_intern_static_string ("DELETE");
|
||
}
|
||
|
||
if (target != GDK_NONE)
|
||
{
|
||
GtkWidget *selection_widget = gtk_drag_get_ipc_widget_for_screen (gdk_window_get_screen (gdk_drag_context_get_source_window (context)));
|
||
|
||
g_object_ref (context);
|
||
|
||
g_object_set_data (G_OBJECT (selection_widget), I_("drag-context"), context);
|
||
g_signal_connect (selection_widget, "selection-received",
|
||
G_CALLBACK (gtk_drag_selection_received),
|
||
NULL);
|
||
|
||
gtk_selection_convert (selection_widget,
|
||
gdk_drag_get_selection (context),
|
||
target,
|
||
time);
|
||
}
|
||
|
||
if (!(success && del))
|
||
gdk_drop_finish (context, success, time);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_highlight: (method)
|
||
* @widget: a widget to highlight
|
||
*
|
||
* Highlights a widget as a currently hovered drop target.
|
||
* To end the highlight, call gtk_drag_unhighlight().
|
||
* GTK+ calls this automatically if %GTK_DEST_DEFAULT_HIGHLIGHT is set.
|
||
*/
|
||
void
|
||
gtk_drag_highlight (GtkWidget *widget)
|
||
{
|
||
g_return_if_fail (GTK_IS_WIDGET (widget));
|
||
|
||
gtk_widget_set_state_flags (widget, GTK_STATE_FLAG_DROP_ACTIVE, FALSE);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_unhighlight: (method)
|
||
* @widget: a widget to remove the highlight from
|
||
*
|
||
* Removes a highlight set by gtk_drag_highlight() from
|
||
* a widget.
|
||
*/
|
||
void
|
||
gtk_drag_unhighlight (GtkWidget *widget)
|
||
{
|
||
g_return_if_fail (GTK_IS_WIDGET (widget));
|
||
|
||
gtk_widget_unset_state_flags (widget, GTK_STATE_FLAG_DROP_ACTIVE);
|
||
}
|
||
|
||
/*
|
||
* _gtk_drag_dest_handle_event:
|
||
* @toplevel: Toplevel widget that received the event
|
||
* @event: the event to handle
|
||
*
|
||
* Called from widget event handling code on Drag events
|
||
* for destinations.
|
||
*/
|
||
void
|
||
_gtk_drag_dest_handle_event (GtkWidget *toplevel,
|
||
GdkEvent *event)
|
||
{
|
||
GtkDragDestInfo *info;
|
||
GdkDragContext *context;
|
||
|
||
g_return_if_fail (toplevel != NULL);
|
||
g_return_if_fail (event != NULL);
|
||
|
||
context = event->dnd.context;
|
||
|
||
info = gtk_drag_get_dest_info (context, TRUE);
|
||
|
||
/* Find the widget for the event */
|
||
switch (event->type)
|
||
{
|
||
case GDK_DRAG_ENTER:
|
||
break;
|
||
|
||
case GDK_DRAG_LEAVE:
|
||
if (info->widget)
|
||
{
|
||
gtk_drag_dest_leave (info->widget, context, event->dnd.time);
|
||
gtk_drag_dest_set_widget (info, NULL);
|
||
}
|
||
break;
|
||
|
||
case GDK_DRAG_MOTION:
|
||
case GDK_DROP_START:
|
||
{
|
||
GdkWindow *window;
|
||
gint tx, ty;
|
||
gboolean found;
|
||
|
||
if (event->type == GDK_DROP_START)
|
||
{
|
||
info->dropped = TRUE;
|
||
/* We send a leave here so that the widget unhighlights
|
||
* properly.
|
||
*/
|
||
if (info->widget)
|
||
{
|
||
gtk_drag_dest_leave (info->widget, context, event->dnd.time);
|
||
gtk_drag_dest_set_widget (info, NULL);
|
||
}
|
||
}
|
||
|
||
window = gtk_widget_get_window (toplevel);
|
||
|
||
#ifdef GDK_WINDOWING_X11
|
||
/* Hackaround for: http://bugzilla.gnome.org/show_bug.cgi?id=136112
|
||
*
|
||
* Currently gdk_window_get_position doesn't provide reliable
|
||
* information for embedded windows, so we call the much more
|
||
* expensive gdk_window_get_origin().
|
||
*/
|
||
if (GTK_IS_PLUG (toplevel))
|
||
gdk_window_get_origin (window, &tx, &ty);
|
||
else
|
||
#endif /* GDK_WINDOWING_X11 */
|
||
gdk_window_get_position (window, &tx, &ty);
|
||
|
||
found = gtk_drag_find_widget (toplevel,
|
||
context,
|
||
info,
|
||
event->dnd.x_root - tx,
|
||
event->dnd.y_root - ty,
|
||
event->dnd.time,
|
||
(event->type == GDK_DRAG_MOTION) ?
|
||
gtk_drag_dest_motion :
|
||
gtk_drag_dest_drop);
|
||
|
||
if (info->widget && !found)
|
||
{
|
||
gtk_drag_dest_leave (info->widget, context, event->dnd.time);
|
||
gtk_drag_dest_set_widget (info, NULL);
|
||
}
|
||
|
||
/* Send a reply.
|
||
*/
|
||
if (event->type == GDK_DRAG_MOTION)
|
||
{
|
||
if (!found)
|
||
gdk_drag_status (context, 0, event->dnd.time);
|
||
}
|
||
else if (event->type == GDK_DROP_START && !info->proxy_source)
|
||
{
|
||
gdk_drop_reply (context, found, event->dnd.time);
|
||
}
|
||
}
|
||
break;
|
||
|
||
default:
|
||
g_assert_not_reached ();
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_drag_selection_received (GtkWidget *widget,
|
||
GtkSelectionData *selection_data,
|
||
guint time,
|
||
gpointer data)
|
||
{
|
||
GdkDragContext *context;
|
||
GtkDragDestInfo *info;
|
||
GtkWidget *drop_widget;
|
||
GdkAtom target;
|
||
|
||
drop_widget = data;
|
||
|
||
context = g_object_get_data (G_OBJECT (widget), "drag-context");
|
||
info = gtk_drag_get_dest_info (context, FALSE);
|
||
|
||
if (info->proxy_data &&
|
||
gtk_selection_data_get_target (info->proxy_data) == gtk_selection_data_get_target (selection_data))
|
||
{
|
||
gtk_selection_data_set (info->proxy_data,
|
||
gtk_selection_data_get_data_type (selection_data),
|
||
gtk_selection_data_get_format (selection_data),
|
||
gtk_selection_data_get_data (selection_data),
|
||
gtk_selection_data_get_length (selection_data));
|
||
gtk_main_quit ();
|
||
return;
|
||
}
|
||
|
||
target = gtk_selection_data_get_target (selection_data);
|
||
if (target == gdk_atom_intern_static_string ("DELETE"))
|
||
{
|
||
gtk_drag_finish (context, TRUE, FALSE, time);
|
||
}
|
||
else if ((target == gdk_atom_intern_static_string ("XmTRANSFER_SUCCESS")) ||
|
||
(target == gdk_atom_intern_static_string ("XmTRANSFER_FAILURE")))
|
||
{
|
||
/* Do nothing */
|
||
}
|
||
else
|
||
{
|
||
GtkDragDestSite *site;
|
||
|
||
site = g_object_get_data (G_OBJECT (drop_widget), "gtk-drag-dest");
|
||
|
||
if (site && site->target_list)
|
||
{
|
||
guint target_info;
|
||
|
||
if (gtk_target_list_find (site->target_list,
|
||
target,
|
||
&target_info))
|
||
{
|
||
if (!(site->flags & GTK_DEST_DEFAULT_DROP) ||
|
||
gtk_selection_data_get_length (selection_data) >= 0)
|
||
g_signal_emit_by_name (drop_widget,
|
||
"drag-data-received",
|
||
context, info->drop_x, info->drop_y,
|
||
selection_data,
|
||
target_info, time);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
g_signal_emit_by_name (drop_widget,
|
||
"drag-data-received",
|
||
context, info->drop_x, info->drop_y,
|
||
selection_data,
|
||
0, time);
|
||
}
|
||
|
||
if (site && site->flags & GTK_DEST_DEFAULT_DROP)
|
||
{
|
||
|
||
gtk_drag_finish (context,
|
||
(gtk_selection_data_get_length (selection_data) >= 0),
|
||
(gdk_drag_context_get_selected_action (context) == GDK_ACTION_MOVE),
|
||
time);
|
||
}
|
||
|
||
g_object_unref (drop_widget);
|
||
}
|
||
|
||
g_signal_handlers_disconnect_by_func (widget,
|
||
gtk_drag_selection_received,
|
||
data);
|
||
|
||
g_object_set_data (G_OBJECT (widget), I_("drag-context"), NULL);
|
||
g_object_unref (context);
|
||
|
||
gtk_drag_release_ipc_widget (widget);
|
||
}
|
||
|
||
static gboolean
|
||
gtk_drag_find_widget (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
GtkDragDestInfo *info,
|
||
gint x,
|
||
gint y,
|
||
guint32 time,
|
||
GtkDragDestCallback callback)
|
||
{
|
||
if (!gtk_widget_get_mapped (widget) ||
|
||
!gtk_widget_get_sensitive (widget))
|
||
return FALSE;
|
||
|
||
/* Get the widget at the pointer coordinates and travel up
|
||
* the widget hierarchy from there.
|
||
*/
|
||
widget = _gtk_widget_find_at_coords (gtk_widget_get_window (widget),
|
||
x, y, &x, &y);
|
||
if (!widget)
|
||
return FALSE;
|
||
|
||
while (widget)
|
||
{
|
||
GtkWidget *parent;
|
||
GList *hierarchy = NULL;
|
||
gboolean found = FALSE;
|
||
|
||
if (!gtk_widget_get_mapped (widget))
|
||
return FALSE;
|
||
|
||
if (gtk_widget_get_state_flags (widget) & GTK_STATE_FLAG_INSENSITIVE)
|
||
{
|
||
widget = gtk_widget_get_parent (widget);
|
||
continue;
|
||
}
|
||
|
||
/* need to reference the entire hierarchy temporarily in case the
|
||
* ::drag-motion/::drag-drop callbacks change the widget hierarchy.
|
||
*/
|
||
for (parent = widget;
|
||
parent;
|
||
parent = gtk_widget_get_parent (parent))
|
||
{
|
||
hierarchy = g_list_prepend (hierarchy, g_object_ref (parent));
|
||
}
|
||
|
||
/* If the current widget is registered as a drop site, check to
|
||
* emit "drag-motion" to check if we are actually in a drop
|
||
* site.
|
||
*/
|
||
if (g_object_get_data (G_OBJECT (widget), "gtk-drag-dest"))
|
||
{
|
||
found = callback (widget, context, x, y, time);
|
||
|
||
/* If so, send a "drag-leave" to the last widget */
|
||
if (found && info->widget != widget)
|
||
{
|
||
if (info->widget)
|
||
gtk_drag_dest_leave (info->widget, context, time);
|
||
|
||
gtk_drag_dest_set_widget (info, widget);
|
||
}
|
||
}
|
||
|
||
if (!found)
|
||
{
|
||
/* Get the parent before unreffing the hierarchy because
|
||
* invoking the callback might have destroyed the widget
|
||
*/
|
||
parent = gtk_widget_get_parent (widget);
|
||
|
||
/* The parent might be going away when unreffing the
|
||
* hierarchy, so also protect againt that
|
||
*/
|
||
if (parent)
|
||
g_object_add_weak_pointer (G_OBJECT (parent), (gpointer *) &parent);
|
||
}
|
||
|
||
g_list_free_full (hierarchy, g_object_unref);
|
||
|
||
if (found)
|
||
return TRUE;
|
||
|
||
if (parent)
|
||
g_object_remove_weak_pointer (G_OBJECT (parent), (gpointer *) &parent);
|
||
else
|
||
return FALSE;
|
||
|
||
if (!gtk_widget_translate_coordinates (widget, parent, x, y, &x, &y))
|
||
return FALSE;
|
||
|
||
widget = parent;
|
||
}
|
||
|
||
return FALSE;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_proxy_begin (GtkWidget *widget,
|
||
GtkDragDestInfo *dest_info,
|
||
guint32 time)
|
||
{
|
||
GtkDragSourceInfo *source_info;
|
||
GList *tmp_list;
|
||
GdkDragContext *context;
|
||
GtkWidget *ipc_widget;
|
||
|
||
if (dest_info->proxy_source)
|
||
{
|
||
gdk_drag_abort (dest_info->proxy_source->context, time);
|
||
gtk_drag_source_info_destroy (dest_info->proxy_source);
|
||
dest_info->proxy_source = NULL;
|
||
}
|
||
|
||
ipc_widget = gtk_drag_get_ipc_widget (widget);
|
||
context = gdk_drag_begin (gtk_widget_get_window (ipc_widget),
|
||
gdk_drag_context_list_targets (dest_info->context));
|
||
|
||
source_info = gtk_drag_get_source_info (context, TRUE);
|
||
|
||
source_info->ipc_widget = ipc_widget;
|
||
source_info->widget = g_object_ref (widget);
|
||
|
||
source_info->target_list = gtk_target_list_new (NULL, 0);
|
||
tmp_list = gdk_drag_context_list_targets (dest_info->context);
|
||
while (tmp_list)
|
||
{
|
||
gtk_target_list_add (source_info->target_list,
|
||
GDK_POINTER_TO_ATOM (tmp_list->data), 0, 0);
|
||
tmp_list = tmp_list->next;
|
||
}
|
||
|
||
source_info->proxy_dest = dest_info;
|
||
|
||
g_signal_connect (ipc_widget,
|
||
"selection-get",
|
||
G_CALLBACK (gtk_drag_selection_get),
|
||
source_info);
|
||
|
||
dest_info->proxy_source = source_info;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_dest_set_widget (GtkDragDestInfo *info,
|
||
GtkWidget *widget)
|
||
{
|
||
if (info->widget)
|
||
g_object_remove_weak_pointer (G_OBJECT (info->widget), (gpointer *) &info->widget);
|
||
|
||
info->widget = widget;
|
||
|
||
if (info->widget)
|
||
g_object_add_weak_pointer (G_OBJECT (info->widget), (gpointer *) &info->widget);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_dest_info_destroy (gpointer data)
|
||
{
|
||
GtkDragDestInfo *info = (GtkDragDestInfo *)data;
|
||
|
||
gtk_drag_dest_set_widget (info, NULL);
|
||
|
||
g_slice_free (GtkDragDestInfo, data);
|
||
}
|
||
|
||
static GtkDragDestInfo *
|
||
gtk_drag_get_dest_info (GdkDragContext *context,
|
||
gboolean create)
|
||
{
|
||
GtkDragDestInfo *info;
|
||
static GQuark info_quark = 0;
|
||
if (!info_quark)
|
||
info_quark = g_quark_from_static_string ("gtk-dest-info");
|
||
|
||
info = g_object_get_qdata (G_OBJECT (context), info_quark);
|
||
if (!info && create)
|
||
{
|
||
info = g_slice_new0 (GtkDragDestInfo);
|
||
info->context = context;
|
||
g_object_set_qdata_full (G_OBJECT (context), info_quark,
|
||
info, gtk_drag_dest_info_destroy);
|
||
}
|
||
|
||
return info;
|
||
}
|
||
|
||
static GQuark dest_info_quark = 0;
|
||
|
||
static GtkDragSourceInfo *
|
||
gtk_drag_get_source_info (GdkDragContext *context,
|
||
gboolean create)
|
||
{
|
||
GtkDragSourceInfo *info;
|
||
if (!dest_info_quark)
|
||
dest_info_quark = g_quark_from_static_string ("gtk-source-info");
|
||
|
||
info = g_object_get_qdata (G_OBJECT (context), dest_info_quark);
|
||
if (!info && create)
|
||
{
|
||
info = g_new0 (GtkDragSourceInfo, 1);
|
||
info->context = context;
|
||
g_object_set_qdata (G_OBJECT (context), dest_info_quark, info);
|
||
}
|
||
|
||
return info;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_clear_source_info (GdkDragContext *context)
|
||
{
|
||
g_object_set_qdata (G_OBJECT (context), dest_info_quark, NULL);
|
||
}
|
||
|
||
/*
|
||
* Default drag handlers
|
||
*/
|
||
static void
|
||
gtk_drag_dest_leave (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
guint time)
|
||
{
|
||
GtkDragDestSite *site;
|
||
|
||
site = g_object_get_data (G_OBJECT (widget), "gtk-drag-dest");
|
||
g_return_if_fail (site != NULL);
|
||
|
||
if (site->do_proxy)
|
||
{
|
||
GtkDragDestInfo *info = gtk_drag_get_dest_info (context, FALSE);
|
||
|
||
if (info->proxy_source && info->proxy_source->widget == widget && !info->dropped)
|
||
{
|
||
gdk_drag_abort (info->proxy_source->context, time);
|
||
gtk_drag_source_info_destroy (info->proxy_source);
|
||
info->proxy_source = NULL;
|
||
}
|
||
|
||
return;
|
||
}
|
||
else
|
||
{
|
||
if ((site->flags & GTK_DEST_DEFAULT_HIGHLIGHT) && site->have_drag)
|
||
gtk_drag_unhighlight (widget);
|
||
|
||
if (!(site->flags & GTK_DEST_DEFAULT_MOTION) || site->have_drag ||
|
||
site->track_motion)
|
||
g_signal_emit_by_name (widget, "drag-leave", context, time);
|
||
|
||
site->have_drag = FALSE;
|
||
}
|
||
}
|
||
|
||
static gboolean
|
||
gtk_drag_dest_motion (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
gint x,
|
||
gint y,
|
||
guint time)
|
||
{
|
||
GtkDragDestSite *site;
|
||
GdkDragAction action = 0;
|
||
gboolean retval;
|
||
|
||
site = g_object_get_data (G_OBJECT (widget), "gtk-drag-dest");
|
||
g_return_val_if_fail (site != NULL, FALSE);
|
||
|
||
if (site->do_proxy)
|
||
{
|
||
GdkAtom selection;
|
||
GdkEvent *current_event;
|
||
GdkWindow *dest_window;
|
||
GdkDragProtocol proto;
|
||
|
||
GtkDragDestInfo *info = gtk_drag_get_dest_info (context, FALSE);
|
||
|
||
if (!info->proxy_source || info->proxy_source->widget != widget)
|
||
gtk_drag_proxy_begin (widget, info, time);
|
||
|
||
current_event = gtk_get_current_event ();
|
||
|
||
if (site->proxy_window)
|
||
{
|
||
dest_window = site->proxy_window;
|
||
proto = site->proxy_protocol;
|
||
}
|
||
else
|
||
{
|
||
gdk_drag_find_window_for_screen (info->proxy_source->context,
|
||
NULL,
|
||
gdk_window_get_screen (current_event->dnd.window),
|
||
current_event->dnd.x_root,
|
||
current_event->dnd.y_root,
|
||
&dest_window, &proto);
|
||
}
|
||
|
||
gdk_drag_motion (info->proxy_source->context,
|
||
dest_window, proto,
|
||
current_event->dnd.x_root,
|
||
current_event->dnd.y_root,
|
||
gdk_drag_context_get_suggested_action (context),
|
||
gdk_drag_context_get_actions (context),
|
||
time);
|
||
|
||
if (!site->proxy_window && dest_window)
|
||
g_object_unref (dest_window);
|
||
|
||
selection = gdk_drag_get_selection (info->proxy_source->context);
|
||
if (selection &&
|
||
selection != gdk_drag_get_selection (info->context))
|
||
gtk_drag_source_check_selection (info->proxy_source, selection, time);
|
||
|
||
gdk_event_free (current_event);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
if (site->track_motion || site->flags & GTK_DEST_DEFAULT_MOTION)
|
||
{
|
||
if (gdk_drag_context_get_suggested_action (context) & site->actions)
|
||
action = gdk_drag_context_get_suggested_action (context);
|
||
else
|
||
{
|
||
gint i;
|
||
|
||
for (i = 0; i < 8; i++)
|
||
{
|
||
if ((site->actions & (1 << i)) &&
|
||
(gdk_drag_context_get_actions (context) & (1 << i)))
|
||
{
|
||
action = (1 << i);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (action && gtk_drag_dest_find_target (widget, context, NULL))
|
||
{
|
||
if (!site->have_drag)
|
||
{
|
||
site->have_drag = TRUE;
|
||
if (site->flags & GTK_DEST_DEFAULT_HIGHLIGHT)
|
||
gtk_drag_highlight (widget);
|
||
}
|
||
|
||
gdk_drag_status (context, action, time);
|
||
}
|
||
else
|
||
{
|
||
gdk_drag_status (context, 0, time);
|
||
if (!site->track_motion)
|
||
return TRUE;
|
||
}
|
||
}
|
||
|
||
g_signal_emit_by_name (widget, "drag-motion",
|
||
context, x, y, time, &retval);
|
||
|
||
return (site->flags & GTK_DEST_DEFAULT_MOTION) ? TRUE : retval;
|
||
}
|
||
|
||
static gboolean
|
||
gtk_drag_dest_drop (GtkWidget *widget,
|
||
GdkDragContext *context,
|
||
gint x,
|
||
gint y,
|
||
guint time)
|
||
{
|
||
GtkDragDestSite *site;
|
||
GtkDragDestInfo *info;
|
||
|
||
site = g_object_get_data (G_OBJECT (widget), "gtk-drag-dest");
|
||
g_return_val_if_fail (site != NULL, FALSE);
|
||
|
||
info = gtk_drag_get_dest_info (context, FALSE);
|
||
g_return_val_if_fail (info != NULL, FALSE);
|
||
|
||
info->drop_x = x;
|
||
info->drop_y = y;
|
||
|
||
if (site->do_proxy)
|
||
{
|
||
if (info->proxy_source ||
|
||
(gdk_drag_context_get_protocol (info->context) == GDK_DRAG_PROTO_ROOTWIN))
|
||
{
|
||
gtk_drag_drop (info->proxy_source, time);
|
||
}
|
||
else
|
||
{
|
||
/* We need to synthesize a motion event, wait for a status,
|
||
* and, if we get a good one, do a drop.
|
||
*/
|
||
|
||
GdkEvent *current_event;
|
||
GdkAtom selection;
|
||
GdkWindow *dest_window;
|
||
GdkDragProtocol proto;
|
||
|
||
gtk_drag_proxy_begin (widget, info, time);
|
||
info->proxy_drop_wait = TRUE;
|
||
info->proxy_drop_time = time;
|
||
|
||
current_event = gtk_get_current_event ();
|
||
|
||
if (site->proxy_window)
|
||
{
|
||
dest_window = site->proxy_window;
|
||
proto = site->proxy_protocol;
|
||
}
|
||
else
|
||
{
|
||
gdk_drag_find_window_for_screen (info->proxy_source->context,
|
||
NULL,
|
||
gdk_window_get_screen (current_event->dnd.window),
|
||
current_event->dnd.x_root,
|
||
current_event->dnd.y_root,
|
||
&dest_window, &proto);
|
||
}
|
||
|
||
gdk_drag_motion (info->proxy_source->context,
|
||
dest_window, proto,
|
||
current_event->dnd.x_root,
|
||
current_event->dnd.y_root,
|
||
gdk_drag_context_get_suggested_action (context),
|
||
gdk_drag_context_get_actions (context),
|
||
time);
|
||
|
||
if (!site->proxy_window && dest_window)
|
||
g_object_unref (dest_window);
|
||
|
||
selection = gdk_drag_get_selection (info->proxy_source->context);
|
||
if (selection &&
|
||
selection != gdk_drag_get_selection (info->context))
|
||
gtk_drag_source_check_selection (info->proxy_source, selection, time);
|
||
|
||
gdk_event_free (current_event);
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
else
|
||
{
|
||
gboolean retval;
|
||
|
||
if (site->flags & GTK_DEST_DEFAULT_DROP)
|
||
{
|
||
GdkAtom target = gtk_drag_dest_find_target (widget, context, NULL);
|
||
|
||
if (target == GDK_NONE)
|
||
{
|
||
gtk_drag_finish (context, FALSE, FALSE, time);
|
||
return TRUE;
|
||
}
|
||
else
|
||
gtk_drag_get_data (widget, context, target, time);
|
||
}
|
||
|
||
g_signal_emit_by_name (widget, "drag-drop",
|
||
context, x, y, time, &retval);
|
||
|
||
return (site->flags & GTK_DEST_DEFAULT_DROP) ? TRUE : retval;
|
||
}
|
||
}
|
||
|
||
/***************
|
||
* Source side *
|
||
***************/
|
||
|
||
|
||
static gboolean
|
||
gtk_drag_is_managed (GtkWidget *source_widget)
|
||
{
|
||
return
|
||
#ifdef GDK_WINDOWING_X11
|
||
GDK_IS_X11_DISPLAY (gtk_widget_get_display (source_widget)) ||
|
||
#endif
|
||
#ifdef GDK_WINDOWING_WAYLAND
|
||
GDK_IS_WAYLAND_DISPLAY (gtk_widget_get_display (source_widget)) ||
|
||
#endif
|
||
#ifdef GDK_WINDOWING_WIN32
|
||
GDK_IS_WIN32_DISPLAY (gtk_widget_get_display (source_widget)) ||
|
||
#endif
|
||
FALSE;
|
||
}
|
||
|
||
/* Like gtk_drag_begin(), but also communicates the need to
|
||
* create an icon for the drag operation back to the caller.
|
||
* If the caller passes out_needs_icon == NULL, it means that
|
||
* the caller does not care.
|
||
*/
|
||
GdkDragContext *
|
||
gtk_drag_begin_internal (GtkWidget *widget,
|
||
gboolean *out_needs_icon,
|
||
GtkTargetList *target_list,
|
||
GdkDragAction actions,
|
||
gint button,
|
||
const GdkEvent *event,
|
||
int x,
|
||
int y)
|
||
{
|
||
GtkDragSourceInfo *info;
|
||
GList *targets = NULL;
|
||
GList *tmp_list;
|
||
guint32 time = GDK_CURRENT_TIME;
|
||
GdkDragAction possible_actions, suggested_action;
|
||
GdkDragContext *context;
|
||
GtkWidget *ipc_widget;
|
||
GdkCursor *cursor;
|
||
GdkDevice *pointer, *keyboard;
|
||
GdkWindow *ipc_window;
|
||
gint start_x, start_y;
|
||
GdkAtom selection;
|
||
gboolean managed;
|
||
|
||
managed = gtk_drag_is_managed (widget);
|
||
|
||
pointer = keyboard = NULL;
|
||
ipc_widget = gtk_drag_get_ipc_widget (widget);
|
||
|
||
gtk_drag_get_event_actions (event, button, actions,
|
||
&suggested_action, &possible_actions);
|
||
|
||
cursor = gtk_drag_get_cursor (widget,
|
||
gtk_widget_get_display (widget),
|
||
suggested_action,
|
||
NULL);
|
||
|
||
if (event)
|
||
{
|
||
time = gdk_event_get_time (event);
|
||
if (time == GDK_CURRENT_TIME)
|
||
time = gtk_get_current_event_time ();
|
||
|
||
pointer = gdk_event_get_device (event);
|
||
|
||
if (gdk_device_get_source (pointer) == GDK_SOURCE_KEYBOARD)
|
||
{
|
||
keyboard = pointer;
|
||
pointer = gdk_device_get_associated_device (keyboard);
|
||
}
|
||
else
|
||
keyboard = gdk_device_get_associated_device (pointer);
|
||
}
|
||
else
|
||
{
|
||
GdkSeat *seat;
|
||
|
||
seat = gdk_display_get_default_seat (gtk_widget_get_display (widget));
|
||
pointer = gdk_seat_get_pointer (seat);
|
||
keyboard = gdk_seat_get_keyboard (seat);
|
||
}
|
||
|
||
if (!pointer)
|
||
return NULL;
|
||
|
||
ipc_window = gtk_widget_get_window (ipc_widget);
|
||
|
||
if (!managed)
|
||
{
|
||
gboolean grabbed;
|
||
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
grabbed = gdk_device_grab (pointer, ipc_window,
|
||
GDK_OWNERSHIP_APPLICATION, FALSE,
|
||
GDK_POINTER_MOTION_MASK |
|
||
GDK_BUTTON_RELEASE_MASK,
|
||
cursor, time) == GDK_GRAB_SUCCESS;
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
|
||
if (!grabbed)
|
||
{
|
||
gtk_drag_release_ipc_widget (ipc_widget);
|
||
return NULL;
|
||
}
|
||
|
||
if (keyboard)
|
||
grab_dnd_keys (ipc_widget, keyboard, time);
|
||
|
||
/* We use a GTK grab here to override any grabs that the widget
|
||
* we are dragging from might have held
|
||
*/
|
||
gtk_device_grab_add (ipc_widget, pointer, FALSE);
|
||
}
|
||
|
||
tmp_list = g_list_last (target_list->list);
|
||
while (tmp_list)
|
||
{
|
||
GtkTargetPair *pair = tmp_list->data;
|
||
targets = g_list_prepend (targets,
|
||
GINT_TO_POINTER (pair->target));
|
||
tmp_list = tmp_list->prev;
|
||
}
|
||
|
||
source_widgets = g_slist_prepend (source_widgets, ipc_widget);
|
||
|
||
if (x != -1 && y != -1)
|
||
{
|
||
GtkWidget *toplevel = gtk_widget_get_toplevel (widget);
|
||
gtk_widget_translate_coordinates (widget, toplevel,
|
||
x, y, &x, &y);
|
||
gdk_window_get_root_coords (gtk_widget_get_window (toplevel),
|
||
x, y, &start_x, &start_y);
|
||
}
|
||
else if (event && event->type == GDK_MOTION_NOTIFY)
|
||
{
|
||
start_x = event->motion.x_root;
|
||
start_y = event->motion.y_root;
|
||
}
|
||
else
|
||
gdk_device_get_position (pointer, NULL, &start_x, &start_y);
|
||
|
||
context = gdk_drag_begin_from_point (ipc_window, pointer, targets, start_x, start_y);
|
||
|
||
gdk_drag_context_set_device (context, pointer);
|
||
g_list_free (targets);
|
||
|
||
if (managed &&
|
||
!gdk_drag_context_manage_dnd (context, ipc_window, actions))
|
||
{
|
||
gtk_drag_release_ipc_widget (ipc_widget);
|
||
g_object_unref (context);
|
||
return NULL;
|
||
}
|
||
|
||
info = gtk_drag_get_source_info (context, TRUE);
|
||
|
||
info->ipc_widget = ipc_widget;
|
||
g_object_set_data (G_OBJECT (info->ipc_widget), I_("gtk-info"), info);
|
||
|
||
info->widget = g_object_ref (widget);
|
||
|
||
info->button = button;
|
||
info->cursor = cursor;
|
||
info->target_list = target_list;
|
||
gtk_target_list_ref (target_list);
|
||
|
||
info->possible_actions = actions;
|
||
|
||
info->status = GTK_DRAG_STATUS_DRAG;
|
||
info->last_event = NULL;
|
||
info->selections = NULL;
|
||
info->icon_window = NULL;
|
||
info->icon_widget = NULL;
|
||
info->destroy_icon = FALSE;
|
||
|
||
if (event)
|
||
info->cur_screen = gdk_event_get_screen (event);
|
||
else
|
||
gdk_device_get_position (pointer, &info->cur_screen, NULL, NULL);
|
||
|
||
info->start_x = start_x;
|
||
info->start_y = start_y;
|
||
|
||
gtk_widget_reset_controllers (widget);
|
||
|
||
g_signal_emit_by_name (widget, "drag-begin", info->context);
|
||
|
||
/* Ensure that we have an icon before we start the drag; the
|
||
* application may have set one in ::drag_begin, or it may
|
||
* not have set one.
|
||
*/
|
||
if (!info->icon_widget && out_needs_icon == NULL)
|
||
{
|
||
GtkImageDefinition *icon = gtk_image_definition_new_icon_name ("text-x-generic");
|
||
set_icon_helper (info->context, icon, 0, 0);
|
||
gtk_image_definition_unref (icon);
|
||
}
|
||
|
||
if (out_needs_icon != NULL)
|
||
*out_needs_icon = (info->icon_widget == NULL);
|
||
|
||
if (managed)
|
||
{
|
||
g_signal_connect (context, "drop-performed",
|
||
G_CALLBACK (gtk_drag_context_drop_performed_cb), info);
|
||
g_signal_connect (context, "dnd-finished",
|
||
G_CALLBACK (gtk_drag_context_dnd_finished_cb), info);
|
||
g_signal_connect (context, "cancel",
|
||
G_CALLBACK (gtk_drag_context_cancel_cb), info);
|
||
g_signal_connect (context, "action-changed",
|
||
G_CALLBACK (gtk_drag_context_action_cb), info);
|
||
|
||
selection = gdk_drag_get_selection (context);
|
||
if (selection)
|
||
gtk_drag_source_check_selection (info, selection, time);
|
||
}
|
||
else
|
||
{
|
||
info->cur_x = info->start_x;
|
||
info->cur_y = info->start_y;
|
||
|
||
if (event && event->type == GDK_MOTION_NOTIFY)
|
||
gtk_drag_motion_cb (info->ipc_widget, (GdkEventMotion *)event, info);
|
||
else
|
||
gtk_drag_update (info, info->cur_screen, info->cur_x, info->cur_y, event);
|
||
|
||
g_signal_connect (info->ipc_widget, "grab-broken-event",
|
||
G_CALLBACK (gtk_drag_grab_broken_event_cb), info);
|
||
g_signal_connect (info->ipc_widget, "grab-notify",
|
||
G_CALLBACK (gtk_drag_grab_notify_cb), info);
|
||
g_signal_connect (info->ipc_widget, "button-release-event",
|
||
G_CALLBACK (gtk_drag_button_release_cb), info);
|
||
g_signal_connect (info->ipc_widget, "motion-notify-event",
|
||
G_CALLBACK (gtk_drag_motion_cb), info);
|
||
g_signal_connect (info->ipc_widget, "key-press-event",
|
||
G_CALLBACK (gtk_drag_key_cb), info);
|
||
g_signal_connect (info->ipc_widget, "key-release-event",
|
||
G_CALLBACK (gtk_drag_key_cb), info);
|
||
}
|
||
|
||
g_signal_connect (info->ipc_widget, "selection-get",
|
||
G_CALLBACK (gtk_drag_selection_get), info);
|
||
|
||
info->have_grab = TRUE;
|
||
info->grab_time = time;
|
||
|
||
return info->context;
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_begin_with_coordinates: (method)
|
||
* @widget: the source widget
|
||
* @targets: The targets (data formats) in which the
|
||
* source can provide the data
|
||
* @actions: A bitmask of the allowed drag actions for this drag
|
||
* @button: The button the user clicked to start the drag
|
||
* @event: (nullable): The event that triggered the start of the drag,
|
||
* or %NULL if none can be obtained.
|
||
* @x: The initial x coordinate to start dragging from, in the coordinate space
|
||
* of @widget. If -1 is passed, the coordinates are retrieved from @event or
|
||
* the current pointer position
|
||
* @y: The initial y coordinate to start dragging from, in the coordinate space
|
||
* of @widget. If -1 is passed, the coordinates are retrieved from @event or
|
||
* the current pointer position
|
||
*
|
||
* Initiates a drag on the source side. The function only needs to be used
|
||
* when the application is starting drags itself, and is not needed when
|
||
* gtk_drag_source_set() is used.
|
||
*
|
||
* The @event is used to retrieve the timestamp that will be used internally to
|
||
* grab the pointer. If @event is %NULL, then %GDK_CURRENT_TIME will be used.
|
||
* However, you should try to pass a real event in all cases, since that can be
|
||
* used to get information about the drag.
|
||
*
|
||
* Generally there are three cases when you want to start a drag by hand by
|
||
* calling this function:
|
||
*
|
||
* 1. During a #GtkWidget::button-press-event handler, if you want to start a drag
|
||
* immediately when the user presses the mouse button. Pass the @event
|
||
* that you have in your #GtkWidget::button-press-event handler.
|
||
*
|
||
* 2. During a #GtkWidget::motion-notify-event handler, if you want to start a drag
|
||
* when the mouse moves past a certain threshold distance after a button-press.
|
||
* Pass the @event that you have in your #GtkWidget::motion-notify-event handler.
|
||
*
|
||
* 3. During a timeout handler, if you want to start a drag after the mouse
|
||
* button is held down for some time. Try to save the last event that you got
|
||
* from the mouse, using gdk_event_copy(), and pass it to this function
|
||
* (remember to free the event with gdk_event_free() when you are done).
|
||
* If you really cannot pass a real event, pass %NULL instead.
|
||
*
|
||
* Returns: (transfer none): the context for this drag
|
||
*
|
||
* Since: 3.10
|
||
*/
|
||
GdkDragContext *
|
||
gtk_drag_begin_with_coordinates (GtkWidget *widget,
|
||
GtkTargetList *targets,
|
||
GdkDragAction actions,
|
||
gint button,
|
||
GdkEvent *event,
|
||
gint x,
|
||
gint y)
|
||
{
|
||
g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
|
||
g_return_val_if_fail (gtk_widget_get_realized (widget), NULL);
|
||
g_return_val_if_fail (targets != NULL, NULL);
|
||
|
||
return gtk_drag_begin_internal (widget, NULL, targets,
|
||
actions, button, event, x, y);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_begin: (method)
|
||
* @widget: the source widget
|
||
* @targets: The targets (data formats) in which the
|
||
* source can provide the data
|
||
* @actions: A bitmask of the allowed drag actions for this drag
|
||
* @button: The button the user clicked to start the drag
|
||
* @event: (nullable): The event that triggered the start of the drag,
|
||
* or %NULL if none can be obtained.
|
||
*
|
||
* This function is equivalent to gtk_drag_begin_with_coordinates(),
|
||
* passing -1, -1 as coordinates.
|
||
*
|
||
* Returns: (transfer none): the context for this drag
|
||
*
|
||
* Deprecated: 3.10: Use gtk_drag_begin_with_coordinates() instead
|
||
*/
|
||
GdkDragContext *
|
||
gtk_drag_begin (GtkWidget *widget,
|
||
GtkTargetList *targets,
|
||
GdkDragAction actions,
|
||
gint button,
|
||
GdkEvent *event)
|
||
{
|
||
g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
|
||
g_return_val_if_fail (gtk_widget_get_realized (widget), NULL);
|
||
g_return_val_if_fail (targets != NULL, NULL);
|
||
|
||
return gtk_drag_begin_internal (widget, NULL, targets,
|
||
actions, button, event, -1, -1);
|
||
}
|
||
|
||
static void
|
||
icon_widget_destroyed (GtkWidget *widget,
|
||
GtkDragSourceInfo *info)
|
||
{
|
||
g_clear_object (&info->icon_widget);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_update_icon_window (GtkDragSourceInfo *info)
|
||
{
|
||
if (!gtk_drag_is_managed (info->widget) && info->icon_window)
|
||
{
|
||
gtk_window_move (GTK_WINDOW (info->icon_window),
|
||
info->cur_x - info->hot_x,
|
||
info->cur_y - info->hot_y);
|
||
|
||
if (gtk_widget_get_visible (info->icon_window))
|
||
gdk_window_raise (gtk_widget_get_window (info->icon_window));
|
||
else
|
||
gtk_widget_show (info->icon_window);
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_drag_set_icon_widget_internal (GdkDragContext *context,
|
||
GtkWidget *widget,
|
||
gint hot_x,
|
||
gint hot_y,
|
||
gboolean destroy_on_release)
|
||
{
|
||
GtkDragSourceInfo *info;
|
||
|
||
info = gtk_drag_get_source_info (context, FALSE);
|
||
if (info == NULL)
|
||
{
|
||
if (destroy_on_release)
|
||
gtk_widget_destroy (widget);
|
||
return;
|
||
}
|
||
|
||
gtk_drag_remove_icon (info);
|
||
|
||
if (widget)
|
||
g_object_ref (widget);
|
||
|
||
info->icon_widget = widget;
|
||
info->hot_x = hot_x;
|
||
info->hot_y = hot_y;
|
||
info->destroy_icon = destroy_on_release;
|
||
|
||
if (!widget)
|
||
goto out;
|
||
|
||
g_signal_connect (widget, "destroy", G_CALLBACK (icon_widget_destroyed), info);
|
||
|
||
gdk_drag_context_set_hotspot (context, hot_x, hot_y);
|
||
|
||
if (!info->icon_window)
|
||
{
|
||
GdkScreen *screen;
|
||
GdkVisual *visual;
|
||
gboolean has_rgba;
|
||
|
||
screen = gdk_window_get_screen (gdk_drag_context_get_source_window (context));
|
||
visual = gdk_screen_get_rgba_visual (screen);
|
||
has_rgba = visual != NULL && gdk_screen_is_composited (screen);
|
||
|
||
info->icon_window = gtk_window_new (GTK_WINDOW_POPUP);
|
||
gtk_window_set_type_hint (GTK_WINDOW (info->icon_window), GDK_WINDOW_TYPE_HINT_DND);
|
||
gtk_window_set_screen (GTK_WINDOW (info->icon_window), screen);
|
||
gtk_widget_set_size_request (info->icon_window, 24, 24);
|
||
if (visual)
|
||
gtk_widget_set_visual (info->icon_window, visual);
|
||
gtk_widget_set_events (info->icon_window, GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK);
|
||
|
||
if (has_rgba)
|
||
gtk_widget_set_app_paintable (info->icon_window, TRUE);
|
||
|
||
gtk_window_set_hardcoded_window (GTK_WINDOW (info->icon_window),
|
||
gdk_drag_context_get_drag_window (context));
|
||
gtk_widget_show (info->icon_window);
|
||
}
|
||
|
||
if (GTK_IS_WINDOW (widget))
|
||
{
|
||
gtk_widget_hide (widget);
|
||
gtk_widget_unrealize (widget);
|
||
gtk_widget_set_parent_window (widget, gtk_widget_get_window (info->icon_window));
|
||
gtk_widget_show (widget);
|
||
}
|
||
|
||
if (gtk_bin_get_child (GTK_BIN (info->icon_window)))
|
||
gtk_container_remove (GTK_CONTAINER (info->icon_window), gtk_bin_get_child (GTK_BIN (info->icon_window)));
|
||
gtk_container_add (GTK_CONTAINER (info->icon_window), widget);
|
||
|
||
out:
|
||
gtk_drag_update_cursor (info);
|
||
gtk_drag_update_icon_window (info);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_set_icon_widget:
|
||
* @context: the context for a drag. (This must be called
|
||
with a context for the source side of a drag)
|
||
* @widget: a widget to use as an icon
|
||
* @hot_x: the X offset within @widget of the hotspot
|
||
* @hot_y: the Y offset within @widget of the hotspot
|
||
*
|
||
* Changes the icon for drag operation to a given widget.
|
||
* GTK+ will not destroy the widget, so if you don’t want
|
||
* it to persist, you should connect to the “drag-end”
|
||
* signal and destroy it yourself.
|
||
*/
|
||
void
|
||
gtk_drag_set_icon_widget (GdkDragContext *context,
|
||
GtkWidget *widget,
|
||
gint hot_x,
|
||
gint hot_y)
|
||
{
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
g_return_if_fail (GTK_IS_WIDGET (widget));
|
||
|
||
gtk_drag_set_icon_widget_internal (context, widget, hot_x, hot_y, FALSE);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_draw_icon_pattern (GtkWidget *window,
|
||
cairo_t *cr,
|
||
gpointer pattern)
|
||
{
|
||
cairo_set_source (cr, pattern);
|
||
cairo_paint (cr);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_draw_icon_pattern_and_background (GtkWidget *window,
|
||
cairo_t *cr,
|
||
gpointer pattern)
|
||
{
|
||
GtkStyleContext *context;
|
||
int width, height;
|
||
|
||
context = gtk_widget_get_style_context (window);
|
||
width = gtk_widget_get_allocated_width (window);
|
||
height = gtk_widget_get_allocated_height (window);
|
||
|
||
gtk_render_background (context, cr, 0, 0, width, height);
|
||
gtk_render_frame (context, cr, 0, 0, width, height);
|
||
|
||
cairo_set_source (cr, pattern);
|
||
cairo_paint (cr);
|
||
}
|
||
|
||
static void
|
||
set_icon_helper (GdkDragContext *context,
|
||
GtkImageDefinition *def,
|
||
gint hot_x,
|
||
gint hot_y)
|
||
{
|
||
GtkWidget *widget;
|
||
|
||
widget = gtk_image_new ();
|
||
gtk_widget_show (widget);
|
||
|
||
gtk_image_set_from_definition (GTK_IMAGE (widget), def, GTK_ICON_SIZE_DND);
|
||
|
||
gtk_drag_set_icon_widget_internal (context, widget, hot_x, hot_y, TRUE);
|
||
}
|
||
|
||
void
|
||
gtk_drag_set_icon_definition (GdkDragContext *context,
|
||
GtkImageDefinition *def,
|
||
gint hot_x,
|
||
gint hot_y)
|
||
{
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
g_return_if_fail (def != NULL);
|
||
|
||
set_icon_helper (context, def, hot_x, hot_y);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_set_icon_pixbuf:
|
||
* @context: the context for a drag (This must be called
|
||
* with a context for the source side of a drag)
|
||
* @pixbuf: the #GdkPixbuf to use as the drag icon
|
||
* @hot_x: the X offset within @widget of the hotspot
|
||
* @hot_y: the Y offset within @widget of the hotspot
|
||
*
|
||
* Sets @pixbuf as the icon for a given drag.
|
||
*/
|
||
void
|
||
gtk_drag_set_icon_pixbuf (GdkDragContext *context,
|
||
GdkPixbuf *pixbuf,
|
||
gint hot_x,
|
||
gint hot_y)
|
||
{
|
||
GtkImageDefinition *def;
|
||
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
g_return_if_fail (GDK_IS_PIXBUF (pixbuf));
|
||
|
||
def = gtk_image_definition_new_pixbuf (pixbuf, 1);
|
||
set_icon_helper (context, def, hot_x, hot_y);
|
||
|
||
gtk_image_definition_unref (def);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_set_icon_stock:
|
||
* @context: the context for a drag (This must be called
|
||
* with a context for the source side of a drag)
|
||
* @stock_id: the ID of the stock icon to use for the drag
|
||
* @hot_x: the X offset within the icon of the hotspot
|
||
* @hot_y: the Y offset within the icon of the hotspot
|
||
*
|
||
* Sets the icon for a given drag from a stock ID.
|
||
*
|
||
* Deprecated: 3.10: Use gtk_drag_set_icon_name() instead.
|
||
*/
|
||
void
|
||
gtk_drag_set_icon_stock (GdkDragContext *context,
|
||
const gchar *stock_id,
|
||
gint hot_x,
|
||
gint hot_y)
|
||
{
|
||
GtkImageDefinition *def;
|
||
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
g_return_if_fail (stock_id != NULL);
|
||
|
||
def = gtk_image_definition_new_stock (stock_id);
|
||
set_icon_helper (context, def, hot_x, hot_y);
|
||
|
||
gtk_image_definition_unref (def);
|
||
}
|
||
|
||
/* XXX: This function is in gdk, too. Should it be in Cairo? */
|
||
static gboolean
|
||
_gtk_cairo_surface_extents (cairo_surface_t *surface,
|
||
GdkRectangle *extents)
|
||
{
|
||
double x1, x2, y1, y2;
|
||
cairo_t *cr;
|
||
|
||
g_return_val_if_fail (surface != NULL, FALSE);
|
||
g_return_val_if_fail (extents != NULL, FALSE);
|
||
|
||
cr = cairo_create (surface);
|
||
cairo_clip_extents (cr, &x1, &y1, &x2, &y2);
|
||
cairo_destroy (cr);
|
||
|
||
x1 = floor (x1);
|
||
y1 = floor (y1);
|
||
x2 = ceil (x2);
|
||
y2 = ceil (y2);
|
||
x2 -= x1;
|
||
y2 -= y1;
|
||
|
||
if (x1 < G_MININT || x1 > G_MAXINT ||
|
||
y1 < G_MININT || y1 > G_MAXINT ||
|
||
x2 > G_MAXINT || y2 > G_MAXINT)
|
||
{
|
||
extents->x = extents->y = extents->width = extents->height = 0;
|
||
return FALSE;
|
||
}
|
||
|
||
extents->x = x1;
|
||
extents->y = y1;
|
||
extents->width = x2;
|
||
extents->height = y2;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_set_icon_surface:
|
||
* @context: the context for a drag (This must be called
|
||
* with a context for the source side of a drag)
|
||
* @surface: the surface to use as icon
|
||
*
|
||
* Sets @surface as the icon for a given drag. GTK+ retains
|
||
* references for the arguments, and will release them when
|
||
* they are no longer needed.
|
||
*
|
||
* To position the surface relative to the mouse, use
|
||
* cairo_surface_set_device_offset() on @surface. The mouse
|
||
* cursor will be positioned at the (0,0) coordinate of the
|
||
* surface.
|
||
*/
|
||
void
|
||
gtk_drag_set_icon_surface (GdkDragContext *context,
|
||
cairo_surface_t *surface)
|
||
{
|
||
GtkWidget *window;
|
||
GdkScreen *screen;
|
||
GdkRectangle extents;
|
||
cairo_pattern_t *pattern;
|
||
GdkVisual *rgba_visual;
|
||
gboolean has_rgba;
|
||
cairo_matrix_t matrix;
|
||
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
g_return_if_fail (surface != NULL);
|
||
|
||
_gtk_cairo_surface_extents (surface, &extents);
|
||
|
||
screen = gdk_window_get_screen (gdk_drag_context_get_source_window (context));
|
||
rgba_visual = gdk_screen_get_rgba_visual (screen);
|
||
|
||
window = gtk_window_new (GTK_WINDOW_POPUP);
|
||
has_rgba = rgba_visual != NULL && gdk_screen_is_composited (screen);
|
||
|
||
gtk_window_set_screen (GTK_WINDOW (window), screen);
|
||
|
||
if (has_rgba)
|
||
gtk_widget_set_visual (GTK_WIDGET (window), rgba_visual);
|
||
|
||
gtk_window_set_type_hint (GTK_WINDOW (window), GDK_WINDOW_TYPE_HINT_DND);
|
||
|
||
gtk_widget_set_events (window, GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK);
|
||
gtk_widget_set_app_paintable (window, TRUE);
|
||
|
||
gtk_widget_set_size_request (window, extents.width, extents.height);
|
||
gtk_widget_realize (window);
|
||
|
||
pattern = cairo_pattern_create_for_surface (surface);
|
||
cairo_matrix_init_translate (&matrix, extents.x, extents.y);
|
||
cairo_pattern_set_matrix (pattern, &matrix);
|
||
|
||
g_signal_connect_data (window,
|
||
"draw",
|
||
has_rgba ? G_CALLBACK (gtk_drag_draw_icon_pattern)
|
||
: G_CALLBACK (gtk_drag_draw_icon_pattern_and_background),
|
||
pattern,
|
||
(GClosureNotify) cairo_pattern_destroy,
|
||
G_CONNECT_AFTER);
|
||
|
||
gtk_drag_set_icon_widget_internal (context, window, extents.x, extents.y, TRUE);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_set_icon_name:
|
||
* @context: the context for a drag (This must be called
|
||
* with a context for the source side of a drag)
|
||
* @icon_name: name of icon to use
|
||
* @hot_x: the X offset of the hotspot within the icon
|
||
* @hot_y: the Y offset of the hotspot within the icon
|
||
*
|
||
* Sets the icon for a given drag from a named themed icon. See
|
||
* the docs for #GtkIconTheme for more details. Note that the
|
||
* size of the icon depends on the icon theme (the icon is
|
||
* loaded at the symbolic size #GTK_ICON_SIZE_DND), thus
|
||
* @hot_x and @hot_y have to be used with care.
|
||
*
|
||
* Since: 2.8
|
||
*/
|
||
void
|
||
gtk_drag_set_icon_name (GdkDragContext *context,
|
||
const gchar *icon_name,
|
||
gint hot_x,
|
||
gint hot_y)
|
||
{
|
||
GtkImageDefinition *def;
|
||
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
g_return_if_fail (icon_name != NULL && icon_name[0] != '\0');
|
||
|
||
def = gtk_image_definition_new_icon_name (icon_name);
|
||
set_icon_helper (context, def, hot_x, hot_y);
|
||
|
||
gtk_image_definition_unref (def);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_set_icon_gicon:
|
||
* @context: the context for a drag (This must be called
|
||
* with a context for the source side of a drag)
|
||
* @icon: a #GIcon
|
||
* @hot_x: the X offset of the hotspot within the icon
|
||
* @hot_y: the Y offset of the hotspot within the icon
|
||
*
|
||
* Sets the icon for a given drag from the given @icon.
|
||
* See the documentation for gtk_drag_set_icon_name()
|
||
* for more details about using icons in drag and drop.
|
||
*
|
||
* Since: 3.2
|
||
*/
|
||
void
|
||
gtk_drag_set_icon_gicon (GdkDragContext *context,
|
||
GIcon *icon,
|
||
gint hot_x,
|
||
gint hot_y)
|
||
{
|
||
GtkImageDefinition *def;
|
||
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
g_return_if_fail (icon != NULL);
|
||
|
||
def = gtk_image_definition_new_gicon (icon);
|
||
set_icon_helper (context, def, hot_x, hot_y);
|
||
|
||
gtk_image_definition_unref (def);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_set_icon_default:
|
||
* @context: the context for a drag (This must be called
|
||
* with a context for the source side of a drag)
|
||
*
|
||
* Sets the icon for a particular drag to the default
|
||
* icon.
|
||
*/
|
||
void
|
||
gtk_drag_set_icon_default (GdkDragContext *context)
|
||
{
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
|
||
gtk_drag_set_icon_name (context, "text-x-generic", -2, -2);
|
||
}
|
||
|
||
/*
|
||
* _gtk_drag_source_handle_event:
|
||
* @toplevel: Toplevel widget that received the event
|
||
* @event: the event to handle
|
||
*
|
||
* Called from widget event handling code on Drag events
|
||
* for drag sources.
|
||
*/
|
||
void
|
||
_gtk_drag_source_handle_event (GtkWidget *widget,
|
||
GdkEvent *event)
|
||
{
|
||
GtkDragSourceInfo *info;
|
||
GdkDragContext *context;
|
||
|
||
g_return_if_fail (widget != NULL);
|
||
g_return_if_fail (event != NULL);
|
||
|
||
context = event->dnd.context;
|
||
info = gtk_drag_get_source_info (context, FALSE);
|
||
if (!info)
|
||
return;
|
||
|
||
switch (event->type)
|
||
{
|
||
case GDK_DRAG_STATUS:
|
||
{
|
||
GdkCursor *cursor;
|
||
if (info->proxy_dest)
|
||
{
|
||
if (!event->dnd.send_event)
|
||
{
|
||
if (info->proxy_dest->proxy_drop_wait)
|
||
{
|
||
gboolean result = gdk_drag_context_get_selected_action (context) != 0;
|
||
|
||
/* Aha - we can finally pass the DROP on... */
|
||
gdk_drop_reply (info->proxy_dest->context, result, info->proxy_dest->proxy_drop_time);
|
||
if (result)
|
||
gdk_drag_drop (info->context, info->proxy_dest->proxy_drop_time);
|
||
else
|
||
gtk_drag_finish (info->proxy_dest->context, FALSE, FALSE, info->proxy_dest->proxy_drop_time);
|
||
}
|
||
else
|
||
{
|
||
gdk_drag_status (info->proxy_dest->context,
|
||
gdk_drag_context_get_selected_action (event->dnd.context),
|
||
event->dnd.time);
|
||
}
|
||
}
|
||
}
|
||
else if (info->have_grab)
|
||
{
|
||
cursor = gtk_drag_get_cursor (widget,
|
||
gtk_widget_get_display (widget),
|
||
gdk_drag_context_get_selected_action (event->dnd.context),
|
||
info);
|
||
if (info->cursor != cursor)
|
||
{
|
||
GdkDevice *pointer;
|
||
|
||
pointer = gdk_drag_context_get_device (context);
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
gdk_device_grab (pointer, gtk_widget_get_window (widget),
|
||
GDK_OWNERSHIP_APPLICATION, FALSE,
|
||
GDK_POINTER_MOTION_MASK | GDK_BUTTON_RELEASE_MASK,
|
||
cursor, info->grab_time);
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
info->cursor = cursor;
|
||
}
|
||
|
||
gtk_drag_add_update_idle (info);
|
||
}
|
||
}
|
||
break;
|
||
|
||
case GDK_DROP_FINISHED:
|
||
gtk_drag_drop_finished (info, GTK_DRAG_RESULT_SUCCESS, event->dnd.time);
|
||
break;
|
||
default:
|
||
g_assert_not_reached ();
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_drag_source_check_selection (GtkDragSourceInfo *info,
|
||
GdkAtom selection,
|
||
guint32 time)
|
||
{
|
||
GList *tmp_list;
|
||
|
||
tmp_list = info->selections;
|
||
while (tmp_list)
|
||
{
|
||
if (GDK_POINTER_TO_ATOM (tmp_list->data) == selection)
|
||
return;
|
||
tmp_list = tmp_list->next;
|
||
}
|
||
|
||
gtk_selection_owner_set_for_display (gtk_widget_get_display (info->widget),
|
||
info->ipc_widget,
|
||
selection,
|
||
time);
|
||
info->selections = g_list_prepend (info->selections,
|
||
GUINT_TO_POINTER (selection));
|
||
|
||
tmp_list = info->target_list->list;
|
||
while (tmp_list)
|
||
{
|
||
GtkTargetPair *pair = tmp_list->data;
|
||
|
||
gtk_selection_add_target (info->ipc_widget,
|
||
selection,
|
||
pair->target,
|
||
pair->info);
|
||
tmp_list = tmp_list->next;
|
||
}
|
||
|
||
gtk_selection_add_target (info->ipc_widget,
|
||
selection,
|
||
gdk_atom_intern_static_string ("DELETE"),
|
||
TARGET_DELETE);
|
||
}
|
||
|
||
|
||
/* Clean up from the drag, and display snapback, if necessary. */
|
||
static void
|
||
gtk_drag_drop_finished (GtkDragSourceInfo *info,
|
||
GtkDragResult result,
|
||
guint time)
|
||
{
|
||
gboolean success;
|
||
|
||
success = (result == GTK_DRAG_RESULT_SUCCESS);
|
||
gtk_drag_source_release_selections (info, time);
|
||
|
||
if (info->proxy_dest)
|
||
{
|
||
/* The time from the event isn't reliable for Xdnd drags */
|
||
gtk_drag_finish (info->proxy_dest->context, success, FALSE,
|
||
info->proxy_dest->proxy_drop_time);
|
||
gtk_drag_source_info_destroy (info);
|
||
}
|
||
else
|
||
{
|
||
if (!success)
|
||
g_signal_emit_by_name (info->widget, "drag-failed",
|
||
info->context, result, &success);
|
||
|
||
gdk_drag_drop_done (info->context, success);
|
||
gtk_drag_source_info_destroy (info);
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_drag_source_release_selections (GtkDragSourceInfo *info,
|
||
guint32 time)
|
||
{
|
||
GdkDisplay *display = gtk_widget_get_display (info->widget);
|
||
GList *tmp_list = info->selections;
|
||
|
||
while (tmp_list)
|
||
{
|
||
GdkAtom selection = GDK_POINTER_TO_ATOM (tmp_list->data);
|
||
if (gdk_selection_owner_get_for_display (display, selection) == gtk_widget_get_window (info->ipc_widget))
|
||
gtk_selection_owner_set_for_display (display, NULL, selection, time);
|
||
|
||
tmp_list = tmp_list->next;
|
||
}
|
||
|
||
g_list_free (info->selections);
|
||
info->selections = NULL;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_drop (GtkDragSourceInfo *info,
|
||
guint32 time)
|
||
{
|
||
if (gdk_drag_context_get_protocol (info->context) == GDK_DRAG_PROTO_ROOTWIN)
|
||
{
|
||
GtkSelectionData selection_data;
|
||
GList *tmp_list;
|
||
/* GTK+ traditionally has used application/x-rootwin-drop, but the
|
||
* XDND spec specifies x-rootwindow-drop.
|
||
*/
|
||
GdkAtom target1 = gdk_atom_intern_static_string ("application/x-rootwindow-drop");
|
||
GdkAtom target2 = gdk_atom_intern_static_string ("application/x-rootwin-drop");
|
||
|
||
tmp_list = info->target_list->list;
|
||
while (tmp_list)
|
||
{
|
||
GtkTargetPair *pair = tmp_list->data;
|
||
|
||
if (pair->target == target1 || pair->target == target2)
|
||
{
|
||
selection_data.selection = GDK_NONE;
|
||
selection_data.target = pair->target;
|
||
selection_data.data = NULL;
|
||
selection_data.length = -1;
|
||
|
||
g_signal_emit_by_name (info->widget, "drag-data-get",
|
||
info->context, &selection_data,
|
||
pair->info,
|
||
time);
|
||
|
||
/* FIXME: Should we check for length >= 0 here? */
|
||
gtk_drag_drop_finished (info, GTK_DRAG_RESULT_SUCCESS, time);
|
||
return;
|
||
}
|
||
tmp_list = tmp_list->next;
|
||
}
|
||
gtk_drag_drop_finished (info, GTK_DRAG_RESULT_NO_TARGET, time);
|
||
}
|
||
else
|
||
{
|
||
if (info->icon_window)
|
||
gtk_widget_hide (info->icon_window);
|
||
|
||
gdk_drag_drop (info->context, time);
|
||
info->drop_timeout = gdk_threads_add_timeout (DROP_ABORT_TIME,
|
||
gtk_drag_abort_timeout,
|
||
info);
|
||
g_source_set_name_by_id (info->drop_timeout, "[gtk+] gtk_drag_abort_timeout");
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Source side callbacks.
|
||
*/
|
||
static void
|
||
gtk_drag_selection_get (GtkWidget *widget,
|
||
GtkSelectionData *selection_data,
|
||
guint sel_info,
|
||
guint32 time,
|
||
gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = data;
|
||
static GdkAtom null_atom = GDK_NONE;
|
||
guint target_info;
|
||
|
||
if (!null_atom)
|
||
null_atom = gdk_atom_intern_static_string ("NULL");
|
||
|
||
switch (sel_info)
|
||
{
|
||
case TARGET_DELETE:
|
||
g_signal_emit_by_name (info->widget,
|
||
"drag-data-delete",
|
||
info->context);
|
||
gtk_selection_data_set (selection_data, null_atom, 8, NULL, 0);
|
||
break;
|
||
default:
|
||
if (info->proxy_dest)
|
||
{
|
||
/* This is sort of dangerous and needs to be thought
|
||
* through better
|
||
*/
|
||
info->proxy_dest->proxy_data = selection_data;
|
||
gtk_drag_get_data (info->widget,
|
||
info->proxy_dest->context,
|
||
gtk_selection_data_get_target (selection_data),
|
||
time);
|
||
gtk_main ();
|
||
info->proxy_dest->proxy_data = NULL;
|
||
}
|
||
else
|
||
{
|
||
if (gtk_target_list_find (info->target_list,
|
||
gtk_selection_data_get_target (selection_data),
|
||
&target_info))
|
||
{
|
||
g_signal_emit_by_name (info->widget, "drag-data-get",
|
||
info->context,
|
||
selection_data,
|
||
target_info,
|
||
time);
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_drag_remove_icon (GtkDragSourceInfo *info)
|
||
{
|
||
if (info->icon_widget)
|
||
{
|
||
GtkWidget *widget;
|
||
|
||
widget = info->icon_widget;
|
||
info->icon_widget = NULL;
|
||
|
||
g_signal_handlers_disconnect_by_func (widget, icon_widget_destroyed, info);
|
||
|
||
gtk_widget_hide (widget);
|
||
gtk_widget_set_opacity (widget, 1.0);
|
||
|
||
if (info->destroy_icon)
|
||
gtk_widget_destroy (widget);
|
||
else
|
||
gtk_container_remove (GTK_CONTAINER (info->icon_window), widget);
|
||
|
||
g_object_unref (widget);
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_drag_source_info_free (GtkDragSourceInfo *info)
|
||
{
|
||
gtk_drag_remove_icon (info);
|
||
gtk_widget_destroy (info->icon_window);
|
||
g_free (info);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_source_info_destroy (GtkDragSourceInfo *info)
|
||
{
|
||
GdkDragContext *context;
|
||
GdkEvent *last_event;
|
||
|
||
g_signal_handlers_disconnect_by_func (info->context,
|
||
gtk_drag_context_drop_performed_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->context,
|
||
gtk_drag_context_dnd_finished_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->context,
|
||
gtk_drag_context_cancel_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->context,
|
||
gtk_drag_context_action_cb,
|
||
info);
|
||
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_grab_broken_event_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_grab_notify_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_button_release_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_motion_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_key_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_selection_get,
|
||
info);
|
||
|
||
if (!info->proxy_dest)
|
||
g_signal_emit_by_name (info->widget, "drag-end", info->context);
|
||
|
||
g_clear_object (&info->widget);
|
||
|
||
gtk_selection_remove_all (info->ipc_widget);
|
||
g_object_set_data (G_OBJECT (info->ipc_widget), I_("gtk-info"), NULL);
|
||
source_widgets = g_slist_remove (source_widgets, info->ipc_widget);
|
||
gtk_drag_release_ipc_widget (info->ipc_widget);
|
||
|
||
gtk_target_list_unref (info->target_list);
|
||
|
||
if (info->drop_timeout)
|
||
g_source_remove (info->drop_timeout);
|
||
|
||
if (info->update_idle)
|
||
g_source_remove (info->update_idle);
|
||
|
||
/* keep the icon_window alive until the (possible) drag cancel animation is done */
|
||
g_object_set_data_full (G_OBJECT (info->context), "former-gtk-source-info", info, (GDestroyNotify)gtk_drag_source_info_free);
|
||
context = info->context;
|
||
last_event = info->last_event;
|
||
|
||
gtk_drag_clear_source_info (context);
|
||
|
||
if (last_event)
|
||
gdk_event_free (last_event);
|
||
|
||
g_object_unref (context);
|
||
}
|
||
|
||
static gboolean
|
||
gtk_drag_update_idle (gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = data;
|
||
GdkWindow *dest_window;
|
||
GdkDragProtocol protocol;
|
||
GdkAtom selection;
|
||
|
||
GdkDragAction action;
|
||
GdkDragAction possible_actions;
|
||
guint32 time;
|
||
|
||
info->update_idle = 0;
|
||
|
||
if (info->last_event)
|
||
{
|
||
time = gtk_drag_get_event_time (info->last_event);
|
||
gtk_drag_get_event_actions (info->last_event,
|
||
info->button,
|
||
info->possible_actions,
|
||
&action, &possible_actions);
|
||
|
||
gtk_drag_update_icon_window (info);
|
||
gdk_drag_find_window_for_screen (info->context,
|
||
info->icon_window ? gtk_widget_get_window (info->icon_window) : NULL,
|
||
info->cur_screen, info->cur_x, info->cur_y,
|
||
&dest_window, &protocol);
|
||
|
||
if (!gdk_drag_motion (info->context, dest_window, protocol,
|
||
info->cur_x, info->cur_y, action,
|
||
possible_actions,
|
||
time))
|
||
{
|
||
gdk_event_free ((GdkEvent *)info->last_event);
|
||
info->last_event = NULL;
|
||
}
|
||
|
||
if (dest_window)
|
||
g_object_unref (dest_window);
|
||
|
||
selection = gdk_drag_get_selection (info->context);
|
||
if (selection)
|
||
gtk_drag_source_check_selection (info, selection, time);
|
||
|
||
}
|
||
|
||
return FALSE;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_add_update_idle (GtkDragSourceInfo *info)
|
||
{
|
||
/* We use an idle lower than GDK_PRIORITY_REDRAW so that exposes
|
||
* from the last move can catch up before we move again.
|
||
*/
|
||
if (!info->update_idle)
|
||
{
|
||
info->update_idle = gdk_threads_add_idle_full (GDK_PRIORITY_REDRAW + 5,
|
||
gtk_drag_update_idle,
|
||
info,
|
||
NULL);
|
||
g_source_set_name_by_id (info->update_idle, "[gtk+] gtk_drag_update_idle");
|
||
}
|
||
}
|
||
|
||
/*
|
||
* gtk_drag_update:
|
||
* @info: DragSourceInfo for the drag
|
||
* @screen: new screen
|
||
* @x_root: new X position
|
||
* @y_root: new y position
|
||
* @event: event received requiring update
|
||
*
|
||
* Updates the status of the drag; called when the
|
||
* cursor moves or the modifier changes
|
||
*/
|
||
static void
|
||
gtk_drag_update (GtkDragSourceInfo *info,
|
||
GdkScreen *screen,
|
||
gint x_root,
|
||
gint y_root,
|
||
const GdkEvent *event)
|
||
{
|
||
info->cur_screen = screen;
|
||
info->cur_x = x_root;
|
||
info->cur_y = y_root;
|
||
if (info->last_event)
|
||
{
|
||
gdk_event_free ((GdkEvent *)info->last_event);
|
||
info->last_event = NULL;
|
||
}
|
||
if (event)
|
||
info->last_event = gdk_event_copy ((GdkEvent *)event);
|
||
|
||
gtk_drag_add_update_idle (info);
|
||
}
|
||
|
||
/* Called when the user finishes to drag, either by
|
||
* releasing the mouse, or by pressing Esc.
|
||
*/
|
||
static void
|
||
gtk_drag_end (GtkDragSourceInfo *info,
|
||
guint32 time)
|
||
{
|
||
GdkDevice *pointer, *keyboard;
|
||
|
||
pointer = gdk_drag_context_get_device (info->context);
|
||
keyboard = gdk_device_get_associated_device (pointer);
|
||
|
||
/* Prevent ungrab before grab (see bug 623865) */
|
||
if (info->grab_time == GDK_CURRENT_TIME)
|
||
time = GDK_CURRENT_TIME;
|
||
|
||
if (info->update_idle)
|
||
{
|
||
g_source_remove (info->update_idle);
|
||
info->update_idle = 0;
|
||
}
|
||
|
||
if (info->last_event)
|
||
{
|
||
gdk_event_free (info->last_event);
|
||
info->last_event = NULL;
|
||
}
|
||
|
||
info->have_grab = FALSE;
|
||
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_grab_broken_event_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_grab_notify_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_button_release_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_motion_cb,
|
||
info);
|
||
g_signal_handlers_disconnect_by_func (info->ipc_widget,
|
||
gtk_drag_key_cb,
|
||
info);
|
||
|
||
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
||
gdk_device_ungrab (pointer, time);
|
||
G_GNUC_END_IGNORE_DEPRECATIONS;
|
||
|
||
ungrab_dnd_keys (info->ipc_widget, keyboard, time);
|
||
gtk_device_grab_remove (info->ipc_widget, pointer);
|
||
}
|
||
|
||
/* Called on cancellation of a drag, either by the user
|
||
* or programmatically.
|
||
*/
|
||
static void
|
||
gtk_drag_cancel_internal (GtkDragSourceInfo *info,
|
||
GtkDragResult result,
|
||
guint32 time)
|
||
{
|
||
gtk_drag_end (info, time);
|
||
gdk_drag_abort (info->context, time);
|
||
gtk_drag_drop_finished (info, result, time);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_context_drop_performed_cb (GdkDragContext *context,
|
||
guint32 time_,
|
||
GtkDragSourceInfo *info)
|
||
{
|
||
gtk_drag_end (info, time_);
|
||
gtk_drag_drop (info, time_);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_context_cancel_cb (GdkDragContext *context,
|
||
GdkDragCancelReason reason,
|
||
GtkDragSourceInfo *info)
|
||
{
|
||
GtkDragResult result;
|
||
|
||
switch (reason)
|
||
{
|
||
case GDK_DRAG_CANCEL_NO_TARGET:
|
||
result = GTK_DRAG_RESULT_NO_TARGET;
|
||
break;
|
||
case GDK_DRAG_CANCEL_USER_CANCELLED:
|
||
result = GTK_DRAG_RESULT_USER_CANCELLED;
|
||
break;
|
||
case GDK_DRAG_CANCEL_ERROR:
|
||
default:
|
||
result = GTK_DRAG_RESULT_ERROR;
|
||
break;
|
||
}
|
||
gtk_drag_cancel_internal (info, result, GDK_CURRENT_TIME);
|
||
}
|
||
|
||
static void
|
||
gtk_drag_context_action_cb (GdkDragContext *context,
|
||
GdkDragAction action,
|
||
GtkDragSourceInfo *info)
|
||
{
|
||
if (info->proxy_dest)
|
||
{
|
||
if (info->proxy_dest->proxy_drop_wait)
|
||
{
|
||
gboolean result = gdk_drag_context_get_selected_action (context) != 0;
|
||
|
||
/* Aha - we can finally pass the DROP on... */
|
||
gdk_drop_reply (info->proxy_dest->context, result, info->proxy_dest->proxy_drop_time);
|
||
if (result)
|
||
gdk_drag_drop (info->context, info->proxy_dest->proxy_drop_time);
|
||
else
|
||
gtk_drag_finish (info->proxy_dest->context, FALSE, FALSE, info->proxy_dest->proxy_drop_time);
|
||
}
|
||
else
|
||
{
|
||
gdk_drag_status (info->proxy_dest->context,
|
||
gdk_drag_context_get_selected_action (context),
|
||
GDK_CURRENT_TIME);
|
||
}
|
||
|
||
g_signal_stop_emission_by_name (context, "action");
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_drag_context_dnd_finished_cb (GdkDragContext *context,
|
||
GtkDragSourceInfo *info)
|
||
{
|
||
gtk_drag_source_release_selections (info, GDK_CURRENT_TIME);
|
||
|
||
if (info->proxy_dest)
|
||
{
|
||
/* The time from the event isn't reliable for Xdnd drags */
|
||
gtk_drag_finish (info->proxy_dest->context, TRUE, FALSE,
|
||
info->proxy_dest->proxy_drop_time);
|
||
}
|
||
|
||
gtk_drag_source_info_destroy (info);
|
||
}
|
||
|
||
/* “motion-notify-event” callback during drag. */
|
||
static gboolean
|
||
gtk_drag_motion_cb (GtkWidget *widget,
|
||
GdkEventMotion *event,
|
||
gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = (GtkDragSourceInfo *)data;
|
||
GdkScreen *screen;
|
||
gint x_root, y_root;
|
||
|
||
if (event->is_hint)
|
||
{
|
||
gdk_device_get_position (event->device, &screen, &x_root, &y_root);
|
||
event->x_root = x_root;
|
||
event->y_root = y_root;
|
||
}
|
||
else
|
||
screen = gdk_event_get_screen ((GdkEvent *)event);
|
||
|
||
x_root = (gint)(event->x_root + 0.5);
|
||
y_root = (gint)(event->y_root + 0.5);
|
||
gtk_drag_update (info, screen, x_root, y_root, (GdkEvent *) event);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
#define BIG_STEP 20
|
||
#define SMALL_STEP 1
|
||
|
||
/* “key-press/release-event” callback during drag */
|
||
static gboolean
|
||
gtk_drag_key_cb (GtkWidget *widget,
|
||
GdkEventKey *event,
|
||
gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = (GtkDragSourceInfo *)data;
|
||
GdkModifierType state;
|
||
GdkWindow *root_window;
|
||
GdkDevice *pointer;
|
||
gint dx, dy;
|
||
|
||
dx = dy = 0;
|
||
state = event->state & gtk_accelerator_get_default_mod_mask ();
|
||
pointer = gdk_device_get_associated_device (gdk_event_get_device ((GdkEvent *) event));
|
||
|
||
if (event->type == GDK_KEY_PRESS)
|
||
{
|
||
switch (event->keyval)
|
||
{
|
||
case GDK_KEY_Escape:
|
||
gtk_drag_cancel_internal (info, GTK_DRAG_RESULT_USER_CANCELLED, event->time);
|
||
return TRUE;
|
||
|
||
case GDK_KEY_space:
|
||
case GDK_KEY_Return:
|
||
case GDK_KEY_ISO_Enter:
|
||
case GDK_KEY_KP_Enter:
|
||
case GDK_KEY_KP_Space:
|
||
if ((gdk_drag_context_get_selected_action (info->context) != 0) &&
|
||
(gdk_drag_context_get_dest_window (info->context) != NULL))
|
||
{
|
||
gtk_drag_end (info, event->time);
|
||
gtk_drag_drop (info, event->time);
|
||
}
|
||
else
|
||
{
|
||
gtk_drag_cancel_internal (info, GTK_DRAG_RESULT_NO_TARGET, event->time);
|
||
}
|
||
|
||
return TRUE;
|
||
|
||
case GDK_KEY_Up:
|
||
case GDK_KEY_KP_Up:
|
||
dy = (state & GDK_MOD1_MASK) ? -BIG_STEP : -SMALL_STEP;
|
||
break;
|
||
|
||
case GDK_KEY_Down:
|
||
case GDK_KEY_KP_Down:
|
||
dy = (state & GDK_MOD1_MASK) ? BIG_STEP : SMALL_STEP;
|
||
break;
|
||
|
||
case GDK_KEY_Left:
|
||
case GDK_KEY_KP_Left:
|
||
dx = (state & GDK_MOD1_MASK) ? -BIG_STEP : -SMALL_STEP;
|
||
break;
|
||
|
||
case GDK_KEY_Right:
|
||
case GDK_KEY_KP_Right:
|
||
dx = (state & GDK_MOD1_MASK) ? BIG_STEP : SMALL_STEP;
|
||
break;
|
||
}
|
||
}
|
||
|
||
/* Now send a "motion" so that the modifier state is updated */
|
||
|
||
/* The state is not yet updated in the event, so we need
|
||
* to query it here. We could use XGetModifierMapping, but
|
||
* that would be overkill.
|
||
*/
|
||
root_window = gdk_screen_get_root_window (gtk_widget_get_screen (widget));
|
||
gdk_window_get_device_position (root_window, pointer, NULL, NULL, &state);
|
||
event->state = state;
|
||
|
||
if (dx != 0 || dy != 0)
|
||
{
|
||
info->cur_x += dx;
|
||
info->cur_y += dy;
|
||
gdk_device_warp (pointer,
|
||
gtk_widget_get_screen (widget),
|
||
info->cur_x, info->cur_y);
|
||
}
|
||
|
||
gtk_drag_update (info, info->cur_screen, info->cur_x, info->cur_y, (GdkEvent *)event);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
static gboolean
|
||
gtk_drag_grab_broken_event_cb (GtkWidget *widget,
|
||
GdkEventGrabBroken *event,
|
||
gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = (GtkDragSourceInfo *)data;
|
||
|
||
/* Don't cancel if we break the implicit grab from the initial button_press.
|
||
* Also, don't cancel if we re-grab on the widget or on our IPC window, for
|
||
* example, when changing the drag cursor.
|
||
*/
|
||
if (event->implicit
|
||
|| event->grab_window == gtk_widget_get_window (info->widget)
|
||
|| event->grab_window == gtk_widget_get_window (info->ipc_widget))
|
||
return FALSE;
|
||
|
||
gtk_drag_cancel_internal (info, GTK_DRAG_RESULT_GRAB_BROKEN, gtk_get_current_event_time ());
|
||
return TRUE;
|
||
}
|
||
|
||
static void
|
||
gtk_drag_grab_notify_cb (GtkWidget *widget,
|
||
gboolean was_grabbed,
|
||
gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = (GtkDragSourceInfo *)data;
|
||
GdkDevice *pointer;
|
||
|
||
pointer = gdk_drag_context_get_device (info->context);
|
||
|
||
if (gtk_widget_device_is_shadowed (widget, pointer))
|
||
{
|
||
/* We have to block callbacks to avoid recursion here, because
|
||
* gtk_drag_cancel_internal calls gtk_grab_remove (via gtk_drag_end)
|
||
*/
|
||
g_signal_handlers_block_by_func (widget, gtk_drag_grab_notify_cb, data);
|
||
gtk_drag_cancel_internal (info, GTK_DRAG_RESULT_GRAB_BROKEN, gtk_get_current_event_time ());
|
||
g_signal_handlers_unblock_by_func (widget, gtk_drag_grab_notify_cb, data);
|
||
}
|
||
}
|
||
|
||
/* “button-release-event” callback during drag */
|
||
static gboolean
|
||
gtk_drag_button_release_cb (GtkWidget *widget,
|
||
GdkEventButton *event,
|
||
gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = (GtkDragSourceInfo *)data;
|
||
|
||
if (event->button != info->button)
|
||
return FALSE;
|
||
|
||
if ((gdk_drag_context_get_selected_action (info->context) != 0) &&
|
||
(gdk_drag_context_get_dest_window (info->context) != NULL))
|
||
{
|
||
gtk_drag_end (info, event->time);
|
||
gtk_drag_drop (info, event->time);
|
||
}
|
||
else
|
||
{
|
||
gtk_drag_cancel_internal (info, GTK_DRAG_RESULT_NO_TARGET, event->time);
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
static gboolean
|
||
gtk_drag_abort_timeout (gpointer data)
|
||
{
|
||
GtkDragSourceInfo *info = data;
|
||
guint32 time = GDK_CURRENT_TIME;
|
||
|
||
if (info->proxy_dest)
|
||
time = info->proxy_dest->proxy_drop_time;
|
||
|
||
info->drop_timeout = 0;
|
||
gtk_drag_drop_finished (info, GTK_DRAG_RESULT_TIMEOUT_EXPIRED, time);
|
||
|
||
return FALSE;
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_check_threshold: (method)
|
||
* @widget: a #GtkWidget
|
||
* @start_x: X coordinate of start of drag
|
||
* @start_y: Y coordinate of start of drag
|
||
* @current_x: current X coordinate
|
||
* @current_y: current Y coordinate
|
||
*
|
||
* Checks to see if a mouse drag starting at (@start_x, @start_y) and ending
|
||
* at (@current_x, @current_y) has passed the GTK+ drag threshold, and thus
|
||
* should trigger the beginning of a drag-and-drop operation.
|
||
*
|
||
* Returns: %TRUE if the drag threshold has been passed.
|
||
*/
|
||
gboolean
|
||
gtk_drag_check_threshold (GtkWidget *widget,
|
||
gint start_x,
|
||
gint start_y,
|
||
gint current_x,
|
||
gint current_y)
|
||
{
|
||
gint drag_threshold;
|
||
|
||
g_return_val_if_fail (GTK_IS_WIDGET (widget), FALSE);
|
||
|
||
drag_threshold = gtk_settings_get_dnd_drag_threshold (gtk_widget_get_settings (widget));
|
||
|
||
return (ABS (current_x - start_x) > drag_threshold ||
|
||
ABS (current_y - start_y) > drag_threshold);
|
||
}
|
||
|
||
/**
|
||
* gtk_drag_cancel: (method)
|
||
* @context: a #GdkDragContext, as e.g. returned by gtk_drag_begin_with_coordinates()
|
||
*
|
||
* Cancels an ongoing drag operation on the source side.
|
||
*
|
||
* If you want to be able to cancel a drag operation in this way,
|
||
* you need to keep a pointer to the drag context, either from an
|
||
* explicit call to gtk_drag_begin_with_coordinates(), or by
|
||
* connecting to #GtkWidget::drag-begin.
|
||
*
|
||
* If @context does not refer to an ongoing drag operation, this
|
||
* function does nothing.
|
||
*
|
||
* If a drag is cancelled in this way, the @result argument of
|
||
* #GtkWidget::drag-failed is set to @GTK_DRAG_RESULT_ERROR.
|
||
*
|
||
* Since: 3.16
|
||
*/
|
||
void
|
||
gtk_drag_cancel (GdkDragContext *context)
|
||
{
|
||
GtkDragSourceInfo *info;
|
||
|
||
g_return_if_fail (GDK_IS_DRAG_CONTEXT (context));
|
||
|
||
info = gtk_drag_get_source_info (context, FALSE);
|
||
if (info != NULL)
|
||
gtk_drag_cancel_internal (info, GTK_DRAG_RESULT_ERROR, gtk_get_current_event_time ());
|
||
}
|