[gimp] app: get rid of server grabs for almost all canvas interaction



commit 01f92a3b486bbc2b8aac1771e613e07cb36d7f43
Author: Michael Natterer <mitch gimp org>
Date:   Fri Dec 9 20:20:02 2011 +0100

    app: get rid of server grabs for almost all canvas interaction
    
    Because it's generally the right thing to do, and server grabs broke
    badly with input devices / client side windows.
    
    gimpdisplayshell-grab.c: change logic to only server-grab if an event
    is passed to the pointer grab/ungrab functions, but always use
    gtk_grab_add/remove() which is sufficient in most cases.
    
    gimpdisplayshell-tool-events.c: have the grab functions grab the
    server only for space-bar scrolling and do all tool interaction,
    including ruler clicks, with gtk_grab_add/remove(). Refactor things
    a bit to also use the grab API for button-2 scrolling.
    
    gimpdeviceinfo-coords.c: transform the event's coords to the canvas'
    coordinate system, they might come from a ruler now.
    
    This fixes the following bugs:
    
    Bug 645315 - gimp_display_shell_pointer_grab: gdk_pointer_grab failed...
    Bug 644351 - Gimp misses some strokes especially when drawing fast
    Bug 645747 - Gimp is now unusable on xfce4

 app/display/gimpdisplayshell-grab.c        |   84 +++++++++++++---------------
 app/display/gimpdisplayshell-tool-events.c |   80 ++++++++++++---------------
 app/display/gimpdisplayshell.h             |    2 +
 app/widgets/gimpdeviceinfo-coords.c        |   25 ++++++++
 4 files changed, 100 insertions(+), 91 deletions(-)
---
diff --git a/app/display/gimpdisplayshell-grab.c b/app/display/gimpdisplayshell-grab.c
index 8769679..b0febc1 100644
--- a/app/display/gimpdisplayshell-grab.c
+++ b/app/display/gimpdisplayshell-grab.c
@@ -23,11 +23,6 @@
 
 #include "display-types.h"
 
-#include "widgets/gimpdeviceinfo.h"
-#include "widgets/gimpdevices.h"
-#include "widgets/gimpdevicemanager.h"
-
-#include "gimpdisplay.h"
 #include "gimpdisplayshell.h"
 #include "gimpdisplayshell-grab.h"
 
@@ -37,46 +32,33 @@ gimp_display_shell_pointer_grab (GimpDisplayShell *shell,
                                  const GdkEvent   *event,
                                  GdkEventMask      event_mask)
 {
-  GdkGrabStatus status;
-
   g_return_val_if_fail (GIMP_IS_DISPLAY_SHELL (shell), FALSE);
-  g_return_val_if_fail (event != NULL, FALSE);
   g_return_val_if_fail (shell->pointer_grabbed == FALSE, FALSE);
+  g_return_val_if_fail (shell->pointer_grab_time == 0, FALSE);
 
-  status = gdk_pointer_grab (gtk_widget_get_window (shell->canvas),
-                             FALSE, event_mask, NULL, NULL,
-                             gdk_event_get_time (event));
-
-  if (status == GDK_GRAB_SUCCESS)
+  if (event)
     {
-      shell->pointer_grabbed = TRUE;
+      GdkGrabStatus status;
 
-      return TRUE;
-    }
-  else if (status == GDK_GRAB_ALREADY_GRABBED)
-    {
-      GimpDeviceManager *manager;
-      GdkDisplay        *gdk_display;
+      status = gdk_pointer_grab (gtk_widget_get_window (shell->canvas),
+                                 FALSE, event_mask, NULL, NULL,
+                                 gdk_event_get_time (event));
 
-      manager = gimp_devices_get_manager (shell->display->gimp);
-      gdk_display = gtk_widget_get_display (GTK_WIDGET (shell));
-
-      /*  EEK: trying to grab an extended device always returns
-       *  ALREADY_GRABBED, so simply assume the grab succeeded anyway
-       */
-      if (gimp_device_manager_get_current_device (manager)->device !=
-          gdk_display_get_core_pointer (gdk_display))
+      if (status != GDK_GRAB_SUCCESS)
         {
-          shell->pointer_grabbed = TRUE;
-
-          return TRUE;
+          g_printerr ("%s: gdk_pointer_grab failed with status %d\n",
+                      G_STRFUNC, status);
+          return FALSE;
         }
+
+      shell->pointer_grab_time = gdk_event_get_time (event);
     }
 
-  g_printerr ("%s: gdk_pointer_grab failed with status %d\n",
-              G_STRFUNC, status);
+  gtk_grab_add (shell->canvas);
+
+  shell->pointer_grabbed = TRUE;
 
-  return FALSE;
+  return TRUE;
 }
 
 void
