[gtk/wip/chergert/gdk-macos-gdkdrag: 6/8] macos: implement GdkDrop for macOS




commit 40933bb28cf9741953102143d3a666ca73d4b588
Author: Christian Hergert <chergert redhat com>
Date:   Thu Jun 17 17:23:10 2021 -0700

    macos: implement GdkDrop for macOS
    
    This gets the basic mechanics of the drop portion of DnD working on the
    macOS backend. You can drag, for example, from TextEdit into GNOME
    Text Editor when using the macOS backend.
    
    Other content formats are supported, and match what is currently
    supported by the clipboard backend as the implementation to read
    from the pasteboard is shared.
    
    Currently, we look up the GdkDrag for the new GdkDrop. However,
    nothing is stashing the drag away for further lookup. More work is
    needed on GdkMacosDrag for that to be doable.

 gdk/macos/GdkMacosWindow.c          |  68 +++++++++++++-
 gdk/macos/gdkmacosdisplay-private.h |  14 +++
 gdk/macos/gdkmacosdisplay.c         |  58 ++++++++++++
 gdk/macos/gdkmacosdrop-private.h    |  66 +++++++++++++
 gdk/macos/gdkmacosdrop.c            | 183 ++++++++++++++++++++++++++++++++++++
 gdk/macos/meson.build               |   1 +
 6 files changed, 387 insertions(+), 3 deletions(-)
---
diff --git a/gdk/macos/GdkMacosWindow.c b/gdk/macos/GdkMacosWindow.c
index 8f3817ee6a..714a5047cd 100644
--- a/gdk/macos/GdkMacosWindow.c
+++ b/gdk/macos/GdkMacosWindow.c
@@ -30,6 +30,7 @@
 
 #include "gdkmacosclipboard-private.h"
 #include "gdkmacosdisplay-private.h"
+#include "gdkmacosdrop-private.h"
 #include "gdkmacosmonitor-private.h"
 #include "gdkmacossurface-private.h"
 #include "gdkmacospopupsurface-private.h"
@@ -601,25 +602,86 @@ typedef NSString *CALayerContentsGravity;
 
 -(NSDragOperation)draggingEntered:(id <NSDraggingInfo>)sender
 {
-  return NSDragOperationNone;
+  NSPoint location = [sender draggingLocation];
+  NSDragOperation ret;
+  GdkMacosDrop *drop;
+
+  if (!(drop = _gdk_macos_drop_new ([self gdkSurface], sender)))
+    return NSDragOperationNone;
+
+  _gdk_macos_display_set_drop ([self gdkDisplay],
+                               [sender draggingSequenceNumber],
+                               GDK_DROP (drop));
+
+  gdk_drop_emit_enter_event (GDK_DROP (drop),
+                             TRUE,
+                             location.x,
+                             GDK_SURFACE (gdk_surface)->height - location.y,
+                             GDK_CURRENT_TIME);
+
+  ret = _gdk_macos_drop_operation (drop);
+
+  g_object_unref (drop);
+
+  return ret;
 }
 
 -(void)draggingEnded:(id <NSDraggingInfo>)sender
 {
+  _gdk_macos_display_set_drop ([self gdkDisplay], [sender draggingSequenceNumber], NULL);
 }
 
 -(void)draggingExited:(id <NSDraggingInfo>)sender
 {
+  NSInteger sequence_number = [sender draggingSequenceNumber];
+  GdkDrop *drop = _gdk_macos_display_find_drop ([self gdkDisplay], sequence_number);
+
+  if (drop != NULL)
+    gdk_drop_emit_leave_event (drop, TRUE, GDK_CURRENT_TIME);
+
+  _gdk_macos_display_set_drop ([self gdkDisplay], sequence_number, NULL);
 }
 
 -(NSDragOperation)draggingUpdated:(id <NSDraggingInfo>)sender
 {
-  return NSDragOperationNone;
+  NSInteger sequence_number = [sender draggingSequenceNumber];
+  GdkDisplay *display = gdk_surface_get_display (GDK_SURFACE (gdk_surface));
+  GdkDrop *drop = _gdk_macos_display_find_drop (GDK_MACOS_DISPLAY (display), sequence_number);
+  NSPoint location = [sender draggingLocation];
+
+  if (drop == NULL)
+    return NSDragOperationNone;
+
+  _gdk_macos_drop_update_actions (GDK_MACOS_DROP (drop), sender);
+
+  gdk_drop_emit_motion_event (drop,
+                              TRUE,
+                              location.x,
+                              GDK_SURFACE (gdk_surface)->height - location.y,
+                              GDK_CURRENT_TIME);
+
+  return _gdk_macos_drop_operation (GDK_MACOS_DROP (drop));
 }
 
 -(BOOL)performDragOperation:(id <NSDraggingInfo>)sender
 {
-  return YES;
+  NSInteger sequence_number = [sender draggingSequenceNumber];
+  GdkDisplay *display = gdk_surface_get_display (GDK_SURFACE (gdk_surface));
+  GdkDrop *drop = _gdk_macos_display_find_drop (GDK_MACOS_DISPLAY (display), sequence_number);
+  NSPoint location = [sender draggingLocation];
+
+  if (drop == NULL)
+    return NO;
+
+  gdk_drop_emit_drop_event (drop,
+                            TRUE,
+                            location.x,
+                            GDK_SURFACE (gdk_surface)->height - location.y,
+                            GDK_CURRENT_TIME);
+
+  gdk_drop_emit_leave_event (drop, TRUE, GDK_CURRENT_TIME);
+
+  return GDK_MACOS_DROP (drop)->finish_action != 0;
 }
 
 -(BOOL)wantsPeriodicDraggingUpdates
