After an evaluation, GNOME has moved from Bugzilla to GitLab. Learn more about GitLab.
No new issues can be reported in GNOME Bugzilla anymore.
To report an issue in a GNOME project, go to GNOME GitLab.
Do not go to GNOME Gitlab for: Bluefish, Doxygen, GnuCash, GStreamer, java-gnome, LDTP, NetworkManager, Tomboy.
Bug 137192 - DND coordinates confusion : screen or client
DND coordinates confusion : screen or client
Status: RESOLVED FIXED
Product: gtk+
Classification: Platform
Component: Backend: Win32
2.4.x
Other other
: Normal normal
: ---
Assigned To: gtk-win32 maintainers
gtk-bugs
Depends on:
Blocks:
 
 
Reported: 2004-03-14 21:30 UTC by Hans Breuer
Modified: 2004-12-22 21:47 UTC
See Also:
GNOME target: ---
GNOME version: ---



Description Hans Breuer 2004-03-14 21:30:02 UTC
DND coordinates confusion : screen or client

To get all the nice inter-app drag & drop of The GIMP working
under win32 I've played a little wit ScreenToClient() 
transformations on the delivered - so called 'root_x' and 
'root_y' coordinates for GDK_DRAG_MOTION and GDK_DROP_START.

Other then the variable names suggest the point seems to be
needed in event.dnd.window coordinates space and not in root
window coordinates. Everything seems to work fine with this
change, but it may hide another bug in the win32 backend.

Or maybe the variable names are just suggesting the wrong
thingh- At least to me it appears so as if there is no
transformation done in the x11 backend for these case 
either. Am I looking at the wrong place ?

Oh: and the patch reverts (or I think highly simplifies the
fix for bug #116320; haven't tested it, but that part appears
to be independent  ;)


Thanks,
    Hans