@@ -84,11 +66,18 @@ gimp_display_shell_pointer_ungrab (GimpDisplayShell *shell,
                                    const GdkEvent   *event)
 {
   g_return_if_fail (GIMP_IS_DISPLAY_SHELL (shell));
-  g_return_if_fail (event != NULL);
   g_return_if_fail (shell->pointer_grabbed == TRUE);
+  g_return_if_fail (event == NULL || shell->pointer_grab_time != 0);
 
-  gdk_display_pointer_ungrab (gtk_widget_get_display (shell->canvas),
-                              gdk_event_get_time (event));
+  gtk_grab_remove (shell->canvas);
+
+  if (event)
+    {
+      gdk_display_pointer_ungrab (gtk_widget_get_display (shell->canvas),
+                                  shell->pointer_grab_time);
+
+      shell->pointer_grab_time = 0;
+    }
 
   shell->pointer_grabbed = FALSE;
 }
@@ -102,22 +91,23 @@ gimp_display_shell_keyboard_grab (GimpDisplayShell *shell,
   g_return_val_if_fail (GIMP_IS_DISPLAY_SHELL (shell), FALSE);
   g_return_val_if_fail (event != NULL, FALSE);
   g_return_val_if_fail (shell->keyboard_grabbed == FALSE, FALSE);
+  g_return_val_if_fail (shell->keyboard_grab_time == 0, FALSE);
 
   status = gdk_keyboard_grab (gtk_widget_get_window (shell->canvas),
                               FALSE,
                               gdk_event_get_time (event));
 
-  if (status == GDK_GRAB_SUCCESS)
+  if (status != GDK_GRAB_SUCCESS)
     {
-      shell->keyboard_grabbed = TRUE;
-
-      return TRUE;
+      g_printerr ("%s: gdk_keyboard_grab failed with status %d\n",
+                  G_STRFUNC, status);
+      return FALSE;
     }
 
-  g_printerr ("%s: gdk_keyboard_grab failed with status %d\n",
-              G_STRFUNC, status);
+  shell->keyboard_grabbed   = TRUE;
+  shell->keyboard_grab_time = gdk_event_get_time (event);
 
-  return FALSE;
+  return TRUE;
 }
 
 void
@@ -127,9 +117,11 @@ gimp_display_shell_keyboard_ungrab (GimpDisplayShell *shell,
   g_return_if_fail (GIMP_IS_DISPLAY_SHELL (shell));
   g_return_if_fail (event != NULL);
   g_return_if_fail (shell->keyboard_grabbed == TRUE);
+  g_return_if_fail (shell->keyboard_grab_time != 0);
 
   gdk_display_keyboard_ungrab (gtk_widget_get_display (shell->canvas),
-                               gdk_event_get_time (event));
+                               shell->keyboard_grab_time);
 
-  shell->keyboard_grabbed = FALSE;
+  shell->keyboard_grabbed   = FALSE;
+  shell->keyboard_grab_time = 0;
 }
diff --git a/app/display/gimpdisplayshell-tool-events.c b/app/display/gimpdisplayshell-tool-events.c
index cc4bfe0..75d0cb9 100644
--- a/app/display/gimpdisplayshell-tool-events.c
+++ b/app/display/gimpdisplayshell-tool-events.c
@@ -78,9 +78,11 @@ static void       gimp_display_shell_proximity_out            (GimpDisplayShell
 static void       gimp_display_shell_check_device_cursor      (GimpDisplayShell  *shell);
 
 static void       gimp_display_shell_start_scrolling          (GimpDisplayShell  *shell,
+                                                               const GdkEvent    *event,
                                                                gint               x,
                                                                gint               y);
-static void       gimp_display_shell_stop_scrolling           (GimpDisplayShell  *shell);
+static void       gimp_display_shell_stop_scrolling           (GimpDisplayShell  *shell,
+                                                               const GdkEvent    *event);
 
 static void       gimp_display_shell_space_pressed            (GimpDisplayShell  *shell,
                                                                const GdkEvent    *event);
@@ -305,17 +307,11 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
   if (gimp_display_shell_events (canvas, event, shell))
     return TRUE;
 
-  /*  ignore events on overlays, but make sure key events go through
-   *  anyway because they are always originating from the toplevel
+  /*  ignore events on overlays, which are the canvas' children
    */
-  if (((GdkEventAny *) event)->window != gtk_widget_get_window (canvas))
+  if (gtk_widget_is_ancestor (gtk_get_event_widget (event), shell->canvas))
     {
-      if ((event->type != GDK_KEY_PRESS &&
-           event->type != GDK_KEY_RELEASE) ||
-          ! gtk_widget_has_focus (canvas))
-        {
-          return FALSE;
-        }
+      return FALSE;
     }
 
   display = shell->display;
@@ -360,6 +356,10 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
         if (cevent->mode != GDK_CROSSING_NORMAL)
           return TRUE;
 
+        /*  ignore enter notify while we have a grab  */
+        if (shell->pointer_grabbed)
+          return TRUE;
+
         gimp_display_shell_proximity_in (shell);
         update_sw_cursor = TRUE;
 
@@ -377,6 +377,10 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
         if (cevent->mode != GDK_CROSSING_NORMAL)
           return TRUE;
 
+        /*  ignore leave notify while we have a grab  */
+        if (shell->pointer_grabbed)
+          return TRUE;
+
         gimp_display_shell_proximity_out (shell);
 
         tool_manager_oper_update_active (gimp,
@@ -413,7 +417,7 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
             if (G_UNLIKELY (! gtk_widget_has_focus (canvas)))
               g_warning ("%s: FOCUS_IN but canvas has no focus", G_STRFUNC);
 
-            /*  ignore any focus changes while we have a grab  */
+            /*  ignore focus changes while we have a grab  */
             if (shell->pointer_grabbed)
               return TRUE;
 
@@ -426,7 +430,7 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
             if (G_UNLIKELY (gtk_widget_has_focus (canvas)))
               g_warning ("%s: FOCUS_OUT but canvas has focus", G_STRFUNC);
 
-             /*  ignore any focus changes while we have a grab  */
+            /*  ignore focus changes while we have a grab  */
             if (shell->pointer_grabbed)
               return TRUE;
 
@@ -520,13 +524,13 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
                   }
               }
 