diff --git a/gdk/macos/gdkmacosdisplay-private.h b/gdk/macos/gdkmacosdisplay-private.h
index 630daceae0..c25ff3b0c1 100644
--- a/gdk/macos/gdkmacosdisplay-private.h
+++ b/gdk/macos/gdkmacosdisplay-private.h
@@ -81,6 +81,10 @@ struct _GdkMacosDisplay
   /* The surface that is receiving keyboard events */
   GdkMacosSurface *keyboard_surface;
 
+  /* [NSDraggingInfo draggingSequenceNumber] to GdkMacosDr(ag,op) */
+  GHashTable *active_drags;
+  GHashTable *active_drops;
+
   /* Used to translate from quartz coordinate space to GDK */
   int width;
   int height;
@@ -160,6 +164,16 @@ void             _gdk_macos_display_warp_pointer                   (GdkMacosDisp
                                                                     int              x,
                                                                     int              y);
 NSEvent         *_gdk_macos_display_get_nsevent                    (GdkEvent        *event);
+GdkDrag         *_gdk_macos_display_find_drag                      (GdkMacosDisplay *self,
+                                                                    NSInteger        sequence_number);
+GdkDrop         *_gdk_macos_display_find_drop                      (GdkMacosDisplay *self,
+                                                                    NSInteger        sequence_number);
+void             _gdk_macos_display_set_drag                       (GdkMacosDisplay *self,
+                                                                    NSInteger        sequence_number,
+                                                                    GdkDrag         *drag);
+void             _gdk_macos_display_set_drop                       (GdkMacosDisplay *self,
+                                                                    NSInteger        sequence_number,
+                                                                    GdkDrop         *drop);
 
 G_END_DECLS
 
diff --git a/gdk/macos/gdkmacosdisplay.c b/gdk/macos/gdkmacosdisplay.c
index ed15a49bf2..20041ac076 100644
--- a/gdk/macos/gdkmacosdisplay.c
+++ b/gdk/macos/gdkmacosdisplay.c
@@ -31,6 +31,8 @@
 #include "gdkmacoscairocontext-private.h"
 #include "gdkmacoseventsource-private.h"
 #include "gdkmacosdisplay-private.h"
+#include "gdkmacosdrag-private.h"
+#include "gdkmacosdrop-private.h"
 #include "gdkmacosglcontext-private.h"
 #include "gdkmacoskeymap-private.h"
 #include "gdkmacosmonitor-private.h"
@@ -663,6 +665,8 @@ gdk_macos_display_finalize (GObject *object)
                                       CFSTR ("NSUserDefaultsDidChangeNotification"),
                                       NULL);
 
