3319 lines
		
	
	
		
			96 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			3319 lines
		
	
	
		
			96 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GDK - The GIMP Drawing Kit
 | ||
|  * Copyright (C) 1995-1999 Peter Mattis, Spencer Kimball and Josh MacDonald
 | ||
|  * Copyright (C) 2001 Archaeopteryx Software Inc.
 | ||
|  * Copyright (C) 1998-2002 Tor Lillqvist
 | ||
|  *
 | ||
|  * 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 <string.h>
 | ||
| 
 | ||
| #include <io.h>
 | ||
| #include <fcntl.h>
 | ||
| 
 | ||
| /*
 | ||
|  * Support for OLE-2 drag and drop added at Archaeopteryx Software, 2001
 | ||
|  * For more information, contact Stephan R.A. Deibel (sdeibel@archaeopteryx.com)
 | ||
|  *
 | ||
|  * Notes on the implementation:
 | ||
|  *
 | ||
|  * Source drag context, IDragSource and IDataObject for it are created
 | ||
|  * (almost) simultaneously, whereas target drag context and IDropTarget
 | ||
|  * are separated in time - IDropTarget is created when a window is made
 | ||
|  * to accept drops, while target drag context is created when a dragging
 | ||
|  * cursor enters the window and is destroyed when that cursor leaves
 | ||
|  * the window.
 | ||
|  *
 | ||
|  * There's a mismatch between data types supported by W32 (W32 formats)
 | ||
|  * and by GTK+ (GDK targets).
 | ||
|  * To account for it the data is transmuted back and forth. There are two
 | ||
|  * main points of transmutation:
 | ||
|  * * GDK convert selection: transmute W32 data to GTK+ data
 | ||
|  * * GDK window property change: transmute GTK+ data to W32 data
 | ||
|  *
 | ||
|  * There are also two points where data formats are considered:
 | ||
|  * * When source drag context is created, it gets a list of GTK+ targets
 | ||
|  *   that it supports, these are matched to the W32 formats they
 | ||
|  *   correspond to (possibly with transmutation). New W32 formats for
 | ||
|  *   GTK+-specific formats are also created here (see below).
 | ||
|  * * When target drag context is created, it queries the IDataObject
 | ||
|  *   for the list of W32 formats it supports and matches these to
 | ||
|  *   corresponding GTK+ formats that it will be able to provide
 | ||
|  *   (possibly with transmutation) later. Missing GDK targets for
 | ||
|  *   W32-specific formats are also created here (see below).
 | ||
|  *
 | ||
|  * W32 formats and GTK+ targets are both integers (CLIPFORMAT and GdkAtom
 | ||
|  * respectively), but cannot be used interchangeably.
 | ||
|  *
 | ||
|  * To accommodate advanced GTK+ applications the code allows them to
 | ||
|  * register drop targets that accept W32 data formats, and to register
 | ||
|  * drag sources that provide W32 data formats. To do that they must
 | ||
|  * register either with the string name of the format in question
 | ||
|  * (for example, "Shell IDList Array") or, for unnamed pre-defined
 | ||
|  * formats, register with the stringified constant name of the format
 | ||
|  * in question (for example, "CF_UNICODETEXT").
 | ||
|  * If such target format is accepted/provided, GDK will not try to
 | ||
|  * transmute it to/from something else. Otherwise GDK will do the following
 | ||
|  * transmutation:
 | ||
|  * * If GTK+ application provides image/png, image/gif or image/jpeg,
 | ||
|  *   GDK will claim to also provide "PNG", "GIF" or "JFIF" respectively,
 | ||
|  *   and will pass these along verbatim.
 | ||
|  * * If GTK+ application provides any GdkPixbuf-compatible target,
 | ||
|  *   GDK will also offer "PNG" and CF_DIB W32 formats.
 | ||
|  * * If GTK+ application provides UTF8_STRING, GDK will also offer
 | ||
|  *   CF_UNICODETEXT (UTF-16-encoded) and CF_TEXT (encoded with thread-
 | ||
|  *   and locale-depenant codepage), and will do the conversion when such
 | ||
|  *   data is requested.
 | ||
|  * * If GTK+ application accepts image/png, image/gif or image/jpeg,
 | ||
|  *   GDK will claim to also accept "PNG", "GIF" or "JFIF" respectively,
 | ||
|  *   and will pass these along verbatim.
 | ||
|  * * If GTK+ application accepts image/bmp, GDK will
 | ||
|  *   claim to accept CF_DIB W32 format, and will convert
 | ||
|  *   it, changing the header, when such data is provided.
 | ||
|  * * If GTK+ application accepts UTF8_STRING, GDK will
 | ||
|  *   claim to accept CF_UNICODETEXT and CF_TEXT, and will do
 | ||
|  *   the conversion when such data is provided.
 | ||
|  * * If GTK+ application accepts text/uri-list, GDK will
 | ||
|  *   claim to accept "Shell IDList Array", and will do the
 | ||
|  *   conversion when such data is provided.
 | ||
|  *
 | ||
|  * Currently the conversion from text/uri-list to Shell IDList Array is not
 | ||
|  * implemented, so it's not possible to drag & drop files from GTK+
 | ||
|  * applications to non-GTK+ applications the same way one can drag files
 | ||
|  * from Windows Explorer.
 | ||
|  *
 | ||
|  * To accommodate GTK+ application compaibility the code allows
 | ||
|  * GTK+ applications to register drop targets that accept GTK+-specific
 | ||
|  * data formats, and to register drag sources that provide GTK+-specific
 | ||
|  * data formats. This is done by simply registering target atom names
 | ||
|  * as clipboard formats. This way two GTK+ applications can exchange
 | ||
|  * data in their native formats (both well-known ones, such as UTF8_STRING,
 | ||
|  * and special, known only to specific applications). This will work just
 | ||
|  * fine as long as both applications agree on what kind of data is stored
 | ||
|  * under such format exactly.
 | ||
|  *
 | ||
|  * Note that clipboard format space is limited, there can only be 16384
 | ||
|  * of them for a particular user session. Therefore it is highly inadvisable
 | ||
|  * to create and register such formats out of the whole cloth, dynamically.
 | ||
|  * If more flexibility is needed, register one format that has some
 | ||
|  * internal indicators of the kind of data it contains, then write the application
 | ||
|  * in such a way that it requests the data and inspects its header before deciding
 | ||
|  * whether to accept it or not. For details see GTK+ drag & drop documentation
 | ||
|  * on the "drag-motion" and "drag-data-received" signals.
 | ||
|  */
 | ||
| 
 | ||
| /* The mingw.org compiler does not export GUIDS in it's import library. To work
 | ||
|  * around that, define INITGUID to have the GUIDS declared. */
 | ||
| #if defined(__MINGW32__) && !defined(__MINGW64_VERSION_MAJOR)
 | ||
| #define INITGUID
 | ||
| #endif
 | ||
| 
 | ||
| /* For C-style COM wrapper macros */
 | ||
| #define COBJMACROS
 | ||
| 
 | ||
| #include "gdkdnd.h"
 | ||
| #include "gdkproperty.h"
 | ||
| #include "gdkinternals.h"
 | ||
| #include "gdkprivate-win32.h"
 | ||
| #include "gdkwin32.h"
 | ||
| #include "gdkwin32dnd.h"
 | ||
| #include "gdkdisplayprivate.h"
 | ||
| #include "gdk/gdkdndprivate.h"
 | ||
| #include "gdkwin32dnd-private.h"
 | ||
| #include "gdkdisplay-win32.h"
 | ||
| #include "gdkselection-win32.h"
 | ||
| 
 | ||
| #include <ole2.h>
 | ||
| 
 | ||
| #include <shlobj.h>
 | ||
| #include <shlguid.h>
 | ||
| #include <objidl.h>
 | ||
| 
 | ||
| #include <gdk/gdk.h>
 | ||
| #include <glib/gstdio.h>
 | ||
| 
 | ||
| /* from gdkselection-win32.c */
 | ||
| extern GdkAtom *known_pixbuf_formats;
 | ||
| extern int n_known_pixbuf_formats;
 | ||
| 
 | ||
| 
 | ||
| typedef enum {
 | ||
|   GDK_DRAG_STATUS_DRAG,
 | ||
|   GDK_DRAG_STATUS_MOTION_WAIT,
 | ||
|   GDK_DRAG_STATUS_ACTION_WAIT,
 | ||
|   GDK_DRAG_STATUS_DROP
 | ||
| } GdkDragStatus;
 | ||
| 
 | ||
| static GList *contexts;
 | ||
| static GdkDragContext *current_dest_drag = NULL;
 | ||
| static gboolean use_ole2_dnd = FALSE;
 | ||
| 
 | ||
| G_DEFINE_TYPE (GdkWin32DragContext, gdk_win32_drag_context, GDK_TYPE_DRAG_CONTEXT)
 | ||
| 
 | ||
| static void
 | ||
| move_drag_window (GdkDragContext *context,
 | ||
|                   guint           x_root,
 | ||
|                   guint           y_root)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   gdk_window_move (context_win32->drag_window,
 | ||
|                    x_root - context_win32->hot_x,
 | ||
|                    y_root - context_win32->hot_y);
 | ||
|   gdk_window_raise (context_win32->drag_window);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_init (GdkWin32DragContext *context)
 | ||
| {
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       contexts = g_list_prepend (contexts, context);
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|     }
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_context_init %p\n", context));
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_finalize (GObject *object)
 | ||
| {
 | ||
|   GdkDragContext *context;
 | ||
|   GdkWin32DragContext *context_win32;
 | ||
|   GdkWindow *drag_window;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_context_finalize %p\n", object));
 | ||
| 
 | ||
|   g_return_if_fail (GDK_IS_WIN32_DRAG_CONTEXT (object));
 | ||
| 
 | ||
|   context = GDK_DRAG_CONTEXT (object);
 | ||
|   context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       contexts = g_list_remove (contexts, context);
 | ||
| 
 | ||
|       if (context == current_dest_drag)
 | ||
| 	current_dest_drag = NULL;
 | ||
|     }
 | ||
| 
 | ||
|   drag_window = context_win32->drag_window;
 | ||
| 
 | ||
|   g_array_unref (context_win32->droptarget_format_target_map);
 | ||
| 
 | ||
|   G_OBJECT_CLASS (gdk_win32_drag_context_parent_class)->finalize (object);
 | ||
| 
 | ||
|   if (drag_window)
 | ||
|     gdk_window_destroy (drag_window);
 | ||
| }
 | ||
| 
 | ||
| /* Drag Contexts */
 | ||
| 
 | ||
| static GdkDragContext *
 | ||
| gdk_drag_context_new (GdkDisplay *display)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32;
 | ||
|   GdkWin32Display *win32_display = GDK_WIN32_DISPLAY (display);
 | ||
|   GdkDragContext *context;
 | ||
| 
 | ||
|   context_win32 = g_object_new (GDK_TYPE_WIN32_DRAG_CONTEXT, NULL);
 | ||
|   context = GDK_DRAG_CONTEXT(context_win32);
 | ||
|   context->display = display;
 | ||
| 
 | ||
|   gdk_drag_context_set_device (context, gdk_seat_get_pointer (gdk_display_get_default_seat (display)));
 | ||
| 
 | ||
|   if (win32_display->has_fixed_scale)
 | ||
|     context_win32->scale = win32_display->window_scale;
 | ||
|   else
 | ||
|     context_win32->scale = _gdk_win32_display_get_monitor_scale_factor (win32_display, NULL, NULL, NULL);
 | ||
| 
 | ||
|   context_win32->droptarget_format_target_map = g_array_new (FALSE, FALSE, sizeof (GdkSelTargetFormat));
 | ||
| 
 | ||
|   return context;
 | ||
| }
 | ||
| 
 | ||
| static GdkDragContext *
 | ||
| gdk_drag_context_find (gboolean   is_source,
 | ||
| 		       GdkWindow *source,
 | ||
| 		       GdkWindow *dest)
 | ||
| {
 | ||
|   GList *tmp_list = contexts;
 | ||
|   GdkDragContext *context;
 | ||
| 
 | ||
|   while (tmp_list)
 | ||
|     {
 | ||
|       context = (GdkDragContext *)tmp_list->data;
 | ||
| 
 | ||
|       if ((!context->is_source == !is_source) &&
 | ||
| 	  ((source == NULL) || (context->source_window && (context->source_window == source))) &&
 | ||
| 	  ((dest == NULL) || (context->dest_window && (context->dest_window == dest))))
 | ||
| 	return context;
 | ||
| 
 | ||
|       tmp_list = tmp_list->next;
 | ||
|     }
 | ||
| 
 | ||
|   return NULL;
 | ||
| }
 | ||
| 
 | ||
| #define PRINT_GUID(guid) \
 | ||
|   g_print ("%.08lx-%.04x-%.04x-%.02x%.02x-%.02x%.02x%.02x%.02x%.02x%.02x", \
 | ||
| 	   ((gulong *)  guid)[0], \
 | ||
| 	   ((gushort *) guid)[2], \
 | ||
| 	   ((gushort *) guid)[3], \
 | ||
| 	   ((guchar *)  guid)[8], \
 | ||
| 	   ((guchar *)  guid)[9], \
 | ||
| 	   ((guchar *)  guid)[10], \
 | ||
| 	   ((guchar *)  guid)[11], \
 | ||
| 	   ((guchar *)  guid)[12], \
 | ||
| 	   ((guchar *)  guid)[13], \
 | ||
| 	   ((guchar *)  guid)[14], \
 | ||
| 	   ((guchar *)  guid)[15]);
 | ||
| 
 | ||
| typedef struct {
 | ||
|   IDropTarget idt;
 | ||
|   GdkDragContext *context;
 | ||
| 
 | ||
|   gint ref_count;
 | ||
|   GdkWindow *dest_window;
 | ||
| 
 | ||
| } target_drag_context;
 | ||
| 
 | ||
| typedef struct {
 | ||
|   IDropSource ids;
 | ||
|   GdkDragContext *context;
 | ||
|   gint ref_count;
 | ||
| } source_drag_context;
 | ||
| 
 | ||
| typedef struct {
 | ||
|   IDataObject ido;
 | ||
|   int ref_count;
 | ||
|   GdkDragContext *context;
 | ||
|   GArray *formats;
 | ||
| } data_object;
 | ||
| 
 | ||
| typedef struct {
 | ||
|   IEnumFORMATETC ief;
 | ||
|   int ref_count;
 | ||
|   int ix;
 | ||
|   data_object *dataobj;
 | ||
| } enum_formats;
 | ||
| 
 | ||
| static source_drag_context *pending_src_context = NULL;
 | ||
| static source_drag_context *current_src_context = NULL;
 | ||
| static data_object         *current_src_object = NULL;
 | ||
| 
 | ||
| static enum_formats *enum_formats_new (data_object *dataobj);
 | ||
| 
 | ||
| /* map windows -> target drag contexts. The table
 | ||
|  * owns a ref to both objects.
 | ||
|  */
 | ||
| static GHashTable* target_ctx_for_window = NULL;
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| idroptarget_addref (LPDROPTARGET This)
 | ||
