[gtk/listview-dnd: 10/15] treeexpander: Auto-expand during DND



commit 9555e611e108d5edb7735e3a11d49eef2f247b60
Author: Matthias Clasen <mclasen redhat com>
Date:   Fri Jun 19 14:15:05 2020 -0400

    treeexpander: Auto-expand during DND
    
    When hovering over a tree expander during DND,
    expand the tree after a timeout. This matches
    the behavior of GtkTreeView and GtkExpander.

 gtk/gtktreeexpander.c | 59 +++++++++++++++++++++++++++++++++++++++++++++++++++
 1 file changed, 59 insertions(+)
---
diff --git a/gtk/gtktreeexpander.c b/gtk/gtktreeexpander.c
index ec4484340b..0dd369e2ff 100644
--- a/gtk/gtktreeexpander.c
+++ b/gtk/gtktreeexpander.c
@@ -23,6 +23,7 @@
 
 #include "gtkboxlayout.h"
 #include "gtkbuiltiniconprivate.h"
+#include "gtkdropcontrollermotion.h"
 #include "gtkgestureclick.h"
 #include "gtkintl.h"
 #include "gtktreelistmodel.h"
@@ -75,6 +76,8 @@ struct _GtkTreeExpander
 
   GtkWidget *expander;
   guint notify_handler;
+
+  guint expand_timer;
 };
 
 enum
@@ -295,6 +298,12 @@ gtk_tree_expander_dispose (GObject *object)
 {
   GtkTreeExpander *self = GTK_TREE_EXPANDER (object);
 
+  if (self->expand_timer)
+    {
+      g_source_remove (self->expand_timer);
+      self->expand_timer = 0;
+    }
+
   gtk_tree_expander_clear_list_row (self);
   gtk_tree_expander_update_for_list_row (self);
 
@@ -557,10 +566,60 @@ gtk_tree_expander_class_init (GtkTreeExpanderClass *klass)
   gtk_widget_class_set_css_name (widget_class, I_("treeexpander"));
 }
 
+static gboolean
+gtk_tree_expander_expand_timeout (gpointer data)
+{
+  GtkTreeExpander *self = GTK_TREE_EXPANDER (data);
+
+  if (self->list_row != NULL)
+    gtk_tree_list_row_set_expanded (self->list_row, TRUE);
+
+  self->expand_timer = 0;
+
+  return G_SOURCE_REMOVE;
+}
+
+#define TIMEOUT_EXPAND 500
+
+static void
+gtk_tree_expander_drag_enter (GtkDropControllerMotion *motion,
+                              double                   x,
+                              double                   y,
+                              GtkTreeExpander         *self)
+{
+  if (self->list_row == NULL)
+    return;
+
+  if (!gtk_tree_list_row_get_expanded (self->list_row) &&
+      !self->expand_timer)
+    {
+      self->expand_timer = g_timeout_add (TIMEOUT_EXPAND, (GSourceFunc) gtk_tree_expander_expand_timeout, 
self);
+      g_source_set_name_by_id (self->expand_timer, "[gtk] gtk_tree_expander_expand_timeout");
+    }
+}
+
+static void
+gtk_tree_expander_drag_leave (GtkDropControllerMotion *motion,
+                              GtkTreeExpander         *self)
+{
+  if (self->expand_timer)
+    {
+      g_source_remove (self->expand_timer);
+      self->expand_timer = 0;
+    }
+}
+
 static void
 gtk_tree_expander_init (GtkTreeExpander *self)
 {
+  GtkEventController *controller;
+
   gtk_widget_set_can_focus (GTK_WIDGET (self), TRUE);
+
+  controller = gtk_drop_controller_motion_new ();
+  g_signal_connect (controller, "enter", G_CALLBACK (gtk_tree_expander_drag_enter), self);
+  g_signal_connect (controller, "leave", G_CALLBACK (gtk_tree_expander_drag_leave), self);
+  gtk_widget_add_controller (GTK_WIDGET (self), controller);
 }
 
 /**


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