+  g_clear_pointer (&self->active_drags, g_hash_table_unref);
+  g_clear_pointer (&self->active_drops, g_hash_table_unref);
   g_clear_object (&GDK_DISPLAY (self)->clipboard);
   g_clear_pointer (&self->frame_source, g_source_unref);
   g_clear_object (&self->monitors);
@@ -701,6 +705,8 @@ static void
 gdk_macos_display_init (GdkMacosDisplay *self)
 {
   self->monitors = g_list_store_new (GDK_TYPE_MONITOR);
+  self->active_drags = g_hash_table_new_full (NULL, NULL, NULL, g_object_unref);
+  self->active_drops = g_hash_table_new_full (NULL, NULL, NULL, g_object_unref);
 
   gdk_display_set_composited (GDK_DISPLAY (self), TRUE);
   gdk_display_set_input_shapes (GDK_DISPLAY (self), FALSE);
@@ -1113,3 +1119,55 @@ _gdk_macos_display_get_nsevent (GdkEvent *event)
 
   return NULL;
 }
+
+GdkDrag *
+_gdk_macos_display_find_drag (GdkMacosDisplay *self,
+                              NSInteger        sequence_number)
+{
+  g_return_val_if_fail (GDK_IS_MACOS_DISPLAY (self), NULL);
+
+  return g_hash_table_lookup (self->active_drags, GSIZE_TO_POINTER (sequence_number));
+}
+
+void
+_gdk_macos_display_set_drag (GdkMacosDisplay *self,
+                             NSInteger        sequence_number,
+                             GdkDrag         *drag)
+{
+  g_return_if_fail (GDK_IS_MACOS_DISPLAY (self));
+  g_return_if_fail (!drag || GDK_IS_MACOS_DRAG (drag));
+
+  if (drag)
+    g_hash_table_insert (self->active_drags,
+                         GSIZE_TO_POINTER (sequence_number),
+                         g_object_ref (drag));
+  else
+    g_hash_table_remove (self->active_drags,
+                         GSIZE_TO_POINTER (sequence_number));
+}
+
+GdkDrop *
+_gdk_macos_display_find_drop (GdkMacosDisplay *self,
+                              NSInteger        sequence_number)
+{
+  g_return_val_if_fail (GDK_IS_MACOS_DISPLAY (self), NULL);
+
+  return g_hash_table_lookup (self->active_drops, GSIZE_TO_POINTER (sequence_number));
+}
+
+void
+_gdk_macos_display_set_drop (GdkMacosDisplay *self,
+                             NSInteger        sequence_number,
+                             GdkDrop         *drop)
+{
+  g_return_if_fail (GDK_IS_MACOS_DISPLAY (self));
+  g_return_if_fail (!drop || GDK_IS_MACOS_DROP (drop));
+
+  if (drop)
+    g_hash_table_insert (self->active_drops,
+                         GSIZE_TO_POINTER (sequence_number),
+                         g_object_ref (drop));
+  else
+    g_hash_table_remove (self->active_drops,
+                         GSIZE_TO_POINTER (sequence_number));
+}
diff --git a/gdk/macos/gdkmacosdrop-private.h b/gdk/macos/gdkmacosdrop-private.h
new file mode 100644
index 0000000000..8472e87c23
--- /dev/null
+++ b/gdk/macos/gdkmacosdrop-private.h
@@ -0,0 +1,66 @@
+/*
+ * Copyright © 2021 Red Hat, Inc.
+ *
+ * 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.1 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/>.
+ *
+ * SPDX-License-Identifier: LGPL-2.1-or-later
+ */
+
+#ifndef __GDK_MACOS_DROP_PRIVATE_H__
+#define __GDK_MACOS_DROP_PRIVATE_H__
+
+#include <AppKit/AppKit.h>
+
+#include "gdkdropprivate.h"
+
+#include "gdkmacossurface-private.h"
+
+G_BEGIN_DECLS
+
+#define GDK_TYPE_MACOS_DROP            (gdk_macos_drop_get_type ())
+#define GDK_MACOS_DROP(object)         (G_TYPE_CHECK_INSTANCE_CAST ((object), GDK_TYPE_MACOS_DROP, 
GdkMacosDrop))
+#define GDK_MACOS_DROP_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST ((klass), GDK_TYPE_MACOS_DROP, 
GdkMacosDropClass))
+#define GDK_IS_MACOS_DROP(object)      (G_TYPE_CHECK_INSTANCE_TYPE ((object), GDK_TYPE_MACOS_DROP))
+#define GDK_IS_MACOS_DROP_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GDK_TYPE_MACOS_DROP))
+#define GDK_MACOS_DROP_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS ((obj), GDK_TYPE_MACOS_DROP, 
GdkMacosDropClass))
+
+typedef struct _GdkMacosDrop GdkMacosDrop;
+typedef struct _GdkMacosDropClass GdkMacosDropClass;
+
+struct _GdkMacosDrop
+{
+  GdkDrop parent_instance;
+
+  NSPasteboard *pasteboard;
+
+  GdkDragAction all_actions;
+  GdkDragAction preferred_action;
+  GdkDragAction finish_action;
+};
+
+struct _GdkMacosDropClass
+{
+  GdkDropClass parent_class;
+};
+
+GType             gdk_macos_drop_get_type       (void) G_GNUC_CONST;
+GdkMacosDrop    *_gdk_macos_drop_new            (GdkMacosSurface    *surface,
+                                                 id<NSDraggingInfo>  info);
+NSDragOperation  _gdk_macos_drop_operation      (GdkMacosDrop       *self);
+void             _gdk_macos_drop_update_actions (GdkMacosDrop       *self,
+                                                 id<NSDraggingInfo>  info);
+
+G_END_DECLS
+
+#endif /* __GDK_MACOS_DROP_PRIVATE_H__ */
diff --git a/gdk/macos/gdkmacosdrop.c b/gdk/macos/gdkmacosdrop.c
new file mode 100644
index 0000000000..28c06f6220
--- /dev/null
+++ b/gdk/macos/gdkmacosdrop.c
@@ -0,0 +1,183 @@
+/*
+ * Copyright © 2021 Red Hat, Inc.
+ *
+ * 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.1 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/>.
+ *
+ * SPDX-License-Identifier: LGPL-2.1-or-later
+ */
+
+#include "config.h"
+
+#include "gdkmacosclipboard-private.h"
+#include "gdkmacosdisplay-private.h"
+#include "gdkmacosdrag-private.h"
+#include "gdkmacosdrop-private.h"
+
+G_DEFINE_TYPE (GdkMacosDrop, gdk_macos_drop, GDK_TYPE_DROP)
+
+static void
+gdk_macos_drop_status (GdkDrop       *drop,
+                       GdkDragAction  actions,
+                       GdkDragAction  preferred)
+{
+  GdkMacosDrop *self = (GdkMacosDrop *)drop;
+
+  g_assert (GDK_IS_MACOS_DROP (self));
+
+  self->all_actions = actions;
+  self->preferred_action = preferred;
+}
+
+static void
+gdk_macos_drop_read_async (GdkDrop             *drop,
+                           GdkContentFormats   *content_formats,
+                           int                  io_priority,
+                           GCancellable        *cancellable,
+                           GAsyncReadyCallback  callback,
+                           gpointer             user_data)
+{
+  _gdk_macos_pasteboard_read_async (G_OBJECT (drop),
+                                    GDK_MACOS_DROP (drop)->pasteboard,
+                                    content_formats,
+                                    io_priority,
+                                    cancellable,
+                                    callback,
+                                    user_data);
+}
+
+static GInputStream *
+gdk_macos_drop_read_finish (GdkDrop       *drop,
+                            GAsyncResult  *result,
+                            const char   **out_mime_type,
+                            GError       **error)
+{
+  return _gdk_macos_pasteboard_read_finish (G_OBJECT (drop), result, out_mime_type, error);
+}
+
+static void
+gdk_macos_drop_finish (GdkDrop       *drop,
+                       GdkDragAction  action)
+{
+  g_assert (GDK_IS_MACOS_DROP (drop));
+
+  GDK_MACOS_DROP (drop)->finish_action = action;
+}
+
+static void
+gdk_macos_drop_finalize (GObject *object)
+{
+  GdkMacosDrop *self = (GdkMacosDrop *)object;
+
+  if (self->pasteboard)
+    {
+      [self->pasteboard release];
+      self->pasteboard = NULL;
+    }
+
+  G_OBJECT_CLASS (gdk_macos_drop_parent_class)->finalize (object);
+}
+
+static void
+gdk_macos_drop_class_init (GdkMacosDropClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+  GdkDropClass *drop_class = GDK_DROP_CLASS (klass);
+
+  object_class->finalize = gdk_macos_drop_finalize;
+
+  drop_class->status = gdk_macos_drop_status;
+  drop_class->read_async = gdk_macos_drop_read_async;
+  drop_class->read_finish = gdk_macos_drop_read_finish;
+  drop_class->finish = gdk_macos_drop_finish;
+}
+
+static void
+gdk_macos_drop_init (GdkMacosDrop *self)
+{
+}
+
+void
+_gdk_macos_drop_update_actions (GdkMacosDrop       *self,
+                                id<NSDraggingInfo>  info)
+{
+  NSDragOperation op;
+  GdkDragAction actions = 0;
+
+  g_assert (GDK_IS_MACOS_DROP (self));
+
+  op = [info draggingSourceOperationMask];
+
+  if (op & NSDragOperationCopy)
+    actions |= GDK_ACTION_COPY;
+
+  if (op & NSDragOperationLink)
+    actions |= GDK_ACTION_LINK;
+
+  if (op & NSDragOperationMove)
+    actions |= GDK_ACTION_MOVE;
+
+  gdk_drop_set_actions (GDK_DROP (self), actions);
+}
+
+GdkMacosDrop *
+_gdk_macos_drop_new (GdkMacosSurface    *surface,
+                     id<NSDraggingInfo>  info)
+{
+  GdkDrag *drag = NULL;
+  GdkContentFormats *content_formats;
+  GdkMacosDrop *self;
+  GdkDisplay *display;
+  GdkDevice *device;
+  GdkSeat *seat;
+
+  g_return_val_if_fail (GDK_IS_MACOS_SURFACE (surface), NULL);
+  g_return_val_if_fail (info != NULL, NULL);
+
+  display = gdk_surface_get_display (GDK_SURFACE (surface));
+  seat = gdk_display_get_default_seat (display);
+  device = gdk_seat_get_pointer (seat);
+  drag = _gdk_macos_display_find_drag (GDK_MACOS_DISPLAY (display), [info draggingSequenceNumber]);
+
+  content_formats = _gdk_macos_pasteboard_load_formats ([info draggingPasteboard]);
+
+  self = g_object_new (GDK_TYPE_MACOS_DROP,
+                       "device", device,
+                       "drag", drag,
+                       "formats", content_formats,
+                       "surface", surface,
+                       NULL);
+
+  self->pasteboard = [[info draggingPasteboard] retain];
+
+  _gdk_macos_drop_update_actions (self, info);
+
+  gdk_content_formats_unref (content_formats);
+
+  return g_steal_pointer (&self);
+}
+
+NSDragOperation
+_gdk_macos_drop_operation (GdkMacosDrop *self)
+{
+  if (self->preferred_action & GDK_ACTION_LINK)
+    return NSDragOperationLink;
+
+  if (self->preferred_action & GDK_ACTION_MOVE)
+    return NSDragOperationMove;
+
+  if (self->preferred_action & GDK_ACTION_COPY)
+    return NSDragOperationCopy;
+  
+  return NSDragOperationNone;
+}
diff --git a/gdk/macos/meson.build b/gdk/macos/meson.build
index 943fb84457..3a10dbf944 100644
--- a/gdk/macos/meson.build
+++ b/gdk/macos/meson.build
@@ -10,6 +10,7 @@ gdk_macos_sources = files([
   'gdkmacosdisplay-settings.c',
   'gdkmacosdisplay-translate.c',
   'gdkmacosdrag.c',
+  'gdkmacosdrop.c',
   'gdkmacosdragsurface.c',
   'gdkmacosglcontext.c',
   'gdkmacoseventsource.c',


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