| {
 | ||
|   target_drag_context *ctx = (target_drag_context *) This;
 | ||
| 
 | ||
|   int ref_count = ++ctx->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_addref %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idroptarget_queryinterface (LPDROPTARGET This,
 | ||
| 			    REFIID       riid,
 | ||
| 			    LPVOID      *ppvObject)
 | ||
| {
 | ||
|   GDK_NOTE (DND, {
 | ||
|       g_print ("idroptarget_queryinterface %p ", This);
 | ||
|       PRINT_GUID (riid);
 | ||
|     });
 | ||
| 
 | ||
|   *ppvObject = NULL;
 | ||
| 
 | ||
|   if (IsEqualGUID (riid, &IID_IUnknown))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IUnknown S_OK\n"));
 | ||
|       idroptarget_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else if (IsEqualGUID (riid, &IID_IDropTarget))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IDropTarget S_OK\n"));
 | ||
|       idroptarget_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...E_NOINTERFACE\n"));
 | ||
|       return E_NOINTERFACE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| idroptarget_release (LPDROPTARGET This)
 | ||
| {
 | ||
|   target_drag_context *ctx = (target_drag_context *) This;
 | ||
| 
 | ||
|   int ref_count = --ctx->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_release %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   if (ref_count == 0)
 | ||
|     {
 | ||
|       g_object_unref (ctx->context);
 | ||
|       g_clear_object (&ctx->dest_window);
 | ||
|       g_free (This);
 | ||
|     }
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| static GdkDragAction
 | ||
| get_suggested_action (DWORD grfKeyState)
 | ||
| {
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
|   /* This is the yucky Windows standard: Force link action if both
 | ||
|    * Control and Alt are down, copy if Control is down alone, move if
 | ||
|    * Alt is down alone, or use default of move within the app or copy
 | ||
|    * when origin of the drag is in another app.
 | ||
|    */
 | ||
|   if (grfKeyState & MK_CONTROL && grfKeyState & MK_SHIFT)
 | ||
|     return GDK_ACTION_LINK; /* Link action not supported */
 | ||
|   else if (grfKeyState & MK_CONTROL)
 | ||
|     return GDK_ACTION_COPY;
 | ||
|   else if (grfKeyState & MK_ALT)
 | ||
|     return GDK_ACTION_MOVE;
 | ||
|   else if (sel_win32->dnd_source_state == GDK_WIN32_DND_DRAGGING)
 | ||
|     return GDK_ACTION_MOVE;
 | ||
|   else
 | ||
|     return GDK_ACTION_COPY;
 | ||
|   /* Any way to determine when to add in DROPEFFECT_SCROLL? */
 | ||
| }
 | ||
| 
 | ||
| /* Process pending events -- we don't want to service non-GUI events
 | ||
|  * forever so do one iteration and then do more only if there’s a
 | ||
|  * pending GDK event.
 | ||
|  */
 | ||
| static void
 | ||
| process_pending_events (GdkDisplay *display)
 | ||
| {
 | ||
|   g_main_context_iteration (NULL, FALSE);
 | ||
|   while (_gdk_event_queue_find_first (display))
 | ||
|     g_main_context_iteration (NULL, FALSE);
 | ||
| }
 | ||
| 
 | ||
| static DWORD
 | ||
| drop_effect_for_action (GdkDragAction action)
 | ||
| {
 | ||
|   switch (action)
 | ||
|     {
 | ||
|     case GDK_ACTION_MOVE:
 | ||
|       return DROPEFFECT_MOVE;
 | ||
|     case GDK_ACTION_LINK:
 | ||
|       return DROPEFFECT_LINK;
 | ||
|     case GDK_ACTION_COPY:
 | ||
|       return DROPEFFECT_COPY;
 | ||
|     default:
 | ||
|       return DROPEFFECT_NONE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static GdkDragAction
 | ||
| action_for_drop_effect (DWORD effect)
 | ||
| {
 | ||
|   switch (effect)
 | ||
|     {
 | ||
|     case DROPEFFECT_MOVE:
 | ||
|       return GDK_ACTION_MOVE;
 | ||
|     case DROPEFFECT_LINK:
 | ||
|       return GDK_ACTION_LINK;
 | ||
|     case DROPEFFECT_COPY:
 | ||
|       return GDK_ACTION_COPY;
 | ||
|     default:
 | ||
|       return 0;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| dnd_event_put (GdkEventType    type,
 | ||
| 	       GdkDragContext *context,
 | ||
| 	       gint            pt_x,
 | ||
| 	       gint            pt_y,
 | ||
| 	       gboolean        to_dest_window)
 | ||
| {
 | ||
|   GdkEvent *e;
 | ||
| 
 | ||
|   e = gdk_event_new (type);
 | ||
| 
 | ||
|   if (to_dest_window)
 | ||
|     g_set_object (&e->dnd.window, context->dest_window);
 | ||
|   else
 | ||
|     g_set_object (&e->dnd.window, context->source_window);
 | ||
|   e->dnd.send_event = FALSE;
 | ||
|   g_set_object (&e->dnd.context, context);
 | ||
|   e->dnd.time = GDK_CURRENT_TIME;
 | ||
|   e->dnd.x_root = pt_x;
 | ||
|   e->dnd.y_root = pt_y;
 | ||
| 
 | ||
|   gdk_event_set_device (e, gdk_drag_context_get_device (context));
 | ||
|   gdk_event_set_seat (e, gdk_device_get_seat (gdk_drag_context_get_device (context)));
 | ||
| 
 | ||
|   GDK_NOTE (EVENTS, _gdk_win32_print_event (e));
 | ||
|   gdk_event_put (e);
 | ||
|   gdk_event_free (e);
 | ||
| }
 | ||
| 
 | ||
| static GList *
 | ||
| query_targets (LPDATAOBJECT  pDataObj,
 | ||
|                GArray       *format_target_map)
 | ||
| {
 | ||
|   IEnumFORMATETC *pfmt = NULL;
 | ||
|   FORMATETC fmt;
 | ||
|   GList *result = NULL;
 | ||
|   HRESULT hr;
 | ||
| 
 | ||
|   if ((LPDATAOBJECT) current_src_object == pDataObj)
 | ||
|     return g_list_copy (current_src_object->context->targets);
 | ||
| 
 | ||
|   hr = IDataObject_EnumFormatEtc (pDataObj, DATADIR_GET, &pfmt);
 | ||
| 
 | ||
|   if (SUCCEEDED (hr))
 | ||
|     hr = IEnumFORMATETC_Next (pfmt, 1, &fmt, NULL);
 | ||
| 
 | ||
|   while (SUCCEEDED (hr) && hr != S_FALSE)
 | ||
|   {
 | ||
|     gboolean is_predef;
 | ||
|     gchar *registered_name = _gdk_win32_get_clipboard_format_name (fmt.cfFormat, &is_predef);
 | ||
| 
 | ||
|     if (registered_name && is_predef)
 | ||
|       GDK_NOTE (DND, g_print ("supported built-in source format 0x%x %s\n", fmt.cfFormat, registered_name));
 | ||
|     else if (registered_name)
 | ||
|       GDK_NOTE (DND, g_print ("supported source format 0x%x %s\n", fmt.cfFormat, registered_name));
 | ||
|     else
 | ||
|       GDK_NOTE (DND, g_print ("supported unnamed? source format 0x%x\n", fmt.cfFormat));
 | ||
| 
 | ||
|     g_free (registered_name);
 | ||
| 
 | ||
|     _gdk_win32_add_format_to_targets (fmt.cfFormat, format_target_map, &result);
 | ||
|     hr = IEnumFORMATETC_Next (pfmt, 1, &fmt, NULL);
 | ||
|   }
 | ||
| 
 | ||
|   if (pfmt)
 | ||
|     IEnumFORMATETC_Release (pfmt);
 | ||
| 
 | ||
|   return g_list_reverse (result);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| set_data_object (LPDATAOBJECT *location, LPDATAOBJECT data_object)
 | ||
| {
 | ||
|   if (*location != NULL)
 | ||
|     IDataObject_Release (*location);
 | ||
| 
 | ||
|   *location = data_object;
 | ||
| 
 | ||
|   if (*location != NULL)
 | ||
|     IDataObject_AddRef (*location);
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idroptarget_dragenter (LPDROPTARGET This,
 | ||
| 		       LPDATAOBJECT pDataObj,
 | ||
| 		       DWORD        grfKeyState,
 | ||
| 		       POINTL       pt,
 | ||
| 		       LPDWORD      pdwEffect)
 | ||
| {
 | ||
|   target_drag_context *ctx = (target_drag_context *) This;
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
|   GdkDragContext *context;
 | ||
|   GdkWin32DragContext *context_win32;
 | ||
|   gint pt_x;
 | ||
|   gint pt_y;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_dragenter %p @ %ld : %ld for dest window 0x%p S_OK\n", This, pt.x, pt.y, ctx->dest_window));
 | ||
| 
 | ||
|   g_clear_object (&ctx->context);
 | ||
| 
 | ||
|   context = gdk_drag_context_new (gdk_window_get_display (ctx->dest_window));
 | ||
|   context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   ctx->context = context;
 | ||
|   g_set_object (&context->dest_window, ctx->dest_window);
 | ||
| 
 | ||
|   context->protocol = GDK_DRAG_PROTO_OLE2;
 | ||
|   context->is_source = FALSE;
 | ||
|   context->source_window = NULL;
 | ||
| 
 | ||
|   /* OLE2 DnD does not allow us to get the source window,
 | ||
|    * but we *can* find it if it's ours. This is needed to
 | ||
|    * support DnD within the same widget, for example.
 | ||
|    */
 | ||
|   if (current_src_context && current_src_context->context)
 | ||
|     g_set_object (&context->source_window, current_src_context->context->source_window);
 | ||
|   else
 | ||
|     g_set_object (&context->source_window, gdk_get_default_root_window ());
 | ||
| 
 | ||
|   g_set_object (&sel_win32->target_drag_context, context);
 | ||
|   context->actions = GDK_ACTION_DEFAULT | GDK_ACTION_COPY | GDK_ACTION_MOVE;
 | ||
|   context->suggested_action = GDK_ACTION_MOVE;
 | ||
|   context->action = GDK_ACTION_MOVE;
 | ||
| 
 | ||
|   g_array_set_size (context_win32->droptarget_format_target_map, 0);
 | ||
|   context->targets = query_targets (pDataObj, context_win32->droptarget_format_target_map);
 | ||
| 
 | ||
|   ctx->context->suggested_action = get_suggested_action (grfKeyState);
 | ||
|   set_data_object (&sel_win32->dnd_data_object_target, pDataObj);
 | ||
|   pt_x = (pt.x + _gdk_offset_x) / context_win32->scale;
 | ||
|   pt_y = (pt.y + _gdk_offset_y) / context_win32->scale;
 | ||
|   dnd_event_put (GDK_DRAG_ENTER, ctx->context, pt_x, pt_y, TRUE);
 | ||
|   dnd_event_put (GDK_DRAG_MOTION, ctx->context, pt_x, pt_y, TRUE);
 | ||
|   context_win32->last_key_state = grfKeyState;
 | ||
|   context_win32->last_x = pt_x;
 | ||
|   context_win32->last_y = pt_y;
 | ||
|   process_pending_events (gdk_device_get_display (gdk_drag_context_get_device (ctx->context)));
 | ||
|   *pdwEffect = drop_effect_for_action (ctx->context->action);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_dragenter returns with action %d and drop effect %lu\n", ctx->context->action, *pdwEffect));
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idroptarget_dragover (LPDROPTARGET This,
 | ||
| 		      DWORD        grfKeyState,
 | ||
| 		      POINTL       pt,
 | ||
| 		      LPDWORD      pdwEffect)
 | ||
| {
 | ||
|   target_drag_context *ctx = (target_drag_context *) This;
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (ctx->context);
 | ||
|   gint pt_x = (pt.x + _gdk_offset_x) / context_win32->scale;
 | ||
|   gint pt_y = (pt.y + _gdk_offset_y) / context_win32->scale;
 | ||
| 
 | ||
|   ctx->context->suggested_action = get_suggested_action (grfKeyState);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_dragover %p @ %ld : %ld, suggests %d action S_OK\n", This, pt.x, pt.y, ctx->context->suggested_action));
 | ||
| 
 | ||
|   if (pt_x != context_win32->last_x ||
 | ||
|       pt_y != context_win32->last_y ||
 | ||
|       grfKeyState != context_win32->last_key_state)
 | ||
|     {
 | ||
|       dnd_event_put (GDK_DRAG_MOTION, ctx->context, pt_x, pt_y, TRUE);
 | ||
|       context_win32->last_key_state = grfKeyState;
 | ||
|       context_win32->last_x = pt_x;
 | ||
|       context_win32->last_y = pt_y;
 | ||
|     }
 | ||
| 
 | ||
|   process_pending_events (gdk_device_get_display (gdk_drag_context_get_device (ctx->context)));
 | ||
| 
 | ||
|   *pdwEffect = drop_effect_for_action (ctx->context->action);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_dragover returns with action %d and effect %lu\n", ctx->context->action, *pdwEffect));
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idroptarget_dragleave (LPDROPTARGET This)
 | ||
| {
 | ||
|   target_drag_context *ctx = (target_drag_context *) This;
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_dragleave %p S_OK\n", This));
 | ||
| 
 | ||
|   dnd_event_put (GDK_DRAG_LEAVE, ctx->context, 0, 0, TRUE);
 | ||
|   process_pending_events (gdk_device_get_display (gdk_drag_context_get_device (ctx->context)));
 | ||
| 
 | ||
|   g_clear_object (&sel_win32->target_drag_context);
 | ||
|   g_clear_object (&ctx->context);
 | ||
|   set_data_object (&sel_win32->dnd_data_object_target, NULL);
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idroptarget_drop (LPDROPTARGET This,
 | ||
| 		  LPDATAOBJECT pDataObj,
 | ||
| 		  DWORD        grfKeyState,
 | ||
| 		  POINTL       pt,
 | ||
| 		  LPDWORD      pdwEffect)
 | ||
| {
 | ||
|   target_drag_context *ctx = (target_drag_context *) This;
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (ctx->context);
 | ||
|   gint pt_x = (pt.x + _gdk_offset_x) / context_win32->scale;
 | ||
|   gint pt_y = (pt.y + _gdk_offset_y) / context_win32->scale;
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idroptarget_drop %p ", This));
 | ||
| 
 | ||
|   if (pDataObj == NULL)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("E_POINTER\n"));
 | ||
|       g_clear_object (&ctx->context);
 | ||
|       return E_POINTER;
 | ||
|     }
 | ||
| 
 | ||
|   ctx->context->suggested_action = get_suggested_action (grfKeyState);
 | ||
| 
 | ||
|   dnd_event_put (GDK_DROP_START, ctx->context, pt_x, pt_y, TRUE);
 | ||
|   process_pending_events (gdk_device_get_display (gdk_drag_context_get_device (ctx->context)));
 | ||
| 
 | ||
|   /* Notify OLE of copy or move */
 | ||
|   if (sel_win32->dnd_target_state != GDK_WIN32_DND_DROPPED)
 | ||
|     *pdwEffect = DROPEFFECT_NONE;
 | ||
|   else
 | ||
|     *pdwEffect = drop_effect_for_action (ctx->context->action);
 | ||
| 
 | ||
|   g_clear_object (&sel_win32->target_drag_context);
 | ||
|   g_clear_object (&ctx->context);
 | ||
|   set_data_object (&sel_win32->dnd_data_object_target, NULL);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("drop S_OK with effect %lx\n", *pdwEffect));
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| idropsource_addref (LPDROPSOURCE This)
 | ||
| {
 | ||
|   source_drag_context *ctx = (source_drag_context *) This;
 | ||
| 
 | ||
|   int ref_count = ++ctx->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idropsource_addref %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idropsource_queryinterface (LPDROPSOURCE This,
 | ||
| 			    REFIID       riid,
 | ||
| 			    LPVOID      *ppvObject)
 | ||
| {
 | ||
|   GDK_NOTE (DND, {
 | ||
|       g_print ("idropsource_queryinterface %p ", This);
 | ||
|       PRINT_GUID (riid);
 | ||
|     });
 | ||
| 
 | ||
|   *ppvObject = NULL;
 | ||
| 
 | ||
|   if (IsEqualGUID (riid, &IID_IUnknown))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IUnknown S_OK\n"));
 | ||
|       idropsource_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else if (IsEqualGUID (riid, &IID_IDropSource))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IDropSource S_OK\n"));
 | ||
|       idropsource_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...E_NOINTERFACE\n"));
 | ||
|       return E_NOINTERFACE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| idropsource_release (LPDROPSOURCE This)
 | ||
| {
 | ||
|   source_drag_context *ctx = (source_drag_context *) This;
 | ||
| 
 | ||
|   int ref_count = --ctx->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idropsource_release %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   if (ref_count == 0)
 | ||
|   {
 | ||
|     g_clear_object (&ctx->context);
 | ||
|     if (current_src_context == ctx)
 | ||
|       current_src_context = NULL;
 | ||
|     g_free (This);
 | ||
|   }
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| /* Emit GDK events for any changes in mouse events or control key
 | ||
|  * state since the last recorded state. Return true if any events
 | ||
|  * have been emitted and false otherwise.
 | ||
|  */
 | ||
| static gboolean
 | ||
| send_change_events (GdkDragContext *context,
 | ||
| 		    DWORD           key_state,
 | ||
| 		    gboolean        esc_pressed)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   POINT pt;
 | ||
|   POINT pt_client;
 | ||
|   HMONITOR monitor = NULL;
 | ||
|   gboolean changed = FALSE;
 | ||
|   HWND hwnd = GDK_WINDOW_HWND (context->source_window);
 | ||
|   LPARAM lparam;
 | ||
|   WPARAM wparam;
 | ||
|   gint pt_x;
 | ||
|   gint pt_y;
 | ||
| 
 | ||
|   if (!API_CALL (GetCursorPos, (&pt)))
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   /* Move the DND IPC window to the monitor the cursor is currently on.
 | ||
|      This esures that OLE2 DND works correctly even if the DPI awareness
 | ||
|      isn't per-monitor.
 | ||
|   */
 | ||
|   monitor = MonitorFromPoint (pt, MONITOR_DEFAULTTONEAREST);
 | ||
|   if (monitor != context_win32->last_monitor)
 | ||
|     {
 | ||
|       MONITORINFO mi;
 | ||
| 
 | ||
|       mi.cbSize = sizeof(mi);
 | ||
|       if (GetMonitorInfoW (monitor, &mi))
 | ||
|         {
 | ||
|           MoveWindow (hwnd, mi.rcWork.left, mi.rcWork.top, 1, 1, FALSE);
 | ||
|         }
 | ||
| 
 | ||
|       context_win32->last_monitor = monitor;
 | ||
|     }
 | ||
| 
 | ||
|   pt_client = pt;
 | ||
| 
 | ||
|   if (!API_CALL (ScreenToClient, (hwnd, &pt_client)))
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   pt_x = (pt.x + _gdk_offset_x) / context_win32->scale;
 | ||
|   pt_y = (pt.y + _gdk_offset_y) / context_win32->scale;
 | ||
| 
 | ||
|   if (pt_x != context_win32->last_x || pt_y != context_win32->last_y ||
 | ||
|       key_state != context_win32->last_key_state)
 | ||
|     {
 | ||
|       lparam = MAKELPARAM (pt_client.x, pt_client.y);
 | ||
|       wparam = key_state;
 | ||
|       if (pt_x != context_win32->last_x || pt_y != context_win32->last_y)
 | ||
| 	{
 | ||
| 	  GDK_NOTE (DND, g_print ("Sending WM_MOUSEMOVE (%ld,%ld)\n", pt.x, pt.y));
 | ||
|       	  SendMessage (hwnd, WM_MOUSEMOVE, wparam, lparam);
 | ||
| 	}
 | ||
| 
 | ||
|       if ((key_state & MK_LBUTTON) != (context_win32->last_key_state & MK_LBUTTON))
 | ||
| 	{
 | ||
| 	  if (key_state & MK_LBUTTON)
 | ||
| 	    SendMessage (hwnd, WM_LBUTTONDOWN, wparam, lparam);
 | ||
| 	  else
 | ||
| 	    SendMessage (hwnd, WM_LBUTTONUP, wparam, lparam);
 | ||
| 	}
 | ||
|       if ((key_state & MK_MBUTTON) != (context_win32->last_key_state & MK_MBUTTON))
 | ||
| 	{
 | ||
| 	  if (key_state & MK_MBUTTON)
 | ||
| 	    SendMessage (hwnd, WM_MBUTTONDOWN, wparam, lparam);
 | ||
| 	  else
 | ||
| 	    SendMessage (hwnd, WM_MBUTTONUP, wparam, lparam);
 | ||
| 	}
 | ||
|       if ((key_state & MK_RBUTTON) != (context_win32->last_key_state & MK_RBUTTON))
 | ||
| 	{
 | ||
| 	  if (key_state & MK_RBUTTON)
 | ||
| 	    SendMessage (hwnd, WM_RBUTTONDOWN, wparam, lparam);
 | ||
| 	  else
 | ||
| 	    SendMessage (hwnd, WM_RBUTTONUP, wparam, lparam);
 | ||
| 	}
 | ||
|       if ((key_state & MK_CONTROL) != (context_win32->last_key_state & MK_CONTROL))
 | ||
| 	{
 | ||
| 	  if (key_state & MK_CONTROL)
 | ||
| 	    SendMessage (hwnd, WM_KEYDOWN, VK_CONTROL, 0);
 | ||
| 	  else
 | ||
| 	    SendMessage (hwnd, WM_KEYUP, VK_CONTROL, 0);
 | ||
| 	}
 | ||
|       if ((key_state & MK_SHIFT) != (context_win32->last_key_state & MK_SHIFT))
 | ||
| 	{
 | ||
| 	  if (key_state & MK_SHIFT)
 | ||
| 	    SendMessage (hwnd, WM_KEYDOWN, VK_SHIFT, 0);
 | ||
| 	  else
 | ||
| 	    SendMessage (hwnd, WM_KEYUP, VK_SHIFT, 0);
 | ||
| 	}
 | ||
| 
 | ||
|       changed = TRUE;
 | ||
|       context_win32->last_key_state = key_state;
 | ||
|       context_win32->last_x = pt_x;
 | ||
|       context_win32->last_y = pt_y;
 | ||
|     }
 | ||
| 
 | ||
|   if (esc_pressed)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("Sending a escape key down message to %p\n", hwnd));
 | ||
|       SendMessage (hwnd, WM_KEYDOWN, VK_ESCAPE, 0);
 | ||
|       changed = TRUE;
 | ||
|     }
 | ||
| 
 | ||
|   return changed;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idropsource_querycontinuedrag (LPDROPSOURCE This,
 | ||
| 			       BOOL         fEscapePressed,
 | ||
| 			       DWORD        grfKeyState)
 | ||
| {
 | ||
|   source_drag_context *ctx = (source_drag_context *) This;
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idropsource_querycontinuedrag %p esc=%d keystate=0x%lx ", This, fEscapePressed, grfKeyState));
 | ||
| 
 | ||
|   if (send_change_events (ctx->context, grfKeyState, fEscapePressed))
 | ||
|     process_pending_events (gdk_device_get_display (gdk_drag_context_get_device (ctx->context)));
 | ||
| 
 | ||
|   if (sel_win32->dnd_source_state == GDK_WIN32_DND_DROPPED)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("DRAGDROP_S_DROP\n"));
 | ||
|       return DRAGDROP_S_DROP;
 | ||
|     }
 | ||
|   else if (sel_win32->dnd_source_state == GDK_WIN32_DND_NONE)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("DRAGDROP_S_CANCEL\n"));
 | ||
|       return DRAGDROP_S_CANCEL;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("S_OK\n"));
 | ||
|       return S_OK;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idropsource_givefeedback (LPDROPSOURCE This,
 | ||
| 			  DWORD        dwEffect)
 | ||
| {
 | ||
|   source_drag_context *ctx = (source_drag_context *) This;
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (ctx->context);
 | ||
|   GdkDragAction suggested_action;
 | ||
|   GdkEvent *e;
 | ||
|   POINT pt;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idropsource_givefeedback %p with drop effect %lu S_OK\n", This, dwEffect));
 | ||
| 
 | ||
|   if (!API_CALL (GetCursorPos, (&pt)))
 | ||
|     return S_OK;
 | ||
| 
 | ||
|   suggested_action = action_for_drop_effect (dwEffect);
 | ||
| 
 | ||
|   ctx->context->action = suggested_action;
 | ||
| 
 | ||
|   if (dwEffect == DROPEFFECT_NONE)
 | ||
|     g_clear_object (&ctx->context->dest_window);
 | ||
|   else if (ctx->context->dest_window == NULL)
 | ||
|     ctx->context->dest_window = g_object_ref (gdk_get_default_root_window ());
 | ||
| 
 | ||
|   context_win32->last_x = (pt.x + _gdk_offset_x) / context_win32->scale;
 | ||
|   context_win32->last_y = (pt.y + _gdk_offset_y) / context_win32->scale;
 | ||
| 
 | ||
|   e = gdk_event_new (GDK_DRAG_STATUS);
 | ||
| 
 | ||
|   g_set_object (&e->dnd.window, ctx->context->source_window);
 | ||
|   e->dnd.send_event = FALSE;
 | ||
|   g_set_object (&e->dnd.context, ctx->context);
 | ||
|   e->dnd.time = GDK_CURRENT_TIME;
 | ||
|   e->dnd.x_root = context_win32->last_x;
 | ||
|   e->dnd.y_root = context_win32->last_y;
 | ||
| 
 | ||
|   gdk_event_set_device (e, gdk_drag_context_get_device (ctx->context));
 | ||
|   gdk_event_set_seat (e, gdk_device_get_seat (gdk_drag_context_get_device (ctx->context)));
 | ||
| 
 | ||
|   GDK_NOTE (EVENTS, _gdk_win32_print_event (e));
 | ||
|   gdk_event_put (e);
 | ||
|   gdk_event_free (e);
 | ||
|   process_pending_events (gdk_device_get_display (gdk_drag_context_get_device (ctx->context)));
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idropsource_givefeedback %p returns\n", This));
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| idataobject_addref (LPDATAOBJECT This)
 | ||
| {
 | ||
|   data_object *dobj = (data_object *) This;
 | ||
|   int ref_count = ++dobj->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_addref %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_queryinterface (LPDATAOBJECT This,
 | ||
| 			    REFIID       riid,
 | ||
| 			    LPVOID      *ppvObject)
 | ||
| {
 | ||
|   GDK_NOTE (DND, {
 | ||
|       g_print ("idataobject_queryinterface %p ", This);
 | ||
|       PRINT_GUID (riid);
 | ||
|     });
 | ||
| 
 | ||
|   *ppvObject = NULL;
 | ||
| 
 | ||
|   if (IsEqualGUID (riid, &IID_IUnknown))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IUnknown S_OK\n"));
 | ||
|       idataobject_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else if (IsEqualGUID (riid, &IID_IDataObject))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IDataObject S_OK\n"));
 | ||
|       idataobject_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...E_NOINTERFACE\n"));
 | ||
|       return E_NOINTERFACE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| idataobject_release (LPDATAOBJECT This)
 | ||
| {
 | ||
|   data_object *dobj = (data_object *) This;
 | ||
|   int ref_count = --dobj->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_release %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   if (ref_count == 0)
 | ||
|     {
 | ||
|       g_array_free (dobj->formats, TRUE);
 | ||
|       g_free (This);
 | ||
|     }
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT
 | ||
| query (LPDATAOBJECT This,
 | ||
|        LPFORMATETC  pFormatEtc)
 | ||
| {
 | ||
|   data_object *ctx = (data_object *) This;
 | ||
|   gint i;
 | ||
| 
 | ||
|   if (!pFormatEtc)
 | ||
|     return DV_E_FORMATETC;
 | ||
| 
 | ||
|   if (pFormatEtc->lindex != -1)
 | ||
|     return DV_E_LINDEX;
 | ||
| 
 | ||
|   if ((pFormatEtc->tymed & TYMED_HGLOBAL) == 0)
 | ||
|     return DV_E_TYMED;
 | ||
| 
 | ||
|   if ((pFormatEtc->dwAspect & DVASPECT_CONTENT) == 0)
 | ||
|     return DV_E_DVASPECT;
 | ||
| 
 | ||
|   for (i = 0; i < ctx->formats->len; i++)
 | ||
|     if (pFormatEtc->cfFormat == g_array_index (ctx->formats, GdkSelTargetFormat, i).format)
 | ||
|       return S_OK;
 | ||
| 
 | ||
|   return DV_E_FORMATETC;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_getdata (LPDATAOBJECT This,
 | ||
| 		     LPFORMATETC  pFormatEtc,
 | ||
| 		     LPSTGMEDIUM  pMedium)
 | ||
| {
 | ||
|   data_object *ctx = (data_object *) This;
 | ||
|   GdkWin32Selection *win32_sel = _gdk_win32_selection_get ();
 | ||
|   HRESULT hr;
 | ||
|   GdkEvent e;
 | ||
|   gint i;
 | ||
|   GdkAtom target;
 | ||
|   gint64 loopend;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_getdata %p %s ",
 | ||
| 			  This, _gdk_win32_cf_to_string (pFormatEtc->cfFormat)));
 | ||
| 
 | ||
|   /* Check whether we can provide requested format */
 | ||
|   hr = query (This, pFormatEtc);
 | ||
|   if (hr != S_OK)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("Unsupported format, returning 0x%lx\n", hr));
 | ||
|       return hr;
 | ||
|     }
 | ||
| 
 | ||
|   /* Append a GDK_SELECTION_REQUEST event and then hope the app sets the
 | ||
|    * property associated with the _gdk_ole2_dnd atom
 | ||
|    */
 | ||
|   win32_sel->property_change_format = pFormatEtc->cfFormat;
 | ||
|   win32_sel->property_change_data = pMedium;
 | ||
| 
 | ||
|   for (i = 0, target = NULL; i < ctx->formats->len; i++)
 | ||
|     {
 | ||
|       GdkSelTargetFormat *frec = &g_array_index (ctx->formats, GdkSelTargetFormat, i);
 | ||
| 
 | ||
|       if (frec->format == pFormatEtc->cfFormat)
 | ||
|         {
 | ||
|           target = frec->target;
 | ||
|           win32_sel->property_change_transmute = frec->transmute;
 | ||
|           win32_sel->property_change_target_atom = frec->target;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|   if (target == NULL)
 | ||
|     {
 | ||
|       GDK_NOTE (EVENTS, g_print ("(target not found)"));
 | ||
|       return E_UNEXPECTED;
 | ||
|     }
 | ||
| 
 | ||
|   GDK_NOTE (DND, {
 | ||
|       gchar *target_name = gdk_atom_name (target);
 | ||
|       g_print ("idataobject_getdata will request target 0x%p (%s) ",
 | ||
|                target, target_name);
 | ||
|       g_free (target_name);
 | ||
|     });
 | ||
| 
 | ||
| 
 | ||
|   memset (&e, 0, sizeof (GdkEvent));
 | ||
|   e.type = GDK_SELECTION_REQUEST;
 | ||
|   g_set_object (&e.selection.window, ctx->context->source_window);
 | ||
|   e.selection.send_event = FALSE; /* ??? */
 | ||
|   /* Both selection and property are OLE2_DND, because change_property()
 | ||
|    * will only get the property and not the selection. Theoretically we
 | ||
|    * could use two different atoms (SELECTION_OLE2_DND and PROPERTY_OLE2_DND),
 | ||
|    * but there's little reason to do so.
 | ||
|    */
 | ||
|   e.selection.selection = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_OLE2_DND);
 | ||
|   e.selection.target = target;
 | ||
|   /* Requestor here is fake, just to allow the event to be processed */
 | ||
|   g_set_object (&e.selection.requestor, ctx->context->source_window);
 | ||
|   e.selection.property = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_OLE2_DND);
 | ||
|   e.selection.time = GDK_CURRENT_TIME;
 | ||
| 
 | ||
|   GDK_NOTE (EVENTS, _gdk_win32_print_event (&e));
 | ||
| 
 | ||
|   gdk_event_put (&e);
 | ||
| 
 | ||
|   /* Don't hold up longer than one second */
 | ||
|   loopend = g_get_monotonic_time () + 1000000000;
 | ||
| 
 | ||
|   while (win32_sel->property_change_data != 0 &&
 | ||
|          g_get_monotonic_time () < loopend)
 | ||
|     process_pending_events (gdk_device_get_display (gdk_drag_context_get_device (ctx->context)));
 | ||
| 
 | ||
|   if (pMedium->hGlobal == NULL) {
 | ||
|     GDK_NOTE (DND, g_print (" E_UNEXPECTED\n"));
 | ||
|     return E_UNEXPECTED;
 | ||
|   }
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print (" S_OK\n"));
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_getdatahere (LPDATAOBJECT This,
 | ||
| 			 LPFORMATETC  pFormatEtc,
 | ||
| 			 LPSTGMEDIUM  pMedium)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_getdatahere %p %s E_UNEXPECTED\n",
 | ||
| 			  This, _gdk_win32_cf_to_string (pFormatEtc->cfFormat)));
 | ||
| 
 | ||
|   return E_UNEXPECTED;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_querygetdata (LPDATAOBJECT This,
 | ||
| 			  LPFORMATETC  pFormatEtc)
 | ||
| {
 | ||
|   HRESULT hr;
 | ||
| 
 | ||
|   hr = query (This, pFormatEtc);
 | ||
| 
 | ||
| #define CASE(x) case x: g_print (#x "\n"); break
 | ||
|   GDK_NOTE (DND, {
 | ||
|       g_print ("idataobject_querygetdata %p %s ",
 | ||
| 	       This, _gdk_win32_cf_to_string (pFormatEtc->cfFormat));
 | ||
|       switch (hr)
 | ||
| 	{
 | ||
| 	CASE (DV_E_FORMATETC);
 | ||
| 	CASE (DV_E_LINDEX);
 | ||
| 	CASE (DV_E_TYMED);
 | ||
| 	CASE (DV_E_DVASPECT);
 | ||
| 	CASE (S_OK);
 | ||
| 	default: g_print ("%#lx", hr);
 | ||
| 	}
 | ||
|     });
 | ||
| 
 | ||
|   return hr;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_getcanonicalformatetc (LPDATAOBJECT This,
 | ||
| 				   LPFORMATETC  pFormatEtcIn,
 | ||
| 				   LPFORMATETC  pFormatEtcOut)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_getcanonicalformatetc %p E_UNEXPECTED\n", This));
 | ||
| 
 | ||
|   return E_UNEXPECTED;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_setdata (LPDATAOBJECT This,
 | ||
| 		     LPFORMATETC  pFormatEtc,
 | ||
| 		     LPSTGMEDIUM  pMedium,
 | ||
| 		     BOOL         fRelease)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_setdata %p %s E_UNEXPECTED\n",
 | ||
| 			  This, _gdk_win32_cf_to_string (pFormatEtc->cfFormat)));
 | ||
| 
 | ||
|   return E_UNEXPECTED;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_enumformatetc (LPDATAOBJECT     This,
 | ||
| 			   DWORD            dwDirection,
 | ||
| 			   LPENUMFORMATETC *ppEnumFormatEtc)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_enumformatetc %p ", This));
 | ||
| 
 | ||
|   if (dwDirection != DATADIR_GET)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("E_NOTIMPL\n"));
 | ||
|       return E_NOTIMPL;
 | ||
|     }
 | ||
| 
 | ||
|   *ppEnumFormatEtc = &enum_formats_new ((data_object *) This)->ief;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print (" %p S_OK\n", *ppEnumFormatEtc));
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_dadvise (LPDATAOBJECT This,
 | ||
| 		     LPFORMATETC  pFormatetc,
 | ||
| 		     DWORD        advf,
 | ||
| 		     LPADVISESINK pAdvSink,
 | ||
| 		     DWORD       *pdwConnection)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_dadvise %p E_NOTIMPL\n", This));
 | ||
| 
 | ||
|   return E_NOTIMPL;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_dunadvise (LPDATAOBJECT This,
 | ||
| 		       DWORD         dwConnection)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_dunadvise %p E_NOTIMPL\n", This));
 | ||
| 
 | ||
|   return E_NOTIMPL;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| idataobject_enumdadvise (LPDATAOBJECT    This,
 | ||
| 			 LPENUMSTATDATA *ppenumAdvise)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("idataobject_enumdadvise %p OLE_E_ADVISENOTSUPPORTED\n", This));
 | ||
| 
 | ||
|   return OLE_E_ADVISENOTSUPPORTED;
 | ||
| }
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| ienumformatetc_addref (LPENUMFORMATETC This)
 | ||
| {
 | ||
|   enum_formats *en = (enum_formats *) This;
 | ||
|   int ref_count = ++en->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("ienumformatetc_addref %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| ienumformatetc_queryinterface (LPENUMFORMATETC This,
 | ||
| 			       REFIID          riid,
 | ||
| 			       LPVOID         *ppvObject)
 | ||
| {
 | ||
|   GDK_NOTE (DND, {
 | ||
|       g_print ("ienumformatetc_queryinterface %p", This);
 | ||
|       PRINT_GUID (riid);
 | ||
|     });
 | ||
| 
 | ||
|   *ppvObject = NULL;
 | ||
| 
 | ||
|   if (IsEqualGUID (riid, &IID_IUnknown))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IUnknown S_OK\n"));
 | ||
|       ienumformatetc_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else if (IsEqualGUID (riid, &IID_IEnumFORMATETC))
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...IEnumFORMATETC S_OK\n"));
 | ||
|       ienumformatetc_addref (This);
 | ||
|       *ppvObject = This;
 | ||
|       return S_OK;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("...E_NOINTERFACE\n"));
 | ||
|       return E_NOINTERFACE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static ULONG STDMETHODCALLTYPE
 | ||
| ienumformatetc_release (LPENUMFORMATETC This)
 | ||
| {
 | ||
|   enum_formats *en = (enum_formats *) This;
 | ||
|   int ref_count = --en->ref_count;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("ienumformatetc_release %p %d\n", This, ref_count));
 | ||
| 
 | ||
|   if (ref_count == 0)
 | ||
|     {
 | ||
|       idataobject_release ((LPDATAOBJECT) en->dataobj);
 | ||
|       g_free (This);
 | ||
|     }
 | ||
| 
 | ||
|   return ref_count;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| ienumformatetc_next (LPENUMFORMATETC This,
 | ||
| 		     ULONG	     celt,
 | ||
| 		     LPFORMATETC     elts,
 | ||
| 		     ULONG	    *nelt)
 | ||
| {
 | ||
|   enum_formats *en = (enum_formats *) This;
 | ||
|   ULONG i, n;
 | ||
|   ULONG formats_to_get = celt;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("ienumformatetc_next %p %d %ld ", This, en->ix, celt));
 | ||
| 
 | ||
|   n = 0;
 | ||
|   for (i = 0; i < formats_to_get; i++)
 | ||
|     {
 | ||
|       UINT fmt;
 | ||
|       if (en->ix >= en->dataobj->formats->len)
 | ||
| 	break;
 | ||
|       fmt = g_array_index (en->dataobj->formats, GdkSelTargetFormat, en->ix++).format;
 | ||
|       /* skip internals */
 | ||
|       if (fmt == 0 || fmt > 0xFFFF)
 | ||
|         {
 | ||
|           formats_to_get += 1;
 | ||
|           continue;
 | ||
|         }
 | ||
|       elts[n].cfFormat = fmt;
 | ||
|       elts[n].ptd = NULL;
 | ||
|       elts[n].dwAspect = DVASPECT_CONTENT;
 | ||
|       elts[n].lindex = -1;
 | ||
|       elts[n].tymed = TYMED_HGLOBAL;
 | ||
| 
 | ||
|       n++;
 | ||
|     }
 | ||
| 
 | ||
|   if (nelt != NULL)
 | ||
|     *nelt = n;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("%s\n", (n == celt) ? "S_OK" : "S_FALSE"));
 | ||
| 
 | ||
|   if (n == celt)
 | ||
|     return S_OK;
 | ||
|   else
 | ||
|     return S_FALSE;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| ienumformatetc_skip (LPENUMFORMATETC This,
 | ||
| 		     ULONG	     celt)
 | ||
| {
 | ||
|   enum_formats *en = (enum_formats *) This;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("ienumformatetc_skip %p %d %ld S_OK\n", This, en->ix, celt));
 | ||
| 
 | ||
|   en->ix += celt;
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| ienumformatetc_reset (LPENUMFORMATETC This)
 | ||
| {
 | ||
|   enum_formats *en = (enum_formats *) This;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("ienumformatetc_reset %p S_OK\n", This));
 | ||
| 
 | ||
|   en->ix = 0;
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static HRESULT STDMETHODCALLTYPE
 | ||
| ienumformatetc_clone (LPENUMFORMATETC  This,
 | ||
| 		      LPENUMFORMATETC *ppEnumFormatEtc)
 | ||
| {
 | ||
|   enum_formats *en = (enum_formats *) This;
 | ||
|   enum_formats *new;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("ienumformatetc_clone %p S_OK\n", This));
 | ||
| 
 | ||
|   new = enum_formats_new (en->dataobj);
 | ||
| 
 | ||
|   new->ix = en->ix;
 | ||
| 
 | ||
|   *ppEnumFormatEtc = &new->ief;
 | ||
| 
 | ||
|   return S_OK;
 | ||
| }
 | ||