diff --exclude-from=c:\util\tool\diff.ign -up -r
from-cvs/gtk+/gdk/win32/gdkdnd-win32.c my-gtk/gtk+/gdk/win32/gdkdnd-win32.c
--- from-cvs/gtk+/gdk/win32/gdkdnd-win32.c	Sat Mar  6 12:42:54 2004
+++ my-gtk/gtk+/gdk/win32/gdkdnd-win32.c	Sun Mar 14 20:17:50 2004
@@ -1146,8 +1146,20 @@ local_send_motion (GdkDragContext *conte
       current_dest_drag->suggested_action = action;
       current_dest_drag->actions = current_dest_drag->suggested_action;
 
+#if 0
       tmp_event.dnd.x_root = x_root;
       tmp_event.dnd.y_root = y_root;
+#else
+      //FIXME: this appears to be working around a gdk/x11 bug
+      {
+        POINT pt;
+        pt.x = x_root;
+        pt.y = y_root;
+        ScreenToClient (GDK_WINDOW_HWND (tmp_event.dnd.window), &pt);
+        tmp_event.dnd.x_root = pt.x;
+        tmp_event.dnd.y_root = pt.y;
+      }
+#endif
 
       (GDK_DRAG_CONTEXT_PRIVATE_DATA (current_dest_drag))->last_x = x_root;
       (GDK_DRAG_CONTEXT_PRIVATE_DATA (current_dest_drag))->last_y = y_root;
@@ -1178,8 +1190,20 @@ local_send_drop (GdkDragContext *context
       tmp_event.dnd.context = current_dest_drag;
       tmp_event.dnd.time = GDK_CURRENT_TIME;
       
+#if 0
       tmp_event.dnd.x_root = private->last_x;
       tmp_event.dnd.y_root = private->last_y;
+#else
+      //FIXME: this appears to be working around a gdk/x11 bug
+      {
+        POINT pt;
+        pt.x = private->last_x;
+        pt.y = private->last_y;
+        ScreenToClient (GDK_WINDOW_HWND (tmp_event.dnd.window), &pt);
+        tmp_event.dnd.x_root = pt.x;
+        tmp_event.dnd.y_root = pt.y;
+      }
+#endif
       
       current_dest_drag = NULL;
 
@@ -1361,13 +1385,21 @@ gdk_drag_find_window_for_screen (GdkDrag
 				 GdkDragProtocol *protocol)
 {
   find_window_enum_arg a;
+  POINT pt;
+  HWND  hwnd;
 
-  a.x = x_root;
-  a.y = y_root;
+  pt.x = a.x = x_root;
+  pt.y = a.y = y_root;
   a.ignore = drag_window ? GDK_WINDOW_HWND (drag_window) : NULL;
   a.result = NULL;
 
+#if 0
   EnumWindows (find_window_enum_proc, (LPARAM) &a);
+#else
+  hwnd = WindowFromPoint (pt);
+  if (hwnd != a.ignore)
+    a.result = hwnd;
+#endif
 
   if (a.result == NULL)
     *dest_window = NULL;
@@ -1409,7 +1441,7 @@ gdk_drag_motion (GdkDragContext *context
 
   g_return_val_if_fail (context != NULL, FALSE);
 
-  GDK_NOTE (DND, g_print ("gdk_drag_motion\n"));
+  GDK_NOTE (DND, g_print ("gdk_drag_motion:"));
 
   private = GDK_DRAG_CONTEXT_PRIVATE_DATA (context);
   
@@ -1431,6 +1463,7 @@ gdk_drag_motion (GdkDragContext *context
 	  switch (protocol)
 	    {
 	    case GDK_DRAG_PROTO_LOCAL:
+                GDK_NOTE (DND, g_print (" local enter,"));
 	      local_send_enter (context, time);
 	      break;
 
@@ -1458,7 +1491,10 @@ gdk_drag_motion (GdkDragContext *context
 
       temp_event.dnd.context = context;
       temp_event.dnd.time = time;
+      temp_event.dnd.x_root = x_root;
+      temp_event.dnd.y_root = y_root;
 
+      GDK_NOTE (DND, g_print (" put,"));
       gdk_event_put (&temp_event);
     }
   else
@@ -1478,6 +1514,7 @@ gdk_drag_motion (GdkDragContext *context
 	  switch (context->protocol)
 	    {
 	    case GDK_DRAG_PROTO_LOCAL:
+                GDK_NOTE (DND, g_print (" local motion,"));
 	      local_send_motion (context, x_root, y_root, suggested_action, time);
 	      break;
 	      
@@ -1490,9 +1527,13 @@ gdk_drag_motion (GdkDragContext *context
 	    }
 	}
       else
-	return TRUE;
+	{
+	  GDK_NOTE (DND, g_print (" ret TRUE.\n"));
+	  return TRUE;
+	}
     }
 
+  GDK_NOTE (DND, g_print (" ret FALSE.\n"));
   return FALSE;
 }
 
@@ -1610,7 +1651,7 @@ gdk_drop_finish (GdkDragContext *context
 	
   g_return_if_fail (context != NULL);
 
-  GDK_NOTE (DND, g_print ("gdk_drop_finish"));
+  GDK_NOTE (DND, g_print ("gdk_drop_finish\n"));
 
   private = GDK_DRAG_CONTEXT_PRIVATE_DATA (context);
Comment 1 Owen Taylor 2004-03-15 00:28:33 UTC
The coordinates are definitely screen coordinates for the
X backend ... is gdk_window_get_position() working correctly
for toplevel windows on Win32? If it was always returning 0,
then it might seem you need window-relative coordinates.
Comment 2 Hans Breuer 2004-03-21 00:13:43 UTC
Thanks for the hint, fixed without touching the DND code at all :

2004-03-20  Hans Breuer  <hans@breuer.org>

	* gdk/gdkevents-win32.c (handle_configure_event) :
             (gdk_event_translate), WM_WINDOWPOSCHANGED : initialize
	GdkWindowObject::x, y with screen coords to make 
	gdk_window_get_position () return the right thing and thus fix
	drag and drop positioning (e.g. Gimp tabs, fixes bug #137192)
Comment 3 Tor Lillqvist 2004-03-21 00:25:19 UTC
Hans, in your commit, shouldn't the tests for window type == 
GDK_WINDOW_TOPLEVEL instead be window type != GDK_WINDOW_CHILD? Also 
GDK_WINDOW_DIALOG and GDK_WINDOW_TEMP windows are toplevel windows.
Comment 4 Tor Lillqvist 2004-03-21 01:47:27 UTC
BTW, Hans, I don't doubt that there was something wrong if you say 
so, but how did the intra-app dnd work wrong in GIMP before this? I 
can't see any difference in behaviour, at least not for drag-and-
dropping the tabs, but I probably am not looking at the right thing.
Comment 5 Hans Breuer 2004-03-21 09:19:45 UTC
If you try to drag a tab (say from layers and channels dialog) to the small 
stripe in the lower toolbox it should activate (higlight blue, the stripe not 
the tab) to show you where to drop.
This does work for me now regardless of the toplevel position. Formerly it was 
very difficult to find the drop target (the more difficult the bigger the distance 
of the window from the screen top and left).

Regarding the toplevel check: you are probably right, I simply was blinded 
by the apparent clear match between specification and code ;-)
Comment 6 Hans Breuer 2004-03-21 12:10:22 UTC
There is just one more step required to reproduce the bug 
described : the window needs to be move to get GdkWindowObject::x 
and y out of sync (and thus let gdk_window_get_position()
return wrong values).
Comment 7 Tor Lillqvist 2004-03-21 20:08:53 UTC
Ah yes, now I notice. Nice! I hadn't paid close enough attention 
earlier.