summaryrefslogtreecommitdiff
path: root/gtk/gtkdropcontrollermotion.c
diff options
context:
space:
mode:
authorBenjamin Otte <otte@redhat.com>2020-02-22 00:58:57 +0100
committerBenjamin Otte <otte@redhat.com>2020-02-22 07:33:40 +0100
commit34e5f36fdf3c596d8520228f755793ee27139b25 (patch)
treefd4677532e56108f0691027cec3eaf96cb866ae3 /gtk/gtkdropcontrollermotion.c
parent9223d1ee047712501232ebe2e480f72320956bca (diff)
downloadgtk+-34e5f36fdf3c596d8520228f755793ee27139b25.tar.gz
Add GtkDropControllerMotion
Diffstat (limited to 'gtk/gtkdropcontrollermotion.c')
-rw-r--r--gtk/gtkdropcontrollermotion.c394
1 files changed, 394 insertions, 0 deletions
diff --git a/gtk/gtkdropcontrollermotion.c b/gtk/gtkdropcontrollermotion.c
new file mode 100644
index 0000000000..51199b729a
--- /dev/null
+++ b/gtk/gtkdropcontrollermotion.c
@@ -0,0 +1,394 @@
+/*
+ * Copyright © 2020 Benjamin Otte
+ *
+ * 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/>.
+ *
+ * Authors: Benjamin Otte <otte@gnome.org>
+ */
+
+/**
+ * SECTION:gtkdropcontrollermotion
+ * @Short_description: Event controller for motion events during a drop
+ * @Title: GtkDropControllerMotion
+ * @See_also: #GtkDropControllerMotion, #GdkDrop, #GtkDropTarget
+ *
+ * #GtkDropControllerMotion is an event controller meant for tracking
+ * the pointer hovering over a widget during a drag and drop operation.
+ *
+ * It is modeled after #GtkEventControllerMotion so if you have used
+ * that, this should feel really familiar.
+ *
+ * The drop controller is not able to accept drops, use #GtkDropTarget
+ * for that purpose.
+ **/
+
+#include "config.h"
+
+#include "gtkdropcontrollermotion.h"
+
+#include "gtkintl.h"
+#include "gtkprivate.h"
+#include "gtkwidgetprivate.h"
+#include "gtkmarshalers.h"
+#include "gtkeventcontrollerprivate.h"
+#include "gtktypebuiltins.h"
+#include "gtkmarshalers.h"
+
+struct _GtkDropControllerMotion
+{
+ GtkEventController parent_instance;
+
+ GdkDrop *drop;
+ guint is_pointer : 1;
+ guint contains_pointer : 1;
+};
+
+struct _GtkDropControllerMotionClass
+{
+ GtkEventControllerClass parent_class;
+};
+
+enum {
+ ENTER,
+ LEAVE,
+ MOTION,
+ N_SIGNALS
+};
+
+enum {
+ PROP_0,
+ PROP_CONTAINS_POINTER,
+ PROP_DROP,
+ PROP_IS_POINTER,
+ NUM_PROPERTIES
+};
+
+static GParamSpec *props[NUM_PROPERTIES] = { NULL, };
+
+static guint signals[N_SIGNALS] = { 0 };
+
+G_DEFINE_TYPE (GtkDropControllerMotion, gtk_drop_controller_motion, GTK_TYPE_EVENT_CONTROLLER)
+
+static gboolean
+gtk_drop_controller_motion_handle_event (GtkEventController *controller,
+ GdkEvent *event,
+ double x,
+ double y)
+{
+ GtkEventControllerClass *parent_class;
+ GdkEventType type;
+
+ type = gdk_event_get_event_type (event);
+ if (type == GDK_DRAG_MOTION)
+ g_signal_emit (controller, signals[MOTION], 0, x, y);
+
+ parent_class = GTK_EVENT_CONTROLLER_CLASS (gtk_drop_controller_motion_parent_class);
+
+ return parent_class->handle_event (controller, event, x, y);
+}
+
+static void
+update_pointer_focus (GtkEventController *controller,
+ const GtkCrossingData *crossing,
+ double x,
+ double y)
+{
+ GtkDropControllerMotion *self = GTK_DROP_CONTROLLER_MOTION (controller);
+ GtkWidget *widget = gtk_event_controller_get_widget (controller);
+ gboolean is_pointer = FALSE;
+ gboolean contains_pointer = FALSE;
+ gboolean enter = FALSE;
+ gboolean leave = FALSE;
+
+ if (crossing->direction == GTK_CROSSING_IN)
+ {
+ if (crossing->new_descendent != NULL)
+ {
+ contains_pointer = TRUE;
+ }
+ if (crossing->new_target == widget)
+ {
+ contains_pointer = TRUE;
+ is_pointer = TRUE;
+ }
+ }
+ else
+ {
+ if (crossing->new_descendent != NULL ||
+ crossing->new_target == widget)
+ contains_pointer = TRUE;
+ is_pointer = FALSE;
+ }
+
+ if (self->contains_pointer != contains_pointer)
+ {
+ enter = contains_pointer;
+ leave = !contains_pointer;
+ }
+
+ if (leave)
+ g_signal_emit (controller, signals[LEAVE], 0);
+
+ g_object_freeze_notify (G_OBJECT (self));
+ if (self->is_pointer != is_pointer)
+ {
+ self->is_pointer = is_pointer;
+ g_object_notify (G_OBJECT (self), "is-pointer");
+ }
+ if (self->contains_pointer != contains_pointer)
+ {
+ self->contains_pointer = contains_pointer;
+ if (contains_pointer)
+ self->drop = g_object_ref (crossing->drop);
+ else
+ g_clear_object (&self->drop);
+ g_object_notify (G_OBJECT (self), "contains-pointer");
+ g_object_notify (G_OBJECT (self), "drop");
+ }
+ g_object_thaw_notify (G_OBJECT (self));
+
+ if (enter)
+ g_signal_emit (controller, signals[ENTER], 0, x, y);
+}
+
+static void
+gtk_drop_controller_motion_handle_crossing (GtkEventController *controller,
+ const GtkCrossingData *crossing,
+ double x,
+ double y)
+{
+ if (crossing->type == GTK_CROSSING_DROP)
+ update_pointer_focus (controller, crossing, x, y);
+}
+
+static void
+gtk_drop_controller_motion_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GtkDropControllerMotion *self = GTK_DROP_CONTROLLER_MOTION (object);
+
+ switch (prop_id)
+ {
+ case PROP_CONTAINS_POINTER:
+ g_value_set_boolean (value, self->contains_pointer);
+ break;
+
+ case PROP_DROP:
+ g_value_set_object (value, self->drop);
+ break;
+
+ case PROP_IS_POINTER:
+ g_value_set_boolean (value, self->is_pointer);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+gtk_drop_controller_motion_class_init (GtkDropControllerMotionClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkEventControllerClass *controller_class = GTK_EVENT_CONTROLLER_CLASS (klass);
+
+ object_class->get_property = gtk_drop_controller_motion_get_property;
+
+ controller_class->handle_event = gtk_drop_controller_motion_handle_event;
+ controller_class->handle_crossing = gtk_drop_controller_motion_handle_crossing;
+
+ /**
+ * GtkDropControllerMotion:contains-pointer:
+ *
+ * Whether the pointer of a drag and drop operation is in the controller's
+ * widget or a descendant.
+ * See also #GtkDropControllerMotion:is-pointer.
+ *
+ * When handling crossing events, this property is updated
+ * before #GtkDropControllerMotion::enter but after
+ * #GtkDropControllerMotion::leave is emitted.
+ */
+ props[PROP_CONTAINS_POINTER] =
+ g_param_spec_boolean ("contains-pointer",
+ P_("Contains Pointer"),
+ P_("Whether the pointer is inthe controllers widget or a descendant"),
+ FALSE,
+ G_PARAM_READABLE);
+
+ /**
+ * GtkDropControllerMotion:drop:
+ *
+ * The ongoing drop operation over the controller's widget or its descendant.
+ * If no drop operation is going on, this property returns %NULL.
+ *
+ * The event controller should not modify the @drop, but it might want to query
+ * its properties.
+ *
+ * When handling crossing events, this property is updated
+ * before #GtkDropControllerMotion::enter but after
+ * #GtkDropControllerMotion::leave is emitted.
+ */
+ props[PROP_DROP] =
+ g_param_spec_object ("drop",
+ P_("Drop"),
+ P_("The ongoing drop operation"),
+ GDK_TYPE_DROP,
+ G_PARAM_READABLE);
+
+ /**
+ * GtkDropControllerMotion:is-pointer:
+ *
+ * Whether the pointer is in the controllers widget itself,
+ * as opposed to in a descendent widget. See also
+ * #GtkDropControllerMotion:contains-pointer.
+ *
+ * When handling crossing events, this property is updated
+ * before #GtkDropControllerMotion::enter but after
+ * #GtkDropControllerMotion::leave is emitted.
+ */
+ props[PROP_IS_POINTER] =
+ g_param_spec_boolean ("is-pointer",
+ P_("Is Pointer"),
+ P_("Whether the pointer is in the controllers widget"),
+ FALSE,
+ G_PARAM_READABLE);
+
+ g_object_class_install_properties (object_class, NUM_PROPERTIES, props);
+
+ /**
+ * GtkDropControllerMotion::enter:
+ * @self: the object which received the signal
+ * @x: coordinates of pointer location
+ * @y: coordinates of pointer location
+ *
+ * Signals that the pointer has entered the widget.
+ */
+ signals[ENTER] =
+ g_signal_new (I_("enter"),
+ GTK_TYPE_DROP_CONTROLLER_MOTION,
+ G_SIGNAL_RUN_LAST,
+ 0, NULL, NULL,
+ NULL,
+ G_TYPE_NONE, 2,
+ G_TYPE_DOUBLE,
+ G_TYPE_DOUBLE);
+ g_signal_set_va_marshaller (signals[ENTER],
+ G_TYPE_FROM_CLASS (klass),
+ _gtk_marshal_VOID__DOUBLE_DOUBLEv);
+
+ /**
+ * GtkDropControllerMotion::leave:
+ * @self: the object which received the signal
+ *
+ * Signals that the pointer has left the widget.
+ */
+ signals[LEAVE] =
+ g_signal_new (I_("leave"),
+ GTK_TYPE_DROP_CONTROLLER_MOTION,
+ G_SIGNAL_RUN_LAST,
+ 0, NULL, NULL,
+ NULL,
+ G_TYPE_NONE, 0);
+
+ /**
+ * GtkDropControllerMotion::motion:
+ * @self: The object that received the signal
+ * @x: the x coordinate
+ * @y: the y coordinate
+ *
+ * Emitted when the pointer moves inside the widget.
+ */
+ signals[MOTION] =
+ g_signal_new (I_("motion"),
+ GTK_TYPE_DROP_CONTROLLER_MOTION,
+ G_SIGNAL_RUN_FIRST,
+ 0, NULL, NULL,
+ _gtk_marshal_VOID__DOUBLE_DOUBLE,
+ G_TYPE_NONE, 2,
+ G_TYPE_DOUBLE,
+ G_TYPE_DOUBLE);
+ g_signal_set_va_marshaller (signals[MOTION],
+ G_TYPE_FROM_CLASS (klass),
+ _gtk_marshal_VOID__DOUBLE_DOUBLEv);
+}
+
+static void
+gtk_drop_controller_motion_init (GtkDropControllerMotion *self)
+{
+}
+
+/**
+ * gtk_drop_controller_motion_new:
+ *
+ * Creates a new event controller that will handle pointer motion
+ * events during drag and drop.
+ *
+ * Returns: a new #GtkDropControllerMotion
+ **/
+GtkEventController *
+gtk_drop_controller_motion_new (void)
+{
+ return g_object_new (GTK_TYPE_DROP_CONTROLLER_MOTION,
+ NULL);
+}
+
+/**
+ * gtk_drop_controller_motion_contains_pointer:
+ * @self: a #GtkDropControllerMotion
+ *
+ * Returns the value of the GtkDropControllerMotion:contains-pointer property.
+ *
+ * Returns: %TRUE if a dragging pointer is within @self or one of its children.
+ */
+gboolean
+gtk_drop_controller_motion_contains_pointer (GtkDropControllerMotion *self)
+{
+ g_return_val_if_fail (GTK_IS_DROP_CONTROLLER_MOTION (self), FALSE);
+
+ return self->contains_pointer;
+}
+
+/**
+ * gtk_drop_controller_motion_get_drop:
+ * @self: a #GtkDropControllerMotion
+ *
+ * Returns the value of the GtkDropControllerMotion:drop property.
+ *
+ * Returns: The #GdkDrop currently happening within @self or %NULL if none
+ */
+GdkDrop *
+gtk_drop_controller_motion_get_drop (GtkDropControllerMotion *self)
+{
+ g_return_val_if_fail (GTK_IS_DROP_CONTROLLER_MOTION (self), FALSE);
+
+ return self->drop;
+}
+
+/**
+ * gtk_drop_controller_motion_is_pointer:
+ * @self: a #GtkEventControllerKey
+ *
+ * Returns the value of the GtkDropControllerMotion:is-pointer property.
+ *
+ * Returns: %TRUE if a dragging pointer is within @self but not one of its children
+ */
+gboolean
+gtk_drop_controller_motion_is_pointer (GtkDropControllerMotion *self)
+{
+ g_return_val_if_fail (GTK_IS_DROP_CONTROLLER_MOTION (self), FALSE);
+
+ return self->is_pointer;
+}