| 
 | ||
| static IDropTargetVtbl idt_vtbl = {
 | ||
|   idroptarget_queryinterface,
 | ||
|   idroptarget_addref,
 | ||
|   idroptarget_release,
 | ||
|   idroptarget_dragenter,
 | ||
|   idroptarget_dragover,
 | ||
|   idroptarget_dragleave,
 | ||
|   idroptarget_drop
 | ||
| };
 | ||
| 
 | ||
| static IDropSourceVtbl ids_vtbl = {
 | ||
|   idropsource_queryinterface,
 | ||
|   idropsource_addref,
 | ||
|   idropsource_release,
 | ||
|   idropsource_querycontinuedrag,
 | ||
|   idropsource_givefeedback
 | ||
| };
 | ||
| 
 | ||
| static IDataObjectVtbl ido_vtbl = {
 | ||
|   idataobject_queryinterface,
 | ||
|   idataobject_addref,
 | ||
|   idataobject_release,
 | ||
|   idataobject_getdata,
 | ||
|   idataobject_getdatahere,
 | ||
|   idataobject_querygetdata,
 | ||
|   idataobject_getcanonicalformatetc,
 | ||
|   idataobject_setdata,
 | ||
|   idataobject_enumformatetc,
 | ||
|   idataobject_dadvise,
 | ||
|   idataobject_dunadvise,
 | ||
|   idataobject_enumdadvise
 | ||
| };
 | ||