-            if (! gimp_display_shell_pointer_grab (shell, event, event_mask))
+            if (! gimp_display_shell_pointer_grab (shell, NULL, 0))
               return TRUE;
 
             if (! shell->space_pressed && ! shell->space_release_pending)
               if (! gimp_display_shell_keyboard_grab (shell, event))
                 {
-                  gimp_display_shell_pointer_ungrab (shell, event);
+                  gimp_display_shell_pointer_ungrab (shell, NULL);
                   return TRUE;
                 }
 
@@ -561,7 +565,8 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
           }
         else if (bevent->button == 2)
           {
-            gimp_display_shell_start_scrolling (shell, bevent->x, bevent->y);
+            gimp_display_shell_start_scrolling (shell, NULL,
+                                                bevent->x, bevent->y);
           }
 
         return_val = TRUE;
@@ -644,10 +649,6 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
             if (! shell->space_pressed && ! shell->space_release_pending)
               gimp_display_shell_keyboard_ungrab (shell, event);
 
-            gimp_display_shell_pointer_ungrab (shell, event);
-
-            gtk_grab_add (canvas);
-
             if (active_tool &&
                 (! gimp_image_is_empty (image) ||
                  gimp_tool_control_get_handle_empty_image (active_tool->control)))
@@ -673,7 +674,7 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
               gimp_display_shell_update_focus (shell, FALSE,
                                                &image_coords, 0);
 
-            gtk_grab_remove (canvas);
+            gimp_display_shell_pointer_ungrab (shell, NULL);
 
             if (shell->space_release_pending)
               gimp_display_shell_space_released (shell, event, &image_coords);
