summaryrefslogtreecommitdiff
path: root/gtk/gtktext.c
diff options
context:
space:
mode:
authorBenjamin Otte <otte@redhat.com>2020-02-29 03:47:17 +0100
committerBenjamin Otte <otte@redhat.com>2020-03-02 03:18:55 +0100
commita411959c910fbf86754b38ffcd7b9615d7788a74 (patch)
tree86e43f40dc05fa41108d667dfbe5f82f3fb47c02 /gtk/gtktext.c
parentf4ac74795c41b2901b5b204aa76ec77010e59f95 (diff)
downloadgtk+-a411959c910fbf86754b38ffcd7b9615d7788a74.tar.gz
droptarget: Redo
This is a huge reorganization of GtkDropTarget. I did not know how to split this up, so it's unfortunately all one commit. Highlights: - Split GtkDropTarget into GtkDropTarget and GtkDropTargetAsync GtkDropTarget is the simple one that only works with GTypes and offers a synchronous interface. GtkDropTargetAsync retains the full old functionality and allows handling mime types. - Drop events are handled differently Instead of picking a single drop target and sending all DND events to it, every event is sent to every drop target. The first one to handle the event gets to call gdk_drop_status(), further handlers do not interact with the GdkDrop. Of course, for the ultimate GDK_DROP_STARTING event, only the first one to accept the drop gets to handle it. This allows stacking DND event controllers that aren't necessarily interested in handling the event or that might decide later to drop it. - Port all widgets to either of those Both have a somewhat changed API due to the new event handling. For the ones who should use the sync version, lots of cleanup was involved to operate on a sync API.
Diffstat (limited to 'gtk/gtktext.c')
-rw-r--r--gtk/gtktext.c183
1 files changed, 66 insertions, 117 deletions
diff --git a/gtk/gtktext.c b/gtk/gtktext.c
index 56f748d312..52af78a211 100644
--- a/gtk/gtktext.c
+++ b/gtk/gtktext.c
@@ -30,6 +30,9 @@
#include "gtkbutton.h"
#include "gtkcssnodeprivate.h"
#include "gtkdebug.h"
+#include "gtkdragicon.h"
+#include "gtkdragsource.h"
+#include "gtkdroptarget.h"
#include "gtkeditable.h"
#include "gtkemojichooser.h"
#include "gtkemojicompletion.h"
@@ -65,9 +68,6 @@
#include "gtkwindow.h"
#include "gtknative.h"
#include "gtkactionmuxerprivate.h"
-#include "gtkdragsource.h"
-#include "gtkdragdest.h"
-#include "gtkdragicon.h"
#include "a11y/gtktextaccessible.h"
@@ -190,7 +190,6 @@ struct _GtkTextPrivate
int dnd_position; /* In chars, -1 == no DND cursor */
int drag_start_x;
int drag_start_y;
- int drop_position; /* where the drop should happen */
int insert_pos;
int selection_bound;
int scroll_offset;
@@ -336,20 +335,18 @@ static void gtk_text_state_flags_changed (GtkWidget *widget,
static void gtk_text_root (GtkWidget *widget);
static gboolean gtk_text_drag_drop (GtkDropTarget *dest,
- GdkDrop *drop,
- int x,
- int y,
+ const GValue *value,
+ double x,
+ double y,
GtkText *text);
static gboolean gtk_text_drag_accept (GtkDropTarget *dest,
GdkDrop *drop,
GtkText *self);
-static void gtk_text_drag_motion (GtkDropTarget *dest,
- GdkDrop *drop,
- int x,
- int y,
+static GdkDragAction gtk_text_drag_motion (GtkDropTarget *dest,
+ double x,
+ double y,
GtkText *text);
static void gtk_text_drag_leave (GtkDropTarget *dest,
- GdkDrop *drop,
GtkText *text);
@@ -1712,7 +1709,7 @@ gtk_text_init (GtkText *self)
GtkGesture *gesture;
GtkEventController *controller;
int i;
- GtkDropTarget *dest;
+ GtkDropTarget *target;
gtk_widget_set_can_focus (GTK_WIDGET (self), TRUE);
gtk_widget_set_overflow (GTK_WIDGET (self), GTK_OVERFLOW_HIDDEN);
@@ -1734,13 +1731,13 @@ gtk_text_init (GtkText *self)
priv->selection_content = g_object_new (GTK_TYPE_TEXT_CONTENT, NULL);
GTK_TEXT_CONTENT (priv->selection_content)->self = self;
- dest = gtk_drop_target_new (gdk_content_formats_new_for_gtype (G_TYPE_STRING),
- GDK_ACTION_COPY | GDK_ACTION_MOVE);
- g_signal_connect (dest, "accept", G_CALLBACK (gtk_text_drag_accept), self);
- g_signal_connect (dest, "drag-motion", G_CALLBACK (gtk_text_drag_motion), self);
- g_signal_connect (dest, "drag-leave", G_CALLBACK (gtk_text_drag_leave), self);
- g_signal_connect (dest, "drag-drop", G_CALLBACK (gtk_text_drag_drop), self);
- gtk_widget_add_controller (GTK_WIDGET (self), GTK_EVENT_CONTROLLER (dest));
+ target = gtk_drop_target_new (G_TYPE_STRING, GDK_ACTION_COPY | GDK_ACTION_MOVE);
+ g_signal_connect (target, "accept", G_CALLBACK (gtk_text_drag_accept), self);
+ g_signal_connect (target, "enter", G_CALLBACK (gtk_text_drag_motion), self);
+ g_signal_connect (target, "motion", G_CALLBACK (gtk_text_drag_motion), self);
+ g_signal_connect (target, "leave", G_CALLBACK (gtk_text_drag_leave), self);
+ g_signal_connect (target, "drop", G_CALLBACK (gtk_text_drag_drop), self);
+ gtk_widget_add_controller (GTK_WIDGET (self), GTK_EVENT_CONTROLLER (target));
/* This object is completely private. No external entity can gain a reference
* to it; so we create it here and destroy it in finalize().
@@ -6120,8 +6117,7 @@ gtk_text_selection_bubble_popup_set (GtkText *self)
static void
gtk_text_drag_leave (GtkDropTarget *dest,
- GdkDrop *drop,
- GtkText *self)
+ GtkText *self)
{
GtkTextPrivate *priv = gtk_text_get_instance_private (self);
GtkWidget *widget = GTK_WIDGET (self);
@@ -6130,94 +6126,45 @@ gtk_text_drag_leave (GtkDropTarget *dest,
gtk_widget_queue_draw (widget);
}
-static GdkDragAction
-gtk_text_get_action (GtkText *self,
- GdkDrop *drop)
+static gboolean
+gtk_text_drag_drop (GtkDropTarget *dest,
+ const GValue *value,
+ double x,
+ double y,
+ GtkText *self)
{
GtkTextPrivate *priv = gtk_text_get_instance_private (self);
- GdkDrag *drag = gdk_drop_get_drag (drop);
- GdkDragAction actions;
-
- actions = gdk_drop_get_actions (drop);
-
- if (drag == priv->drag &&
- actions & GDK_ACTION_MOVE)
- return GDK_ACTION_MOVE;
-
- if (actions & GDK_ACTION_COPY)
- return GDK_ACTION_COPY;
-
- if (actions & GDK_ACTION_MOVE)
- return GDK_ACTION_MOVE;
+ int drop_position;
+ int length;
+ const char *str;
- return 0;
-}
+ if (!priv->editable)
+ return FALSE;
-static void
-got_text (GObject *source,
- GAsyncResult *result,
- gpointer data)
-{
- GdkDrop *drop = GDK_DROP (source);
- GtkText *self = GTK_TEXT (data);
- GtkTextPrivate *priv = gtk_text_get_instance_private (self);
- char *str;
- GdkDragAction action;
+ drop_position = gtk_text_find_position (self, x + priv->scroll_offset);
- str = gdk_drop_read_text_finish (drop, result, NULL);
- action = gtk_text_get_action (self, drop);
+ str = g_value_get_string (value);
+ if (priv->truncate_multiline)
+ length = truncate_multiline (str);
+ else
+ length = -1;
- if (action && str)
+ if (priv->selection_bound == priv->current_pos ||
+ drop_position < priv->selection_bound ||
+ drop_position > priv->current_pos)
{
- int length = -1;
- int pos;
-
- if (priv->truncate_multiline)
- length = truncate_multiline (str);
-
- if (priv->selection_bound == priv->current_pos ||
- priv->drop_position < priv->selection_bound ||
- priv->drop_position > priv->current_pos)
- {
- gtk_text_insert_text (self, str, length, &priv->drop_position);
- }
- else
- {
- /* Replacing selection */
- begin_change (self);
- gtk_text_delete_selection (self);
- pos = MIN (priv->selection_bound, priv->current_pos);
- gtk_text_insert_text (self, str, length, &pos);
- end_change (self);
- }
-
- gdk_drop_finish (drop, action);
+ gtk_text_insert_text (self, str, length, &drop_position);
}
else
{
- /* Drag and drop didn't happen! */
- gdk_drop_finish (drop, 0);
- }
-
- g_free (str);
-}
-
-static gboolean
-gtk_text_drag_drop (GtkDropTarget *dest,
- GdkDrop *drop,
- int x,
- int y,
- GtkText *self)
-{
- GtkTextPrivate *priv = gtk_text_get_instance_private (self);
-
- if (priv->editable && gdk_drop_has_value (drop, G_TYPE_STRING))
- {
- priv->drop_position = gtk_text_find_position (self, x + priv->scroll_offset);
- gdk_drop_read_text_async (drop, NULL, got_text, self);
+ int pos;
+ /* Replacing selection */
+ begin_change (self);
+ gtk_text_delete_selection (self);
+ pos = MIN (priv->selection_bound, priv->current_pos);
+ gtk_text_insert_text (self, str, length, &pos);
+ end_change (self);
}
- else
- gdk_drop_finish (drop, 0);
return TRUE;
}
@@ -6238,40 +6185,42 @@ gtk_text_drag_accept (GtkDropTarget *dest,
return gdk_content_formats_match (gtk_drop_target_get_formats (dest), gdk_drop_get_formats (drop));
}
-static void
-gtk_text_drag_motion (GtkDropTarget *dest,
- GdkDrop *drop,
- int x,
- int y,
- GtkText *self)
+static GdkDragAction
+gtk_text_drag_motion (GtkDropTarget *target,
+ double x,
+ double y,
+ GtkText *self)
{
GtkTextPrivate *priv = gtk_text_get_instance_private (self);
int new_position, old_position;
+ if (!priv->editable)
+ {
+ gtk_drop_target_reject (target);
+ return 0;
+ }
+
old_position = priv->dnd_position;
new_position = gtk_text_find_position (self, x + priv->scroll_offset);
- if (priv->editable)
+ if (priv->selection_bound == priv->current_pos ||
+ new_position < priv->selection_bound ||
+ new_position > priv->current_pos)
{
- if (priv->selection_bound == priv->current_pos ||
- new_position < priv->selection_bound ||
- new_position > priv->current_pos)
- {
- priv->dnd_position = new_position;
- }
- else
- {
- priv->dnd_position = -1;
- }
+ priv->dnd_position = new_position;
}
else
{
- /* Entry not editable, or no text */
priv->dnd_position = -1;
}
if (priv->dnd_position != old_position)
gtk_widget_queue_draw (GTK_WIDGET (self));
+
+ if (priv->drag)
+ return GDK_ACTION_MOVE;
+ else
+ return GDK_ACTION_COPY;
}
/* We display the cursor when