| 
 | ||
| static IEnumFORMATETCVtbl ief_vtbl = {
 | ||
|   ienumformatetc_queryinterface,
 | ||
|   ienumformatetc_addref,
 | ||
|   ienumformatetc_release,
 | ||
|   ienumformatetc_next,
 | ||
|   ienumformatetc_skip,
 | ||
|   ienumformatetc_reset,
 | ||
|   ienumformatetc_clone
 | ||
| };
 | ||
| 
 | ||
| 
 | ||
| static target_drag_context *
 | ||
| target_context_new (GdkWindow *window)
 | ||
| {
 | ||
|   target_drag_context *result;
 | ||
| 
 | ||
|   result = g_new0 (target_drag_context, 1);
 | ||
|   result->idt.lpVtbl = &idt_vtbl;
 | ||
|   result->ref_count = 0;
 | ||
| 
 | ||
|   result->dest_window = g_object_ref (window);
 | ||
| 
 | ||
|   idroptarget_addref (&result->idt);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("target_context_new: %p (window %p)\n", result, result->dest_window));
 | ||
| 
 | ||
|   return result;
 | ||
| }
 | ||
| 
 | ||
| static source_drag_context *
 | ||
| source_context_new (GdkWindow *window,
 | ||
| 		    GList     *targets)
 | ||
| {
 | ||
|   GdkDragContext *context;
 | ||
|   GdkWin32DragContext *context_win32;
 | ||
|   source_drag_context *result;
 | ||
|   GList *p;
 | ||
| 
 | ||
|   context = gdk_drag_context_new (gdk_window_get_display (window));
 | ||
|   context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   result = g_new0 (source_drag_context, 1);
 | ||
|   result->context = context;
 | ||
|   result->ids.lpVtbl = &ids_vtbl;
 | ||
|   result->ref_count = 0;
 | ||
| 
 | ||
|   result->context->protocol = GDK_DRAG_PROTO_OLE2;
 | ||
|   result->context->is_source = TRUE;
 | ||
| 
 | ||
|   result->context->source_window = g_object_ref (window);
 | ||
| 
 | ||
|   result->context->dest_window = NULL;
 | ||
|   result->context->targets = g_list_copy (targets);
 | ||
|   context_win32->has_image_format = FALSE;
 | ||
|   context_win32->has_cf_png = FALSE;
 | ||
|   context_win32->has_cf_dib = FALSE;
 | ||
|   context_win32->has_text_uri_list = FALSE;
 | ||
|   context_win32->has_shell_id_list = FALSE;
 | ||
|   context_win32->has_unicodetext = FALSE;
 | ||
| 
 | ||
|   for (p = result->context->targets; p; p = g_list_next (p))
 | ||
|     {
 | ||
|       gint j;
 | ||
|       GdkAtom target = p->data;
 | ||
| 
 | ||
|       if (target == _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_TEXT_URI_LIST))
 | ||
|         context_win32->has_text_uri_list = TRUE;
 | ||
|       else if (target == _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_CFSTR_SHELLIDLIST))
 | ||
|         context_win32->has_shell_id_list = TRUE;
 | ||
|       else if (target == _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_PNG))
 | ||
|         context_win32->has_cf_png = TRUE;
 | ||
|       else if (target == _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_CF_DIB))
 | ||
|         context_win32->has_cf_dib = TRUE;
 | ||
|       else if (target == _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_GIF))
 | ||
|         context_win32->has_gif = TRUE;
 | ||
|       else if (target == _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_JFIF))
 | ||
|         context_win32->has_jfif = TRUE;
 | ||
|       else if (target == _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_CF_UNICODETEXT))
 | ||
|         context_win32->has_unicodetext = TRUE;
 | ||
| 
 | ||
|       for (j = 0; !context_win32->has_image_format && j < _gdk_win32_selection_get ()->n_known_pixbuf_formats; j++)
 | ||
|         if (target == _gdk_win32_selection_get ()->known_pixbuf_formats[j])
 | ||
|           context_win32->has_image_format = TRUE;
 | ||
|     }
 | ||
| 
 | ||
|   idropsource_addref (&result->ids);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("source_context_new: %p (drag context %p)\n", result, result->context));
 | ||
| 
 | ||
|   if (current_src_context == NULL)
 | ||
|     current_src_context = result;
 | ||
| 
 | ||
|   return result;
 | ||
| }
 | ||
| 
 | ||
| static data_object *
 | ||
| data_object_new (GdkDragContext *context)
 | ||
| {
 | ||
|   data_object *result;
 | ||
|   GList *p;
 | ||
| 
 | ||
|   result = g_new0 (data_object, 1);
 | ||
| 
 | ||
|   result->ido.lpVtbl = &ido_vtbl;
 | ||
|   result->ref_count = 1;
 | ||
|   result->context = context;
 | ||
|   result->formats = g_array_new (FALSE, FALSE, sizeof (GdkSelTargetFormat));
 | ||
| 
 | ||
|   for (p = context->targets; p; p = p->next)
 | ||
|     {
 | ||
|       GdkAtom target = p->data;
 | ||
|       gchar *target_name = gdk_atom_name (target);
 | ||
|       gint added_count = 0;
 | ||
|       gint i;
 | ||
| 
 | ||
|       GDK_NOTE (DND, g_print ("DataObject supports target 0x%p (%s)\n", target, target_name));
 | ||
|       g_free (target_name);
 | ||
| 
 | ||
|       added_count = _gdk_win32_add_target_to_selformats (target, result->formats);
 | ||
| 
 | ||
|       for (i = 0; i < added_count && result->formats->len - 1 - i >= 0; i++)
 | ||
|         GDK_NOTE (DND, g_print ("DataObject will support format 0x%x\n", g_array_index (result->formats, GdkSelTargetFormat, i).format));
 | ||
|     }
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("data_object_new: %p\n", result));
 | ||
| 
 | ||
|   return result;
 | ||
| }
 | ||
| 
 | ||
| static enum_formats *
 | ||
| enum_formats_new (data_object *dataobj)
 | ||
| {
 | ||
|   enum_formats *result;
 | ||
| 
 | ||
|   result = g_new0 (enum_formats, 1);
 | ||
| 
 | ||
|   result->ief.lpVtbl = &ief_vtbl;
 | ||
|   result->ref_count = 1;
 | ||
|   result->ix = 0;
 | ||
|   result->dataobj = dataobj;
 | ||
|   idataobject_addref ((LPDATAOBJECT) dataobj);
 | ||
| 
 | ||
|   return result;
 | ||
| }
 | ||
| 
 | ||
| /* From MS Knowledge Base article Q130698 */
 | ||
| 
 | ||
| static gboolean
 | ||
| resolve_link (HWND     hWnd,
 | ||
| 	      wchar_t *link,
 | ||
| 	      gchar  **lpszPath)
 | ||
| {
 | ||
|   WIN32_FILE_ATTRIBUTE_DATA wfad;
 | ||
|   HRESULT hr;
 | ||
|   IShellLinkW *pslW = NULL;
 | ||
|   IPersistFile *ppf = NULL;
 | ||
| 
 | ||
|   /* Check if the file is empty first because IShellLink::Resolve for
 | ||
|    * some reason succeeds with an empty file and returns an empty
 | ||
|    * "link target". (#524151)
 | ||
|    */
 | ||
|     if (!GetFileAttributesExW (link, GetFileExInfoStandard, &wfad) ||
 | ||
| 	(wfad.nFileSizeHigh == 0 && wfad.nFileSizeLow == 0))
 | ||
|       return FALSE;
 | ||
| 
 | ||
|   /* Assume failure to start with: */
 | ||
|   *lpszPath = 0;
 | ||
| 
 | ||
|   /* Call CoCreateInstance to obtain the IShellLink interface
 | ||
|    * pointer. This call fails if CoInitialize is not called, so it is
 | ||
|    * assumed that CoInitialize has been called.
 | ||
|    */
 | ||
| 
 | ||
|   hr = CoCreateInstance (&CLSID_ShellLink,
 | ||
| 			 NULL,
 | ||
| 			 CLSCTX_INPROC_SERVER,
 | ||
| 			 &IID_IShellLinkW,
 | ||
| 			 (LPVOID *)&pslW);
 | ||
| 
 | ||
|   if (SUCCEEDED (hr))
 | ||
|    {
 | ||
|      /* The IShellLink interface supports the IPersistFile
 | ||
|       * interface. Get an interface pointer to it.
 | ||
|       */
 | ||
|      hr = pslW->lpVtbl->QueryInterface (pslW,
 | ||
| 					&IID_IPersistFile,
 | ||
| 					(LPVOID *) &ppf);
 | ||
|    }
 | ||
| 
 | ||
|   if (SUCCEEDED (hr))
 | ||
|     {
 | ||
|       /* Load the file. */
 | ||
|       hr = ppf->lpVtbl->Load (ppf, link, STGM_READ);
 | ||
|     }
 | ||
| 
 | ||
|   if (SUCCEEDED (hr))
 | ||
|     {
 | ||
|       /* Resolve the link by calling the Resolve()
 | ||
|        * interface function.
 | ||
|        */
 | ||
|       hr = pslW->lpVtbl->Resolve (pslW, hWnd, SLR_ANY_MATCH | SLR_NO_UI);
 | ||
|     }
 | ||
| 
 | ||
|   if (SUCCEEDED (hr))
 | ||
|     {
 | ||
|       wchar_t wtarget[MAX_PATH];
 | ||
| 
 | ||
|       hr = pslW->lpVtbl->GetPath (pslW, wtarget, MAX_PATH, NULL, 0);
 | ||
|       if (SUCCEEDED (hr))
 | ||
| 	*lpszPath = g_utf16_to_utf8 (wtarget, -1, NULL, NULL, NULL);
 | ||
|     }
 | ||
| 
 | ||
|   if (ppf)
 | ||
|     ppf->lpVtbl->Release (ppf);
 | ||
| 
 | ||
|   if (pslW)
 | ||
|     pslW->lpVtbl->Release (pslW);
 | ||
| 
 | ||
|   return SUCCEEDED (hr);
 | ||
| }
 | ||
| 
 | ||
| #if 0
 | ||
| 
 | ||
| /* Check for filenames like C:\Users\tml\AppData\Local\Temp\d5qtkvvs.bmp */
 | ||
| static gboolean
 | ||
| filename_looks_tempish (const char *filename)
 | ||