@@ -681,7 +682,7 @@ gimp_display_shell_canvas_tool_events (GtkWidget        *canvas,
         else if (bevent->button == 2)
           {
             if (shell->scrolling)
-              gimp_display_shell_stop_scrolling (shell);
+              gimp_display_shell_stop_scrolling (shell, NULL);
           }
         else if (bevent->button == 3)
           {
@@ -1280,11 +1281,7 @@ gimp_display_shell_ruler_button_press (GtkWidget        *widget,
           gimp_display_shell_update_focus (shell, TRUE,
                                            NULL, event->state);
 
-          if (gimp_display_shell_pointer_grab (shell,
-                                               (GdkEvent *) event,
-                                               GDK_POINTER_MOTION_HINT_MASK |
-                                               GDK_BUTTON1_MOTION_MASK |
-                                               GDK_BUTTON_RELEASE_MASK))
+          if (gimp_display_shell_pointer_grab (shell, NULL, 0))
             {
               if (gimp_display_shell_keyboard_grab (shell,
                                                     (GdkEvent *) event))
@@ -1300,8 +1297,7 @@ gimp_display_shell_ruler_button_press (GtkWidget        *widget,
                 }
               else
                 {
-                  gimp_display_shell_pointer_ungrab (shell,
-                                                     (GdkEvent *) event);
+                  gimp_display_shell_pointer_ungrab (shell, NULL);
                 }
             }
         }
@@ -1405,32 +1401,34 @@ gimp_display_shell_check_device_cursor (GimpDisplayShell *shell)
 
 static void
 gimp_display_shell_start_scrolling (GimpDisplayShell *shell,
+                                    const GdkEvent   *event,
                                     gint              x,
                                     gint              y)
 {
   g_return_if_fail (! shell->scrolling);
 
+  gimp_display_shell_pointer_grab (shell, event, GDK_POINTER_MOTION_MASK);
+
   shell->scrolling      = TRUE;
   shell->scroll_start_x = x + shell->offset_x;
   shell->scroll_start_y = y + shell->offset_y;
 
   gimp_display_shell_set_override_cursor (shell, GDK_FLEUR);
-
-  gtk_grab_add (shell->canvas);
 }
 
 static void
-gimp_display_shell_stop_scrolling (GimpDisplayShell *shell)
+gimp_display_shell_stop_scrolling (GimpDisplayShell *shell,
+                                   const GdkEvent   *event)
 {
   g_return_if_fail (shell->scrolling);
 
+  gimp_display_shell_unset_override_cursor (shell);
+
   shell->scrolling      = FALSE;
   shell->scroll_start_x = 0;
   shell->scroll_start_y = 0;
 
-  gimp_display_shell_unset_override_cursor (shell);
-
-  gtk_grab_remove (shell->canvas);
+  gimp_display_shell_pointer_ungrab (shell, event);
 }
 
 static void
@@ -1456,14 +1454,6 @@ gimp_display_shell_space_pressed (GimpDisplayShell *shell,
         GimpDeviceInfo    *current_device;
         GimpCoords         coords;
 
-        if (! gimp_display_shell_pointer_grab (shell, event,
-                                               GDK_POINTER_MOTION_MASK |
-                                               GDK_POINTER_MOTION_HINT_MASK))
-          {
-            gimp_display_shell_keyboard_ungrab (shell, event);
-            return;
-          }
-
         manager = gimp_devices_get_manager (gimp);
         current_device = gimp_device_manager_get_current_device (manager);
 
@@ -1471,7 +1461,8 @@ gimp_display_shell_space_pressed (GimpDisplayShell *shell,
                                             gtk_widget_get_window (shell->canvas),
                                             &coords);
 
-        gimp_display_shell_start_scrolling (shell, coords.x, coords.y);
+        gimp_display_shell_start_scrolling (shell, event,
+                                            coords.x, coords.y);
       }
       break;
 
@@ -1516,8 +1507,7 @@ gimp_display_shell_space_released (GimpDisplayShell *shell,
       break;
 
     case GIMP_SPACE_BAR_ACTION_PAN:
-      gimp_display_shell_stop_scrolling (shell);
-      gimp_display_shell_pointer_ungrab (shell, event);
+      gimp_display_shell_stop_scrolling (shell, event);
       break;
 
     case GIMP_SPACE_BAR_ACTION_MOVE:
diff --git a/app/display/gimpdisplayshell.h b/app/display/gimpdisplayshell.h
index 01822b6..42de82e 100644
--- a/app/display/gimpdisplayshell.h
+++ b/app/display/gimpdisplayshell.h
@@ -171,6 +171,8 @@ struct _GimpDisplayShell
   /*  the state of gimp_display_shell_tool_events()  */
   gboolean           pointer_grabbed;
   gboolean           keyboard_grabbed;
+  gboolean           pointer_grab_time;
+  gboolean           keyboard_grab_time;
 
   gboolean           space_pressed;
   gboolean           space_release_pending;
diff --git a/app/widgets/gimpdeviceinfo-coords.c b/app/widgets/gimpdeviceinfo-coords.c
index ba63d58..dfb39f6 100644
--- a/app/widgets/gimpdeviceinfo-coords.c
+++ b/app/widgets/gimpdeviceinfo-coords.c
@@ -45,6 +45,31 @@ gimp_device_info_get_event_coords (GimpDeviceInfo *info,
       coords->x = x;
       gdk_event_get_axis (event, GDK_AXIS_Y, &coords->y);
 
+      /* translate event coordinates to window coordinates, only
+       * happens if we drag a guide from a ruler
+       */
+      if (event->any.window &&
+          event->any.window != window)
+        {
+          GtkWidget *src_widget;
+          GtkWidget *dest_widget;
+
+          src_widget = gtk_get_event_widget ((GdkEvent *) event);
+          gdk_window_get_user_data (window, (gpointer) &dest_widget);
+
+          if (src_widget && dest_widget)
+            {
+              gint offset_x;
+              gint offset_y;
+
+              gtk_widget_translate_coordinates (src_widget, dest_widget,
+                                                0, 0, &offset_x, &offset_y);
+
+              coords->x += offset_x;
+              coords->y += offset_y;
+            }
+        }
+
       if (gdk_event_get_axis (event, GDK_AXIS_PRESSURE, &coords->pressure))
         {
           coords->pressure = gimp_device_info_map_axis (info,



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]