| {
 | ||
|   char *dirname;
 | ||
|   char *p;
 | ||
|   const char *q;
 | ||
|   gboolean retval = FALSE;
 | ||
| 
 | ||
|   dirname = g_path_get_dirname (filename);
 | ||
| 
 | ||
|   p = dirname;
 | ||
|   q = g_get_tmp_dir ();
 | ||
| 
 | ||
|   while (*p && *q &&
 | ||
| 	 ((G_IS_DIR_SEPARATOR (*p) && G_IS_DIR_SEPARATOR (*q)) ||
 | ||
| 	  g_ascii_tolower (*p) == g_ascii_tolower (*q)))
 | ||
|     p++, q++;
 | ||
| 
 | ||
|   if (!*p && !*q)
 | ||
|     retval = TRUE;
 | ||
| 
 | ||
|   g_free (dirname);
 | ||
| 
 | ||
|   return retval;
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| close_it (gpointer data)
 | ||
| {
 | ||
|   close (GPOINTER_TO_INT (data));
 | ||
| 
 | ||
|   return FALSE;
 | ||
| }
 | ||
| 
 | ||
| #endif
 | ||
| 
 | ||
| static GdkFilterReturn
 | ||
| gdk_dropfiles_filter (GdkXEvent *xev,
 | ||
| 		      GdkEvent  *event,
 | ||
| 		      gpointer   data)
 | ||
| {
 | ||
|   GdkDragContext *context;
 | ||
|   GdkWin32DragContext *context_win32;
 | ||
|   GString *result;
 | ||
|   MSG *msg = (MSG *) xev;
 | ||
|   HANDLE hdrop;
 | ||
|   POINT pt;
 | ||
|   gint nfiles, i;
 | ||
|   gchar *fileName, *linkedFile;
 | ||
| 
 | ||
|   if (msg->message == WM_DROPFILES)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("WM_DROPFILES: %p\n", msg->hwnd));
 | ||
| 
 | ||
|       context = gdk_drag_context_new (gdk_window_get_display (event->any.window));
 | ||
|       context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|       context->protocol = GDK_DRAG_PROTO_WIN32_DROPFILES;
 | ||
|       context->is_source = FALSE;
 | ||
| 
 | ||
|       context->source_window = gdk_get_default_root_window ();
 | ||
|       g_object_ref (context->source_window);
 | ||
| 
 | ||
|       g_set_object (&context->dest_window, event->any.window);
 | ||
| 
 | ||
|       /* WM_DROPFILES drops are always file names */
 | ||
|       context->targets =
 | ||
| 	g_list_append (NULL, _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_TEXT_URI_LIST));
 | ||
|       context->actions = GDK_ACTION_COPY;
 | ||
|       context->suggested_action = GDK_ACTION_COPY;
 | ||
|       current_dest_drag = context;
 | ||
| 
 | ||
|       event->dnd.type = GDK_DROP_START;
 | ||
|       event->dnd.context = current_dest_drag;
 | ||
|       gdk_event_set_device (event, gdk_drag_context_get_device (current_dest_drag));
 | ||
|       gdk_event_set_seat (event, gdk_device_get_seat (gdk_drag_context_get_device (current_dest_drag)));
 | ||
| 
 | ||
|       hdrop = (HANDLE) msg->wParam;
 | ||
|       DragQueryPoint (hdrop, &pt);
 | ||
|       ClientToScreen (msg->hwnd, &pt);
 | ||
| 
 | ||
|       event->dnd.x_root = (pt.x + _gdk_offset_x) / context_win32->scale;
 | ||
|       event->dnd.y_root = (pt.y + _gdk_offset_y) / context_win32->scale;
 | ||
|       event->dnd.time = _gdk_win32_get_next_tick (msg->time);
 | ||
| 
 | ||
|       nfiles = DragQueryFile (hdrop, 0xFFFFFFFF, NULL, 0);
 | ||
| 
 | ||
|       result = g_string_new (NULL);
 | ||
|       for (i = 0; i < nfiles; i++)
 | ||
| 	{
 | ||
| 	  gchar *uri;
 | ||
| 	  wchar_t wfn[MAX_PATH];
 | ||
| 
 | ||
| 	  DragQueryFileW (hdrop, i, wfn, MAX_PATH);
 | ||
| 	  fileName = g_utf16_to_utf8 (wfn, -1, NULL, NULL, NULL);
 | ||
| 
 | ||
| 	  /* Resolve shortcuts */
 | ||
| 	  if (resolve_link (msg->hwnd, wfn, &linkedFile))
 | ||
| 	    {
 | ||
| 	      uri = g_filename_to_uri (linkedFile, NULL, NULL);
 | ||
| 	      if (uri != NULL)
 | ||
| 		{
 | ||
| 		  g_string_append (result, uri);
 | ||
| 		  GDK_NOTE (DND, g_print ("... %s link to %s: %s\n",
 | ||
| 					  fileName, linkedFile, uri));
 | ||
| 		  g_free (uri);
 | ||
| 		}
 | ||
| 	      g_free (fileName);
 | ||
| 	      fileName = linkedFile;
 | ||
| 	    }
 | ||
| 	  else
 | ||
| 	    {
 | ||
| 	      uri = g_filename_to_uri (fileName, NULL, NULL);
 | ||
| 	      if (uri != NULL)
 | ||
| 		{
 | ||
| 		  g_string_append (result, uri);
 | ||
| 		  GDK_NOTE (DND, g_print ("... %s: %s\n", fileName, uri));
 | ||
| 		  g_free (uri);
 | ||
| 		}
 | ||
| 	    }
 | ||
| 
 | ||
| #if 0
 | ||
| 	  /* Awful hack to recognize temp files corresponding to
 | ||
| 	   * images dragged from Firefox... Open the file right here
 | ||
| 	   * so that it is less likely that Firefox manages to delete
 | ||
| 	   * it before the GTK+-using app (typically GIMP) has opened
 | ||
| 	   * it.
 | ||
| 	   *
 | ||
| 	   * Not compiled in for now, because it means images dragged
 | ||
| 	   * from Firefox would stay around in the temp folder which
 | ||
| 	   * is not what Firefox intended. I don't feel comfortable
 | ||
| 	   * with that, both from a geenral sanity point of view, and
 | ||
| 	   * from a privacy point of view. It's better to wait for
 | ||
| 	   * Firefox to fix the problem, for instance by deleting the
 | ||
| 	   * temp file after a longer delay, or to wait until we
 | ||
| 	   * implement the OLE2_DND...
 | ||
| 	   */
 | ||
| 	  if (filename_looks_tempish (fileName))
 | ||
| 	    {
 | ||
| 	      int fd = g_open (fileName, _O_RDONLY|_O_BINARY, 0);
 | ||
| 	      if (fd == -1)
 | ||
| 		{
 | ||
| 		  GDK_NOTE (DND, g_print ("Could not open %s, maybe an image dragged from Firefox that it already deleted\n", fileName));
 | ||
| 		}
 | ||
| 	      else
 | ||
| 		{
 | ||
| 		  GDK_NOTE (DND, g_print ("Opened %s as %d so that Firefox won't delete it\n", fileName, fd));
 | ||
| 		  g_timeout_add_seconds (1, close_it, GINT_TO_POINTER (fd));
 | ||
| 		}
 | ||
| 	    }
 | ||
| #endif
 | ||
| 
 | ||
| 	  g_free (fileName);
 | ||
| 	  g_string_append (result, "\015\012");
 | ||
| 	}
 | ||
|       _gdk_dropfiles_store (result->str);
 | ||
|       g_string_free (result, FALSE);
 | ||
| 
 | ||
|       DragFinish (hdrop);
 | ||
| 
 | ||
|       return GDK_FILTER_TRANSLATE;
 | ||
|     }
 | ||
|   else
 | ||
|     return GDK_FILTER_CONTINUE;
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
| void
 | ||
| _gdk_dnd_init (void)
 | ||
| {
 | ||
|   CoInitializeEx (NULL, COINIT_APARTMENTTHREADED);
 | ||
| 
 | ||
|   if (getenv ("GDK_WIN32_USE_EXPERIMENTAL_OLE2_DND"))
 | ||
|     use_ole2_dnd = TRUE;
 | ||
| 
 | ||
|   if (use_ole2_dnd)
 | ||
|     {
 | ||
|       HRESULT hr;
 | ||
| 
 | ||
|       hr = OleInitialize (NULL);
 | ||
| 
 | ||
|       if (! SUCCEEDED (hr))
 | ||
| 	g_error ("OleInitialize failed");
 | ||
| 
 | ||
|       target_ctx_for_window = g_hash_table_new (g_direct_hash, g_direct_equal);
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| void
 | ||
| _gdk_win32_dnd_exit (void)
 | ||
| {
 | ||
|   if (use_ole2_dnd)
 | ||
|     {
 | ||
|       OleUninitialize ();
 | ||
|     }
 | ||
| 
 | ||
|   CoUninitialize ();
 | ||
| }
 | ||
| 
 | ||
| /* Source side */
 | ||
| 
 | ||
| static void
 | ||
| local_send_leave (GdkDragContext *context,
 | ||
| 		  guint32         time)
 | ||
| {
 | ||
|   GdkEvent *tmp_event;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("local_send_leave: context=%p current_dest_drag=%p\n",
 | ||
| 			  context,
 | ||
| 			  current_dest_drag));
 | ||
| 
 | ||
|   if ((current_dest_drag != NULL) &&
 | ||
|       (current_dest_drag->protocol == GDK_DRAG_PROTO_LOCAL) &&
 | ||
|       (current_dest_drag->source_window == context->source_window))
 | ||
|     {
 | ||
|       tmp_event = gdk_event_new (GDK_DRAG_LEAVE);
 | ||
| 
 | ||
|       g_set_object (&tmp_event->dnd.window, context->dest_window);
 | ||
|       /* Pass ownership of context to the event */
 | ||
|       tmp_event->dnd.send_event = FALSE;
 | ||
|       g_set_object (&tmp_event->dnd.context, current_dest_drag);
 | ||
|       tmp_event->dnd.time = GDK_CURRENT_TIME; /* FIXME? */
 | ||
|       gdk_event_set_device (tmp_event, gdk_drag_context_get_device (context));
 | ||
|       gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (context)));
 | ||
| 
 | ||
|       current_dest_drag = NULL;
 | ||
| 
 | ||
|       GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
|       gdk_event_put (tmp_event);
 | ||
|       gdk_event_free (tmp_event);
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| local_send_enter (GdkDragContext *context,
 | ||
| 		  guint32         time)
 | ||
| {
 | ||
|   GdkEvent *tmp_event;
 | ||
|   GdkDragContext *new_context;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("local_send_enter: context=%p current_dest_drag=%p\n",
 | ||
| 			  context,
 | ||
| 			  current_dest_drag));
 | ||
| 
 | ||
|   if (current_dest_drag != NULL)
 | ||
|     {
 | ||
|       g_object_unref (G_OBJECT (current_dest_drag));
 | ||
|       current_dest_drag = NULL;
 | ||
|     }
 | ||
| 
 | ||
|   new_context = gdk_drag_context_new (gdk_window_get_display (context->source_window));
 | ||
|   new_context->protocol = GDK_DRAG_PROTO_LOCAL;
 | ||
|   new_context->is_source = FALSE;
 | ||
| 
 | ||
|   g_set_object (&new_context->source_window, context->source_window);
 | ||
|   g_set_object (&new_context->dest_window, context->dest_window);
 | ||
| 
 | ||
|   new_context->targets = g_list_copy (context->targets);
 | ||
| 
 | ||
|   gdk_window_set_events (new_context->source_window,
 | ||
| 			 gdk_window_get_events (new_context->source_window) |
 | ||
| 			 GDK_PROPERTY_CHANGE_MASK);
 | ||
|   new_context->actions = context->actions;
 | ||
| 
 | ||
|   tmp_event = gdk_event_new (GDK_DRAG_ENTER);
 | ||
|   g_set_object (&tmp_event->dnd.window, context->dest_window);
 | ||
|   tmp_event->dnd.send_event = FALSE;
 | ||
|   g_set_object (&tmp_event->dnd.context, new_context);
 | ||
|   tmp_event->dnd.time = GDK_CURRENT_TIME; /* FIXME? */
 | ||
|   gdk_event_set_device (tmp_event, gdk_drag_context_get_device (context));
 | ||
|   gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (context)));
 | ||
| 
 | ||
|   current_dest_drag = new_context;
 | ||
| 
 | ||
|   GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
|   gdk_event_put (tmp_event);
 | ||
|   gdk_event_free (tmp_event);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| local_send_motion (GdkDragContext *context,
 | ||
| 		   gint            x_root,
 | ||
| 		   gint            y_root,
 | ||
| 		   GdkDragAction   action,
 | ||
| 		   guint32         time)
 | ||
| {
 | ||
|   GdkEvent *tmp_event;
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("local_send_motion: context=%p (%d,%d) current_dest_drag=%p\n",
 | ||
| 			  context, x_root, y_root,
 | ||
| 			  current_dest_drag));
 | ||
| 
 | ||
|   if ((current_dest_drag != NULL) &&
 | ||
|       (current_dest_drag->protocol == GDK_DRAG_PROTO_LOCAL) &&
 | ||
|       (current_dest_drag->source_window == context->source_window))
 | ||
|     {
 | ||
|       GdkWin32DragContext *current_dest_drag_win32;
 | ||
| 
 | ||
|       tmp_event = gdk_event_new (GDK_DRAG_MOTION);
 | ||
|       g_set_object (&tmp_event->dnd.window, current_dest_drag->dest_window);
 | ||
|       tmp_event->dnd.send_event = FALSE;
 | ||
|       g_set_object (&tmp_event->dnd.context, current_dest_drag);
 | ||
|       tmp_event->dnd.time = time;
 | ||
|       gdk_event_set_device (tmp_event, gdk_drag_context_get_device (current_dest_drag));
 | ||
|       gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (current_dest_drag)));
 | ||
| 
 | ||
|       current_dest_drag->suggested_action = action;
 | ||
| 
 | ||
|       tmp_event->dnd.x_root = x_root;
 | ||
|       tmp_event->dnd.y_root = y_root;
 | ||
| 
 | ||
|       current_dest_drag_win32 = GDK_WIN32_DRAG_CONTEXT (current_dest_drag);
 | ||
|       current_dest_drag_win32->last_x = x_root;
 | ||
|       current_dest_drag_win32->last_y = y_root;
 | ||
| 
 | ||
|       context_win32->drag_status = GDK_DRAG_STATUS_MOTION_WAIT;
 | ||
| 
 | ||
|       GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
|       gdk_event_put (tmp_event);
 | ||
|       gdk_event_free (tmp_event);
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| local_send_drop (GdkDragContext *context,
 | ||
| 		 guint32         time)
 | ||
| {
 | ||
|   GdkEvent *tmp_event;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("local_send_drop: context=%p current_dest_drag=%p\n",
 | ||
| 			  context,
 | ||
| 			  current_dest_drag));
 | ||
| 
 | ||
|   if ((current_dest_drag != NULL) &&
 | ||
|       (current_dest_drag->protocol == GDK_DRAG_PROTO_LOCAL) &&
 | ||
|       (current_dest_drag->source_window == context->source_window))
 | ||
|     {
 | ||
|       GdkWin32DragContext *context_win32;
 | ||
| 
 | ||
|       /* Pass ownership of context to the event */
 | ||
|       tmp_event = gdk_event_new (GDK_DROP_START);
 | ||
|       g_set_object (&tmp_event->dnd.window, current_dest_drag->dest_window);
 | ||
|       tmp_event->dnd.send_event = FALSE;
 | ||
|       g_set_object (&tmp_event->dnd.context, current_dest_drag);
 | ||
|       tmp_event->dnd.time = GDK_CURRENT_TIME;
 | ||
|       gdk_event_set_device (tmp_event, gdk_drag_context_get_device (current_dest_drag));
 | ||
|       gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (current_dest_drag)));
 | ||
| 
 | ||
|       context_win32 = GDK_WIN32_DRAG_CONTEXT (current_dest_drag);
 | ||
|       tmp_event->dnd.x_root = context_win32->last_x;
 | ||
|       tmp_event->dnd.y_root = context_win32->last_y;
 | ||
| 
 | ||
|       current_dest_drag = NULL;
 | ||
| 
 | ||
|       GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
|       gdk_event_put (tmp_event);
 | ||
|       gdk_event_free (tmp_event);
 | ||
|     }
 | ||
| 
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_drag_do_leave (GdkDragContext *context,
 | ||
| 		   guint32         time)
 | ||
| {
 | ||
|   if (context->dest_window)
 | ||
|     {
 | ||
|       GDK_NOTE (DND, g_print ("gdk_drag_do_leave\n"));
 | ||
| 
 | ||
|       if (!use_ole2_dnd)
 | ||
| 	{
 | ||
| 	  if (context->protocol == GDK_DRAG_PROTO_LOCAL)
 | ||
| 	    local_send_leave (context, time);
 | ||
| 	}
 | ||
| 
 | ||
|       g_clear_object (&context->dest_window);
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static GdkWindow *
 | ||
| create_drag_window (GdkScreen *screen)
 | ||
| {
 | ||
|   GdkWindowAttr attrs = { 0 };
 | ||
|   guint mask;
 | ||
| 
 | ||
|   attrs.x = attrs.y = 0;
 | ||
|   attrs.width = attrs.height = 100;
 | ||
|   attrs.wclass = GDK_INPUT_OUTPUT;
 | ||
|   attrs.window_type = GDK_WINDOW_TEMP;
 | ||
|   attrs.type_hint = GDK_WINDOW_TYPE_HINT_DND;
 | ||
|   attrs.visual = gdk_screen_get_rgba_visual (screen);
 | ||
|   if (!attrs.visual)
 | ||
|     attrs.visual = gdk_screen_get_system_visual (screen);
 | ||
| 
 | ||
|   mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_TYPE_HINT;
 | ||
| 
 | ||
|   return gdk_window_new (gdk_screen_get_root_window (screen), &attrs, mask);
 | ||
| }
 | ||
| 
 | ||
| GdkDragContext *
 | ||
| _gdk_win32_window_drag_begin (GdkWindow *window,
 | ||
| 			      GdkDevice *device,
 | ||
| 			      GList     *targets,
 | ||
|                               gint       x_root,
 | ||
|                               gint       y_root)
 | ||
| {
 | ||
|   GdkDragContext *new_context;
 | ||
|   GdkWin32DragContext *context_win32;
 | ||
|   BYTE kbd_state[256];
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       g_return_val_if_fail (window != NULL, NULL);
 | ||
| 
 | ||
|       new_context = gdk_drag_context_new (gdk_window_get_display (window));
 | ||
| 
 | ||
|       new_context->is_source = TRUE;
 | ||
|       g_set_object (&new_context->source_window, window);
 | ||
| 
 | ||
|       new_context->targets = g_list_copy (targets);
 | ||
|       new_context->actions = 0;
 | ||
|       context_win32 = GDK_WIN32_DRAG_CONTEXT (new_context);
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       source_drag_context *ctx;
 | ||
| 
 | ||
|       g_return_val_if_fail (window != NULL, NULL);
 | ||
| 
 | ||
|       GDK_NOTE (DND, g_print ("gdk_drag_begin\n"));
 | ||
| 
 | ||
|       ctx = source_context_new (window, targets);
 | ||
| 
 | ||
|       sel_win32->dnd_source_state = GDK_WIN32_DND_PENDING;
 | ||
| 
 | ||
|       pending_src_context = ctx;
 | ||
|       new_context = g_object_ref (ctx->context);
 | ||
| 
 | ||
|       context_win32 = GDK_WIN32_DRAG_CONTEXT (new_context);
 | ||
|     }
 | ||
| 
 | ||
|   context_win32->start_x = x_root;
 | ||
|   context_win32->start_y = y_root;
 | ||
|   context_win32->last_x = context_win32->start_x;
 | ||
|   context_win32->last_y = context_win32->start_y;
 | ||
| 
 | ||
|   context_win32->last_key_state = 0;
 | ||
|   API_CALL (GetKeyboardState, (kbd_state));
 | ||
| 
 | ||
|   if (kbd_state[VK_MENU] & 0x80)
 | ||
|     context_win32->last_key_state |= MK_ALT;
 | ||
|   if (kbd_state[VK_CONTROL] & 0x80)
 | ||
|     context_win32->last_key_state |= MK_CONTROL;
 | ||
|   if (kbd_state[VK_SHIFT] & 0x80)
 | ||
|     context_win32->last_key_state |= MK_SHIFT;
 | ||
|   if (kbd_state[VK_LBUTTON] & 0x80)
 | ||
|     context_win32->last_key_state |= MK_LBUTTON;
 | ||
|   if (kbd_state[VK_MBUTTON] & 0x80)
 | ||
|     context_win32->last_key_state |= MK_MBUTTON;
 | ||
|   if (kbd_state[VK_RBUTTON] & 0x80)
 | ||
|     context_win32->last_key_state |= MK_RBUTTON;
 | ||
| 
 | ||
|   context_win32->drag_window = create_drag_window (gdk_window_get_screen (window));
 | ||
| 
 | ||
|   return new_context;
 | ||
| }
 | ||
| 
 | ||
| void
 | ||
| _gdk_win32_dnd_do_dragdrop (void)
 | ||
| {
 | ||
|   GdkDragContext* drag_ctx;
 | ||
|   data_object *dobj;
 | ||
|   HRESULT hr;
 | ||
|   DWORD dwEffect;
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     return;
 | ||
| 
 | ||
|   if (pending_src_context == NULL)
 | ||
|     return;
 | ||
| 
 | ||
|   drag_ctx = pending_src_context->context;
 | ||
| 
 | ||
|   dobj = data_object_new (drag_ctx);
 | ||
|   current_src_object = dobj;
 | ||
| 
 | ||
|   /* Start dragging with mainloop inside the OLE2 API. Exits only when done */
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("Calling DoDragDrop\n"));
 | ||
| 
 | ||
|   _gdk_win32_begin_modal_call (GDK_WIN32_MODAL_OP_DND);
 | ||
|   hr = DoDragDrop (&dobj->ido, &pending_src_context->ids,
 | ||
|     	       DROPEFFECT_COPY | DROPEFFECT_MOVE,
 | ||
|     	       &dwEffect);
 | ||
|   _gdk_win32_end_modal_call (GDK_WIN32_MODAL_OP_DND);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("DoDragDrop returned %s\n",
 | ||
|     		      (hr == DRAGDROP_S_DROP ? "DRAGDROP_S_DROP" :
 | ||
|     		       (hr == DRAGDROP_S_CANCEL ? "DRAGDROP_S_CANCEL" :
 | ||
|     			(hr == E_UNEXPECTED ? "E_UNEXPECTED" :
 | ||
|     			 g_strdup_printf ("%#.8lx", hr))))));
 | ||
| 
 | ||
|   /* Delete dnd selection after successful move */
 | ||
|   if (hr == DRAGDROP_S_DROP &&
 | ||
|       dwEffect == DROPEFFECT_MOVE &&
 | ||
|       (drag_ctx->actions & GDK_ACTION_MOVE))
 | ||
|     {
 | ||
|       GdkWin32Selection *win32_sel = _gdk_win32_selection_get ();
 | ||
|       GdkEvent tmp_event;
 | ||
| 
 | ||
|       memset (&tmp_event, 0, sizeof (tmp_event));
 | ||
|       tmp_event.type = GDK_SELECTION_REQUEST;
 | ||
|       g_set_object (&tmp_event.selection.window, drag_ctx->source_window);
 | ||
|       tmp_event.selection.send_event = FALSE;
 | ||
|       tmp_event.selection.selection = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_OLE2_DND);
 | ||
|       tmp_event.selection.target = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_DELETE);
 | ||
|       win32_sel->property_change_target_atom = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_DELETE);
 | ||
|       tmp_event.selection.property = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_OLE2_DND);
 | ||
|       g_set_object (&tmp_event.selection.requestor, drag_ctx->source_window);
 | ||
|       tmp_event.selection.time = GDK_CURRENT_TIME; /* ??? */
 | ||
| 
 | ||
|       GDK_NOTE (EVENTS, _gdk_win32_print_event (&tmp_event));
 | ||
|       gdk_event_put (&tmp_event);
 | ||
|     }
 | ||
| 
 | ||
|   {
 | ||
|     GdkEvent *tmp_event;
 | ||
|     tmp_event = gdk_event_new (GDK_DROP_FINISHED);
 | ||
|     g_set_object (&tmp_event->dnd.window, drag_ctx->source_window);
 | ||
|     tmp_event->dnd.send_event = FALSE;
 | ||
|     g_set_object (&tmp_event->dnd.context, drag_ctx);
 | ||
|     gdk_event_set_device (tmp_event, gdk_drag_context_get_device (drag_ctx));
 | ||
|     gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (drag_ctx)));
 | ||
| 
 | ||
|     GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
|     gdk_event_put (tmp_event);
 | ||
|     gdk_event_free (tmp_event);
 | ||
|   }
 | ||
| 
 | ||
|   current_src_object = NULL;
 | ||
|   dobj->ido.lpVtbl->Release (&dobj->ido);
 | ||
|   if (pending_src_context != NULL)
 | ||
|     {
 | ||
|       pending_src_context->ids.lpVtbl->Release (&pending_src_context->ids);
 | ||
|       pending_src_context = NULL;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| /* Untested, may not work ...
 | ||
|  * ... but as of this writing is only used by exlusive X11 gtksocket.c
 | ||
|  */
 | ||
| GdkDragProtocol
 | ||
| _gdk_win32_window_get_drag_protocol (GdkWindow *window,
 | ||
|                                      GdkWindow **target)
 | ||
| {
 | ||
|   GdkDragProtocol protocol = GDK_DRAG_PROTO_NONE;
 | ||
| 
 | ||
|   if (gdk_window_get_window_type (window) != GDK_WINDOW_FOREIGN)
 | ||
|     {
 | ||
|       if (g_object_get_data (G_OBJECT (window), "gdk-dnd-registered") != NULL)
 | ||
| 	{
 | ||
| 	  if (use_ole2_dnd)
 | ||
| 	    protocol = GDK_DRAG_PROTO_OLE2;
 | ||
| 	  else
 | ||
| 	    protocol = GDK_DRAG_PROTO_LOCAL;
 | ||
| 	}
 | ||
|     }
 | ||
| 
 | ||
|   if (target)
 | ||
|     {
 | ||
|       *target = NULL;
 | ||
|     }
 | ||
| 
 | ||
|   return protocol;
 | ||
| }
 | ||
| 
 | ||
| static GdkWindow *
 | ||
| gdk_win32_drag_context_find_window (GdkDragContext  *context,
 | ||
| 				    GdkWindow       *drag_window,
 | ||
| 				    GdkScreen       *screen,
 | ||
| 				    gint             x_root,
 | ||
| 				    gint             y_root,
 | ||
| 				    GdkDragProtocol *protocol)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   GdkWindow *toplevel = NULL;
 | ||
|   HWND hwnd = NULL;
 | ||
|   POINT pt;
 | ||
| 
 | ||
|   pt.x = x_root * context_win32->scale - _gdk_offset_x;
 | ||
|   pt.y = y_root * context_win32->scale - _gdk_offset_y;
 | ||
| 
 | ||
|   GDK_NOTE (DND,
 | ||
|             g_print ("gdk_drag_find_window_real: %+d%+d\n",
 | ||
|             (int) pt.x, (int) pt.y));
 | ||
| 
 | ||
|   hwnd = WindowFromPoint (pt);
 | ||
| 
 | ||
|   if (hwnd)
 | ||
|     {
 | ||
|       GdkWindow *window = gdk_win32_handle_table_lookup (hwnd);
 | ||
| 
 | ||
|       if (window)
 | ||
|         {
 | ||
|           toplevel = gdk_window_get_toplevel (window);
 | ||
|           g_object_ref (toplevel);
 | ||
|         }
 | ||
|       else
 | ||
|         toplevel = gdk_win32_window_foreign_new_for_display (gdk_screen_get_display (screen), hwnd);
 | ||
|     }
 | ||
| 
 | ||
|   if (use_ole2_dnd)
 | ||
|     *protocol = GDK_DRAG_PROTO_OLE2;
 | ||
|   else if (context->source_window)
 | ||
|     *protocol = GDK_DRAG_PROTO_LOCAL;
 | ||
|   else
 | ||
|     *protocol = GDK_DRAG_PROTO_WIN32_DROPFILES;
 | ||
| 
 | ||
|   GDK_NOTE (DND,
 | ||
|             g_print ("gdk_drag_find_window: %+d%+d: %p: %p %s\n",
 | ||
|                      x_root, y_root,
 | ||
|                      hwnd,
 | ||
|                      (toplevel ? GDK_WINDOW_HWND (toplevel) : NULL),
 | ||
|                      _gdk_win32_drag_protocol_to_string (*protocol)));
 | ||
| 
 | ||
|   return toplevel;
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_win32_drag_context_drag_motion (GdkDragContext *context,
 | ||
| 		 GdkWindow      *dest_window,
 | ||
| 		 GdkDragProtocol protocol,
 | ||
| 		 gint            x_root,
 | ||
| 		 gint            y_root,
 | ||
| 		 GdkDragAction   suggested_action,
 | ||
| 		 GdkDragAction   possible_actions,
 | ||
| 		 guint32         time)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32;
 | ||
| 
 | ||
|   g_return_val_if_fail (context != NULL, FALSE);
 | ||
| 
 | ||
|   context->actions = possible_actions;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_motion: @ %+d:%+d %s suggested=%s, possible=%s\n"
 | ||
| 			  " context=%p:{actions=%s,suggested=%s,action=%s}\n",
 | ||
| 			  x_root, y_root,
 | ||
| 			  _gdk_win32_drag_protocol_to_string (protocol),
 | ||
| 			  _gdk_win32_drag_action_to_string (suggested_action),
 | ||
| 			  _gdk_win32_drag_action_to_string (possible_actions),
 | ||
| 			  context,
 | ||
| 			  _gdk_win32_drag_action_to_string (context->actions),
 | ||
| 			  _gdk_win32_drag_action_to_string (context->suggested_action),
 | ||
| 			  _gdk_win32_drag_action_to_string (context->action)));
 | ||
| 
 | ||
|   context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   if (context_win32->drag_window)
 | ||
|     move_drag_window (context, x_root, y_root);
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       if (context->dest_window == dest_window)
 | ||
| 	{
 | ||
| 	  GdkDragContext *dest_context;
 | ||
| 
 | ||
| 	  dest_context = gdk_drag_context_find (FALSE,
 | ||
| 						context->source_window,
 | ||
| 						dest_window);
 | ||
| 
 | ||
| 	  if (dest_context)
 | ||
| 	    dest_context->actions = context->actions;
 | ||
| 
 | ||
| 	  context->suggested_action = suggested_action;
 | ||
| 	}
 | ||
|       else
 | ||
| 	{
 | ||
| 	  GdkEvent *tmp_event;
 | ||
| 
 | ||
| 	  /* Send a leave to the last destination */
 | ||
| 	  gdk_drag_do_leave (context, time);
 | ||
| 	  context_win32->drag_status = GDK_DRAG_STATUS_DRAG;
 | ||
| 
 | ||
| 	  /* Check if new destination accepts drags, and which protocol */
 | ||
| 	  if (dest_window)
 | ||
| 	    {
 | ||
| 	      g_set_object (&context->dest_window, dest_window);
 | ||
| 	      context->protocol = protocol;
 | ||
| 
 | ||
| 	      switch (protocol)
 | ||
| 		{
 | ||
| 		case GDK_DRAG_PROTO_LOCAL:
 | ||
| 		  local_send_enter (context, time);
 | ||
| 		  break;
 | ||
| 
 | ||
| 		default:
 | ||
| 		  break;
 | ||
| 		}
 | ||
| 	      context->suggested_action = suggested_action;
 | ||
| 	    }
 | ||
| 	  else
 | ||
| 	    {
 | ||
| 	      context->dest_window = NULL;
 | ||
| 	      context->action = 0;
 | ||
| 	    }
 | ||
| 
 | ||
| 	  /* Push a status event, to let the client know that
 | ||
| 	   * the drag changed
 | ||
| 	   */
 | ||
| 	  tmp_event = gdk_event_new (GDK_DRAG_STATUS);
 | ||
| 	  g_set_object (&tmp_event->dnd.window, context->source_window);
 | ||
| 	  /* We use this to signal a synthetic status. Perhaps
 | ||
| 	   * we should use an extra field...
 | ||
| 	   */
 | ||
| 	  tmp_event->dnd.send_event = TRUE;
 | ||
| 
 | ||
| 	  g_set_object (&tmp_event->dnd.context, context);
 | ||
| 	  tmp_event->dnd.time = time;
 | ||
| 	  gdk_event_set_device (tmp_event, gdk_drag_context_get_device (context));
 | ||
| 	  gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (context)));
 | ||
| 
 | ||
| 	  GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
| 	  gdk_event_put (tmp_event);
 | ||
| 	  gdk_event_free (tmp_event);
 | ||
| 	}
 | ||
| 
 | ||
|       /* Send a drag-motion event */
 | ||
| 
 | ||
|       context_win32->last_x = x_root;
 | ||
|       context_win32->last_y = y_root;
 | ||
| 
 | ||
|       if (context->dest_window)
 | ||
| 	{
 | ||
| 	  if (context_win32->drag_status == GDK_DRAG_STATUS_DRAG)
 | ||
| 	    {
 | ||
| 	      switch (context->protocol)
 | ||
| 		{
 | ||
| 		case GDK_DRAG_PROTO_LOCAL:
 | ||
| 		  local_send_motion (context, x_root, y_root, suggested_action, time);
 | ||
| 		  break;
 | ||
| 
 | ||
| 		case GDK_DRAG_PROTO_NONE:
 | ||
| 		  g_warning ("GDK_DRAG_PROTO_NONE is not valid in gdk_drag_motion()");
 | ||
| 		  break;
 | ||
| 
 | ||
| 		default:
 | ||
| 		  break;
 | ||
| 		}
 | ||
| 	    }
 | ||
| 	  else
 | ||
| 	    {
 | ||
| 	      GDK_NOTE (DND, g_print (" returning TRUE\n"
 | ||
| 				      " context=%p:{actions=%s,suggested=%s,action=%s}\n",
 | ||
| 				      context,
 | ||
| 				      _gdk_win32_drag_action_to_string (context->actions),
 | ||
| 				      _gdk_win32_drag_action_to_string (context->suggested_action),
 | ||
| 				      _gdk_win32_drag_action_to_string (context->action)));
 | ||
| 	      return TRUE;
 | ||
| 	    }
 | ||
| 	}
 | ||
|     }
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print (" returning FALSE\n"
 | ||
| 			  " context=%p:{actions=%s,suggested=%s,action=%s}\n",
 | ||
| 			  context,
 | ||
| 			  _gdk_win32_drag_action_to_string (context->actions),
 | ||
| 			  _gdk_win32_drag_action_to_string (context->suggested_action),
 | ||
| 			  _gdk_win32_drag_action_to_string (context->action)));
 | ||
|   return FALSE;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_drag_drop (GdkDragContext *context,
 | ||
| 	       guint32         time)
 | ||
| {
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
| 
 | ||
|   g_return_if_fail (context != NULL);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_drop\n"));
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       if (context->dest_window &&
 | ||
| 	  context->protocol == GDK_DRAG_PROTO_LOCAL)
 | ||
| 	local_send_drop (context, time);
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       sel_win32->dnd_source_state = GDK_WIN32_DND_DROPPED;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_drag_abort (GdkDragContext *context,
 | ||
| 		guint32         time)
 | ||
| {
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
| 
 | ||
|   g_return_if_fail (context != NULL);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_abort\n"));
 | ||
| 
 | ||
|   if (use_ole2_dnd)
 | ||
|     sel_win32->dnd_source_state = GDK_WIN32_DND_NONE;
 | ||
| }
 | ||
| 
 | ||
| /* Destination side */
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_drag_status (GdkDragContext *context,
 | ||
| 		 GdkDragAction   action,
 | ||
| 		 guint32         time)
 | ||
| {
 | ||
|   GdkDragContext *src_context;
 | ||
|   GdkEvent *tmp_event;
 | ||
| 
 | ||
|   g_return_if_fail (context != NULL);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_status: %s\n"
 | ||
| 			  " context=%p:{actions=%s,suggested=%s,action=%s}\n",
 | ||
| 			  _gdk_win32_drag_action_to_string (action),
 | ||
| 			  context,
 | ||
| 			  _gdk_win32_drag_action_to_string (context->actions),
 | ||
| 			  _gdk_win32_drag_action_to_string (context->suggested_action),
 | ||
| 			  _gdk_win32_drag_action_to_string (context->action)));
 | ||
| 
 | ||
|   context->action = action;
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       src_context = gdk_drag_context_find (TRUE,
 | ||
| 					   context->source_window,
 | ||
| 					   context->dest_window);
 | ||
| 
 | ||
|       if (src_context)
 | ||
| 	{
 | ||
| 	  GdkWin32DragContext *src_context_win32 = GDK_WIN32_DRAG_CONTEXT (src_context);
 | ||
| 
 | ||
| 	  if (src_context_win32->drag_status == GDK_DRAG_STATUS_MOTION_WAIT)
 | ||
| 	    src_context_win32->drag_status = GDK_DRAG_STATUS_DRAG;
 | ||
| 
 | ||
| 	  tmp_event = gdk_event_new (GDK_DRAG_STATUS);
 | ||
| 	  g_set_object (&tmp_event->dnd.window, context->source_window);
 | ||
| 	  tmp_event->dnd.send_event = FALSE;
 | ||
| 	  g_set_object (&tmp_event->dnd.context, src_context);
 | ||
| 	  tmp_event->dnd.time = GDK_CURRENT_TIME; /* FIXME? */
 | ||
| 	  gdk_event_set_device (tmp_event, gdk_drag_context_get_device (src_context));
 | ||
| 	  gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (src_context)));
 | ||
| 
 | ||
| 	  if (action == GDK_ACTION_DEFAULT)
 | ||
| 	    action = 0;
 | ||
| 
 | ||
| 	  src_context->action = action;
 | ||
| 
 | ||
| 	  GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
| 	  gdk_event_put (tmp_event);
 | ||
| 	  gdk_event_free (tmp_event);
 | ||
| 	}
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_drop_reply (GdkDragContext *context,
 | ||
| 		gboolean        ok,
 | ||
| 		guint32         time)
 | ||
| {
 | ||
|   g_return_if_fail (context != NULL);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drop_reply\n"));
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     if (context->dest_window)
 | ||
|       {
 | ||
| 	if (context->protocol == GDK_DRAG_PROTO_WIN32_DROPFILES)
 | ||
| 	  _gdk_dropfiles_store (NULL);
 | ||
|       }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_drop_finish (GdkDragContext *context,
 | ||
| 		 gboolean        success,
 | ||
| 		 guint32         time)
 | ||
| {
 | ||
|   GdkDragContext *src_context;
 | ||
|   GdkEvent *tmp_event;
 | ||
|   GdkWin32Selection *sel_win32 = _gdk_win32_selection_get ();
 | ||
| 
 | ||
|   g_return_if_fail (context != NULL);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drop_finish\n"));
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       src_context = gdk_drag_context_find (TRUE,
 | ||
| 					   context->source_window,
 | ||
| 					   context->dest_window);
 | ||
|       if (src_context)
 | ||
| 	{
 | ||
|           if (gdk_drag_context_get_selected_action (src_context) == GDK_ACTION_MOVE)
 | ||
|             {
 | ||
|               tmp_event = gdk_event_new (GDK_SELECTION_REQUEST);
 | ||
|               g_set_object (&tmp_event->selection.window, src_context->source_window);
 | ||
|               tmp_event->selection.send_event = FALSE;
 | ||
|               tmp_event->selection.selection = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_LOCAL_DND_SELECTION);
 | ||
|               tmp_event->selection.target = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_DELETE);
 | ||
|               sel_win32->property_change_target_atom = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_DELETE);
 | ||
|               tmp_event->selection.property = _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_LOCAL_DND_SELECTION);
 | ||
|               g_set_object (&tmp_event->selection.requestor, src_context->source_window);
 | ||
|               tmp_event->selection.time = GDK_CURRENT_TIME; /* ??? */
 | ||
| 
 | ||
|               GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
|               gdk_event_put (tmp_event);
 | ||
|               gdk_event_free (tmp_event);
 | ||
|             }
 | ||
| 
 | ||
| 	  tmp_event = gdk_event_new (GDK_DROP_FINISHED);
 | ||
| 	  g_set_object (&tmp_event->dnd.window, src_context->source_window);
 | ||
| 	  tmp_event->dnd.send_event = FALSE;
 | ||
| 	  g_set_object (&tmp_event->dnd.context, src_context);
 | ||
| 	  gdk_event_set_device (tmp_event, gdk_drag_context_get_device (src_context));
 | ||
| 	  gdk_event_set_seat (tmp_event, gdk_device_get_seat (gdk_drag_context_get_device (src_context)));
 | ||
| 
 | ||
| 	  GDK_NOTE (EVENTS, _gdk_win32_print_event (tmp_event));
 | ||
| 	  gdk_event_put (tmp_event);
 | ||
| 	  gdk_event_free (tmp_event);
 | ||
| 	}
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       gdk_drag_do_leave (context, time);
 | ||
| 
 | ||
|       if (success)
 | ||
| 	sel_win32->dnd_target_state = GDK_WIN32_DND_DROPPED;
 | ||
|       else
 | ||
| 	sel_win32->dnd_target_state = GDK_WIN32_DND_FAILED;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| #if 0
 | ||
| 
 | ||
| static GdkFilterReturn
 | ||
| gdk_destroy_filter (GdkXEvent *xev,
 | ||
| 		    GdkEvent  *event,
 | ||
| 		    gpointer   data)
 | ||
| {
 | ||
|   MSG *msg = (MSG *) xev;
 | ||
| 
 | ||
|   if (msg->message == WM_DESTROY)
 | ||
|     {
 | ||
|       IDropTarget *idtp = (IDropTarget *) data;
 | ||
| 
 | ||
|       GDK_NOTE (DND, g_print ("gdk_destroy_filter: WM_DESTROY: %p\n", msg->hwnd));
 | ||
| #if 0
 | ||
|       idtp->lpVtbl->Release (idtp);
 | ||
| #endif
 | ||
|       RevokeDragDrop (msg->hwnd);
 | ||
|       CoLockObjectExternal ((IUnknown*) idtp, FALSE, TRUE);
 | ||
|     }
 | ||
|   return GDK_FILTER_CONTINUE;
 | ||
| }
 | ||
| 
 | ||
| #endif
 | ||
| 
 | ||
| void
 | ||
| _gdk_win32_window_register_dnd (GdkWindow *window)
 | ||
| {
 | ||
|   target_drag_context *ctx;
 | ||
|   HRESULT hr;
 | ||
| 
 | ||
|   g_return_if_fail (window != NULL);
 | ||
| 
 | ||
|   if (gdk_window_get_window_type (window) == GDK_WINDOW_OFFSCREEN)
 | ||
|     return;
 | ||
| 
 | ||
|   if (g_object_get_data (G_OBJECT (window), "gdk-dnd-registered") != NULL)
 | ||
|     return;
 | ||
|   else
 | ||
|     g_object_set_data (G_OBJECT (window), "gdk-dnd-registered", GINT_TO_POINTER (TRUE));
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_window_register_dnd: %p\n", GDK_WINDOW_HWND (window)));
 | ||
| 
 | ||
|   if (!use_ole2_dnd)
 | ||
|     {
 | ||
|       /* We always claim to accept dropped files, but in fact we might not,
 | ||
|        * of course. This function is called in such a way that it cannot know
 | ||
|        * whether the window (widget) in question actually accepts files
 | ||
|        * (in gtk, data of type text/uri-list) or not.
 | ||
|        */
 | ||
|       gdk_window_add_filter (window, gdk_dropfiles_filter, NULL);
 | ||
|       DragAcceptFiles (GDK_WINDOW_HWND (window), TRUE);
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       /* Return if window is already setup for DND. */
 | ||
|       if (g_hash_table_lookup (target_ctx_for_window, GDK_WINDOW_HWND (window)) != NULL)
 | ||
| 	return;
 | ||
| 
 | ||
|       ctx = target_context_new (window);
 | ||
| 
 | ||
|       hr = CoLockObjectExternal ((IUnknown *) &ctx->idt, TRUE, FALSE);
 | ||
|       if (!SUCCEEDED (hr))
 | ||
| 	OTHER_API_FAILED ("CoLockObjectExternal");
 | ||
|       else
 | ||
| 	{
 | ||
| 	  hr = RegisterDragDrop (GDK_WINDOW_HWND (window), &ctx->idt);
 | ||
| 	  if (hr == DRAGDROP_E_ALREADYREGISTERED)
 | ||
| 	    {
 | ||
| 	      g_print ("DRAGDROP_E_ALREADYREGISTERED\n");
 | ||
| 	      CoLockObjectExternal ((IUnknown *) &ctx->idt, FALSE, FALSE);
 | ||
| 	    }
 | ||
| 	  else if (!SUCCEEDED (hr))
 | ||
| 	    OTHER_API_FAILED ("RegisterDragDrop");
 | ||
| 	  else
 | ||
| 	    {
 | ||
| 	      g_object_ref (window);
 | ||
| 	      g_hash_table_insert (target_ctx_for_window, GDK_WINDOW_HWND (window), ctx);
 | ||
| 	    }
 | ||
| 	}
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_win32_drag_context_drop_status (GdkDragContext *context)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   return ! context_win32->drop_failed;
 | ||
| }
 | ||
| 
 | ||
| static GdkAtom
 | ||
| gdk_win32_drag_context_get_selection (GdkDragContext *context)
 | ||
| {
 | ||
|   switch (context->protocol)
 | ||
|     {
 | ||
|     case GDK_DRAG_PROTO_LOCAL:
 | ||
|       return _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_LOCAL_DND_SELECTION);
 | ||
|     case GDK_DRAG_PROTO_WIN32_DROPFILES:
 | ||
|       return _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_DROPFILES_DND);
 | ||
|     case GDK_DRAG_PROTO_OLE2:
 | ||
|       return _gdk_win32_selection_atom (GDK_WIN32_ATOM_INDEX_OLE2_DND);
 | ||
|     default:
 | ||
|       return GDK_NONE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_set_cursor (GdkDragContext *context,
 | ||
|                                    GdkCursor      *cursor)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_context_set_cursor: 0x%p 0x%p\n", context, cursor));
 | ||
| 
 | ||
|   if (!g_set_object (&context_win32->cursor, cursor))
 | ||
|     return;
 | ||
| 
 | ||
|   if (context_win32->grab_seat)
 | ||
|     {
 | ||
|       G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
 | ||
|       gdk_device_grab (gdk_seat_get_pointer (context_win32->grab_seat),
 | ||
|                        context_win32->ipc_window,
 | ||
|                        GDK_OWNERSHIP_APPLICATION, FALSE,
 | ||
|                        GDK_POINTER_MOTION_MASK | GDK_BUTTON_RELEASE_MASK,
 | ||
|                        cursor, GDK_CURRENT_TIME);
 | ||
|       G_GNUC_END_IGNORE_DEPRECATIONS;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static double
 | ||
| ease_out_cubic (double t)
 | ||
| {
 | ||
|   double p = t - 1;
 | ||
|   return p * p * p + 1;
 | ||
| }
 | ||
| 
 | ||
| #define ANIM_TIME 500000 /* half a second */
 | ||
| 
 | ||
| typedef struct _GdkDragAnim GdkDragAnim;
 | ||
| struct _GdkDragAnim {
 | ||
|   GdkWin32DragContext *context;
 | ||
|   GdkFrameClock *frame_clock;
 | ||
|   gint64 start_time;
 | ||
| };
 | ||
| 
 | ||
| static void
 | ||
| gdk_drag_anim_destroy (GdkDragAnim *anim)
 | ||
| {
 | ||
|   g_object_unref (anim->context);
 | ||
|   g_slice_free (GdkDragAnim, anim);
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_drag_anim_timeout (gpointer data)
 | ||
| {
 | ||
|   GdkDragAnim *anim = data;
 | ||
|   GdkWin32DragContext *context = anim->context;
 | ||
|   GdkFrameClock *frame_clock = anim->frame_clock;
 | ||
|   gint64 current_time;
 | ||
|   double f;
 | ||
|   double t;
 | ||
| 
 | ||
|   if (!frame_clock)
 | ||
|     return G_SOURCE_REMOVE;
 | ||
| 
 | ||
|   current_time = gdk_frame_clock_get_frame_time (frame_clock);
 | ||
| 
 | ||
|   f = (current_time - anim->start_time) / (double) ANIM_TIME;
 | ||
| 
 | ||
|   if (f >= 1.0)
 | ||
|     return G_SOURCE_REMOVE;
 | ||
| 
 | ||
|   t = ease_out_cubic (f);
 | ||
| 
 | ||
|   gdk_window_show (context->drag_window);
 | ||
|   gdk_window_move (context->drag_window,
 | ||
|                    context->last_x + (context->start_x - context->last_x) * t - context->hot_x,
 | ||
|                    context->last_y + (context->start_y - context->last_y) * t - context->hot_y);
 | ||
|   gdk_window_set_opacity (context->drag_window, 1.0 - f);
 | ||
| 
 | ||
|   return G_SOURCE_CONTINUE;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_drop_done (GdkDragContext *context,
 | ||
|                                   gboolean        success)
 | ||
| {
 | ||
|   GdkWin32DragContext *win32_context = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   GdkDragAnim *anim;
 | ||
|   cairo_surface_t *win_surface;
 | ||
|   cairo_surface_t *surface;
 | ||
|   cairo_pattern_t *pattern;
 | ||
|   cairo_t *cr;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_context_drop_done: 0x%p %s\n",
 | ||
|                           context,
 | ||
|                           success ? "dropped successfully" : "dropped unsuccessfully"));
 | ||
| 
 | ||
|   if (success)
 | ||
|     {
 | ||
|       gdk_window_hide (win32_context->drag_window);
 | ||
|       return;
 | ||
|     }
 | ||
| 
 | ||
|   win_surface = _gdk_window_ref_cairo_surface (win32_context->drag_window);
 | ||
|   surface = gdk_window_create_similar_surface (win32_context->drag_window,
 | ||
|                                                cairo_surface_get_content (win_surface),
 | ||
|                                                gdk_window_get_width (win32_context->drag_window),
 | ||
|                                                gdk_window_get_height (win32_context->drag_window));
 | ||
|   cr = cairo_create (surface);
 | ||
|   cairo_set_source_surface (cr, win_surface, 0, 0);
 | ||
|   cairo_paint (cr);
 | ||
|   cairo_destroy (cr);
 | ||
|   cairo_surface_destroy (win_surface);
 | ||
| 
 | ||
|   pattern = cairo_pattern_create_for_surface (surface);
 | ||
| 
 | ||
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS
 | ||
|   gdk_window_set_background_pattern (win32_context->drag_window, pattern);
 | ||
| G_GNUC_END_IGNORE_DEPRECATIONS
 | ||
| 
 | ||
|   cairo_pattern_destroy (pattern);
 | ||
|   cairo_surface_destroy (surface);
 | ||
| 
 | ||
|   anim = g_slice_new0 (GdkDragAnim);
 | ||
|   g_set_object (&anim->context, win32_context);
 | ||
|   anim->frame_clock = gdk_window_get_frame_clock (win32_context->drag_window);
 | ||
|   anim->start_time = gdk_frame_clock_get_frame_time (anim->frame_clock);
 | ||
| 
 | ||
|   gdk_threads_add_timeout_full (G_PRIORITY_DEFAULT, 17,
 | ||
|                                 gdk_drag_anim_timeout, anim,
 | ||
|                                 (GDestroyNotify) gdk_drag_anim_destroy);
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| drag_context_grab (GdkDragContext *context)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   GdkSeatCapabilities capabilities;
 | ||
|   GdkSeat *seat;
 | ||
|   GdkCursor *cursor;
 | ||
| 
 | ||
|   if (!context_win32->ipc_window)
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   seat = gdk_device_get_seat (gdk_drag_context_get_device (context));
 | ||
| 
 | ||
|   capabilities = GDK_SEAT_CAPABILITY_ALL;
 | ||
| 
 | ||
|   cursor = gdk_drag_get_cursor (context, gdk_drag_context_get_selected_action (context));
 | ||
|   g_set_object (&context_win32->cursor, cursor);
 | ||
| 
 | ||
|   if (gdk_seat_grab (seat, context_win32->ipc_window,
 | ||
|                      capabilities, FALSE,
 | ||
|                      context_win32->cursor, NULL, NULL, NULL) != GDK_GRAB_SUCCESS)
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   g_set_object (&context_win32->grab_seat, seat);
 | ||
| 
 | ||
|   /* TODO: Should be grabbing keys here, to support keynav. SetWindowsHookEx()? */
 | ||
| 
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| drag_context_ungrab (GdkDragContext *context)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   if (!context_win32->grab_seat)
 | ||
|     return;
 | ||
| 
 | ||
|   gdk_seat_ungrab (context_win32->grab_seat);
 | ||
| 
 | ||
|   g_clear_object (&context_win32->grab_seat);
 | ||
| 
 | ||
|   /* TODO: Should be ungrabbing keys here */
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_win32_drag_context_manage_dnd (GdkDragContext *context,
 | ||
|                                    GdkWindow      *ipc_window,
 | ||
|                                    GdkDragAction   actions)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   if (context_win32->ipc_window)
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   if (use_ole2_dnd)
 | ||
|     context->protocol = GDK_DRAG_PROTO_OLE2;
 | ||
|   else
 | ||
|     context->protocol = GDK_DRAG_PROTO_LOCAL;
 | ||
| 
 | ||
|   g_set_object (&context_win32->ipc_window, ipc_window);
 | ||
| 
 | ||
|   if (drag_context_grab (context))
 | ||
|     {
 | ||
|       context_win32->actions = actions;
 | ||
|       move_drag_window (context, context_win32->start_x, context_win32->start_y);
 | ||
|       return TRUE;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       g_clear_object (&context_win32->ipc_window);
 | ||
|       return FALSE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_cancel (GdkDragContext      *context,
 | ||
|                                GdkDragCancelReason  reason)
 | ||
| {
 | ||
|   const gchar *reason_str = NULL;
 | ||
|   switch (reason)
 | ||
|     {
 | ||
|     case GDK_DRAG_CANCEL_NO_TARGET:
 | ||
|       reason_str = "no target";
 | ||
|       break;
 | ||
|     case GDK_DRAG_CANCEL_USER_CANCELLED:
 | ||
|       reason_str = "user cancelled";
 | ||
|       break;
 | ||
|     case GDK_DRAG_CANCEL_ERROR:
 | ||
|       reason_str = "error";
 | ||
|       break;
 | ||
|     default:
 | ||
|       reason_str = "<unknown>";
 | ||
|       break;
 | ||
|     }
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_context_cancel: 0x%p %s\n",
 | ||
|                           context,
 | ||
|                           reason_str));
 | ||
|   drag_context_ungrab (context);
 | ||
|   gdk_drag_drop_done (context, FALSE);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_drop_performed (GdkDragContext *context,
 | ||
|                                        guint32         time_)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_context_drop_performed: 0x%p %u\n",
 | ||
|                           context,
 | ||
|                           time_));
 | ||
|   gdk_drag_drop (context, time_);
 | ||
|   drag_context_ungrab (context);
 | ||
| }
 | ||
| 
 | ||
| #define BIG_STEP 20
 | ||
| #define SMALL_STEP 1
 | ||
| 
 | ||
| static void
 | ||
| gdk_drag_get_current_actions (GdkModifierType  state,
 | ||
|                               gint             button,
 | ||
|                               GdkDragAction    actions,
 | ||
|                               GdkDragAction   *suggested_action,
 | ||
|                               GdkDragAction   *possible_actions)
 | ||
| {
 | ||
|   *suggested_action = 0;
 | ||
|   *possible_actions = 0;
 | ||
| 
 | ||
|   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;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_drag_update (GdkDragContext  *context,
 | ||
|                  gdouble          x_root,
 | ||
|                  gdouble          y_root,
 | ||
|                  GdkModifierType  mods,
 | ||
|                  guint32          evtime)
 | ||
| {
 | ||
|   GdkWin32DragContext *win32_context = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   GdkDragAction action, possible_actions;
 | ||
|   GdkWindow *dest_window;
 | ||
|   GdkDragProtocol protocol;
 | ||
| 
 | ||
|   gdk_drag_get_current_actions (mods, GDK_BUTTON_PRIMARY, win32_context->actions,
 | ||
|                                 &action, &possible_actions);
 | ||
| 
 | ||
|   gdk_drag_find_window_for_screen (context,
 | ||
|                                    win32_context->drag_window,
 | ||
|                                    gdk_display_get_default_screen (gdk_display_get_default ()),
 | ||
|                                    x_root, y_root, &dest_window, &protocol);
 | ||
| 
 | ||
|   gdk_drag_motion (context, dest_window, protocol, x_root, y_root,
 | ||
|                    action, possible_actions, evtime);
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_dnd_handle_motion_event (GdkDragContext       *context,
 | ||
|                              const GdkEventMotion *event)
 | ||
| {
 | ||
|   GdkModifierType state;
 | ||
| 
 | ||
|   if (!gdk_event_get_state ((GdkEvent *) event, &state))
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gd_dnd_handle_motion_event: 0x%p\n",
 | ||
|                           context));
 | ||
| 
 | ||
|   gdk_drag_update (context, event->x_root, event->y_root, state,
 | ||
|                    gdk_event_get_time ((GdkEvent *) event));
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_dnd_handle_key_event (GdkDragContext    *context,
 | ||
|                           const GdkEventKey *event)
 | ||
| {
 | ||
|   GdkWin32DragContext *win32_context = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   GdkModifierType state;
 | ||
|   GdkWindow *root_window;
 | ||
|   GdkDevice *pointer;
 | ||
|   gint dx, dy;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_dnd_handle_key_event: 0x%p\n",
 | ||
|                           context));
 | ||
| 
 | ||
|   dx = dy = 0;
 | ||
|   state = event->state;
 | ||
|   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:
 | ||
|           gdk_drag_context_cancel (context, GDK_DRAG_CANCEL_USER_CANCELLED);
 | ||
|           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 (context) != 0) &&
 | ||
|               (gdk_drag_context_get_dest_window (context) != NULL))
 | ||
|             {
 | ||
|               g_signal_emit_by_name (context, "drop-performed",
 | ||
|                                      gdk_event_get_time ((GdkEvent *) event));
 | ||
|             }
 | ||
|           else
 | ||
|             gdk_drag_context_cancel (context, GDK_DRAG_CANCEL_NO_TARGET);
 | ||
| 
 | ||
|           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;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|   /* The state is not yet updated in the event, so we need
 | ||
|    * to query it here.
 | ||
|    */
 | ||
|   root_window = gdk_screen_get_root_window (gdk_window_get_screen (win32_context->ipc_window));
 | ||
|   gdk_window_get_device_position (root_window, pointer, NULL, NULL, &state);
 | ||
| 
 | ||
|   if (dx != 0 || dy != 0)
 | ||
|     {
 | ||
|       win32_context->last_x += dx;
 | ||
|       win32_context->last_y += dy;
 | ||
|       gdk_device_warp (pointer,
 | ||
|                        gdk_window_get_screen (win32_context->ipc_window),
 | ||
|                        win32_context->last_x, win32_context->last_y);
 | ||
|     }
 | ||
| 
 | ||
|   gdk_drag_update (context, win32_context->last_x, win32_context->last_y, state,
 | ||
|                    gdk_event_get_time ((GdkEvent *) event));
 | ||
| 
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_dnd_handle_grab_broken_event (GdkDragContext           *context,
 | ||
|                                   const GdkEventGrabBroken *event)
 | ||
| {
 | ||
|   GdkWin32DragContext *win32_context = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_dnd_handle_grab_broken_event: 0x%p\n",
 | ||
|                           context));
 | ||
| 
 | ||
|   /* 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 == win32_context->drag_window ||
 | ||
|       event->grab_window == win32_context->ipc_window)
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   if (gdk_event_get_device ((GdkEvent *) event) !=
 | ||
|       gdk_drag_context_get_device (context))
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   gdk_drag_context_cancel (context, GDK_DRAG_CANCEL_ERROR);
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_dnd_handle_button_event (GdkDragContext       *context,
 | ||
|                              const GdkEventButton *event)
 | ||
| {
 | ||
|   GDK_NOTE (DND, g_print ("gdk_dnd_handle_button_event: 0x%p\n",
 | ||
|                           context));
 | ||
| 
 | ||
| #if 0
 | ||
|   /* FIXME: Check the button matches */
 | ||
|   if (event->button != win32_context->button)
 | ||
|     return FALSE;
 | ||
| #endif
 | ||
| 
 | ||
|   if ((gdk_drag_context_get_selected_action (context) != 0) &&
 | ||
|       (gdk_drag_context_get_dest_window (context) != NULL))
 | ||
|     {
 | ||
|       g_signal_emit_by_name (context, "drop-performed",
 | ||
|                              gdk_event_get_time ((GdkEvent *) event));
 | ||
|     }
 | ||
|   else
 | ||
|     gdk_drag_context_cancel (context, GDK_DRAG_CANCEL_NO_TARGET);
 | ||
| 
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| gboolean
 | ||
| gdk_dnd_handle_drag_status (GdkDragContext    *context,
 | ||
|                             const GdkEventDND *event)
 | ||
| {
 | ||
|   GdkWin32DragContext *context_win32 = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
|   GdkDragAction action;
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_dnd_handle_drag_status: 0x%p\n",
 | ||
|                           context));
 | ||
| 
 | ||
|   if (context != event->context)
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   action = gdk_drag_context_get_selected_action (context);
 | ||
| 
 | ||
|   if (action != context_win32->current_action)
 | ||
|     {
 | ||
|       context_win32->current_action = action;
 | ||
|       g_signal_emit_by_name (context, "action-changed", action);
 | ||
|     }
 | ||
| 
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| gdk_dnd_handle_drop_finished (GdkDragContext *context,
 | ||
|                               const GdkEventDND *event)
 | ||
| {
 | ||
|   GdkWin32DragContext *win32_context = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_dnd_handle_drop_finihsed: 0x%p\n",
 | ||
|                           context));
 | ||
| 
 | ||
|   if (context != event->context)
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   g_signal_emit_by_name (context, "dnd-finished");
 | ||
|   gdk_drag_drop_done (context, !win32_context->drop_failed);
 | ||
|   gdk_win32_selection_clear_targets (gdk_display_get_default (),
 | ||
|                                      gdk_win32_drag_context_get_selection (context));
 | ||
| 
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| gboolean
 | ||
| gdk_win32_drag_context_handle_event (GdkDragContext *context,
 | ||
|                                      const GdkEvent *event)
 | ||
| {
 | ||
|   GdkWin32DragContext *win32_context = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   if (!context->is_source)
 | ||
|     return FALSE;
 | ||
|   if (!win32_context->grab_seat && event->type != GDK_DROP_FINISHED)
 | ||
|     return FALSE;
 | ||
| 
 | ||
|   switch (event->type)
 | ||
|     {
 | ||
|     case GDK_MOTION_NOTIFY:
 | ||
|       return gdk_dnd_handle_motion_event (context, &event->motion);
 | ||
|     case GDK_BUTTON_RELEASE:
 | ||
|       return gdk_dnd_handle_button_event (context, &event->button);
 | ||
|     case GDK_KEY_PRESS:
 | ||
|     case GDK_KEY_RELEASE:
 | ||
|       return gdk_dnd_handle_key_event (context, &event->key);
 | ||
|     case GDK_GRAB_BROKEN:
 | ||
|       return gdk_dnd_handle_grab_broken_event (context, &event->grab_broken);
 | ||
|     case GDK_DRAG_STATUS:
 | ||
|       return gdk_dnd_handle_drag_status (context, &event->dnd);
 | ||
|     case GDK_DROP_FINISHED:
 | ||
|       return gdk_dnd_handle_drop_finished (context, &event->dnd);
 | ||
|     default:
 | ||
|       break;
 | ||
|     }
 | ||
| 
 | ||
|   return FALSE;
 | ||
| }
 | ||
| 
 | ||
| void
 | ||
| gdk_win32_drag_context_action_changed (GdkDragContext *context,
 | ||
|                                        GdkDragAction   action)
 | ||
| {
 | ||
|   GdkCursor *cursor;
 | ||
| 
 | ||
|   cursor = gdk_drag_get_cursor (context, action);
 | ||
|   gdk_drag_context_set_cursor (context, cursor);
 | ||
| }
 | ||
| 
 | ||
| static GdkWindow *
 | ||
| gdk_win32_drag_context_get_drag_window (GdkDragContext *context)
 | ||
| {
 | ||
|   return GDK_WIN32_DRAG_CONTEXT (context)->drag_window;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_set_hotspot (GdkDragContext *context,
 | ||
|                                     gint            hot_x,
 | ||
|                                     gint            hot_y)
 | ||
| {
 | ||
|   GdkWin32DragContext *win32_context = GDK_WIN32_DRAG_CONTEXT (context);
 | ||
| 
 | ||
|   GDK_NOTE (DND, g_print ("gdk_drag_context_set_hotspot: 0x%p %d:%d\n",
 | ||
|                           context,
 | ||
|                           hot_x, hot_y));
 | ||
| 
 | ||
|   win32_context->hot_x = hot_x;
 | ||
|   win32_context->hot_y = hot_y;
 | ||
| 
 | ||
|   if (win32_context->grab_seat)
 | ||
|     {
 | ||
|       /* DnD is managed, update current position */
 | ||
|       move_drag_window (context, win32_context->last_x, win32_context->last_y);
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gdk_win32_drag_context_class_init (GdkWin32DragContextClass *klass)
 | ||
| {
 | ||
|   GObjectClass *object_class = G_OBJECT_CLASS (klass);
 | ||
|   GdkDragContextClass *context_class = GDK_DRAG_CONTEXT_CLASS (klass);
 | ||
| 
 | ||
|   object_class->finalize = gdk_win32_drag_context_finalize;
 | ||
| 
 | ||
|   context_class->find_window = gdk_win32_drag_context_find_window;
 | ||
|   context_class->drag_status = gdk_win32_drag_context_drag_status;
 | ||
|   context_class->drag_motion = gdk_win32_drag_context_drag_motion;
 | ||
|   context_class->drag_abort = gdk_win32_drag_context_drag_abort;
 | ||
|   context_class->drag_drop = gdk_win32_drag_context_drag_drop;
 | ||
|   context_class->drop_reply = gdk_win32_drag_context_drop_reply;
 | ||
|   context_class->drop_finish = gdk_win32_drag_context_drop_finish;
 | ||
|   context_class->drop_status = gdk_win32_drag_context_drop_status;
 | ||
|   context_class->get_selection = gdk_win32_drag_context_get_selection;
 | ||
| 
 | ||
|   context_class->get_drag_window = gdk_win32_drag_context_get_drag_window;
 | ||
|   context_class->set_hotspot = gdk_win32_drag_context_set_hotspot;
 | ||
|   context_class->drop_done = gdk_win32_drag_context_drop_done;
 | ||
|   context_class->manage_dnd = gdk_win32_drag_context_manage_dnd;
 | ||
|   context_class->set_cursor = gdk_win32_drag_context_set_cursor;
 | ||
|   context_class->cancel = gdk_win32_drag_context_cancel;
 | ||
|   context_class->drop_performed = gdk_win32_drag_context_drop_performed;
 | ||
|   context_class->handle_event = gdk_win32_drag_context_handle_event;
 | ||
|   context_class->action_changed = gdk_win32_drag_context_action_changed;
 | ||
| 
 | ||
| }
 | 
