gesture: Replace gtk_gesture_attach/detach with event controller API

Event controllers now auto-attach, and the GtkCapturePhase only determines
when are events dispatched, but all controllers are managed by the widget wrt
grabs.

All callers have been updated.
This commit is contained in:
Carlos Garnacho 2014-05-26 14:02:30 +02:00
parent ad6606e8af
commit d05a9f9a7b
22 changed files with 209 additions and 163 deletions

View File

@ -156,8 +156,9 @@ do_gestures (GtkWidget *do_widget)
gesture = gtk_gesture_swipe_new (drawing_area); gesture = gtk_gesture_swipe_new (drawing_area);
g_signal_connect (gesture, "swipe", g_signal_connect (gesture, "swipe",
G_CALLBACK (swipe_gesture_swept), drawing_area); G_CALLBACK (swipe_gesture_swept), drawing_area);
gtk_gesture_attach (gesture, GTK_PHASE_BUBBLE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture),
g_object_unref (gesture); GTK_PHASE_BUBBLE);
g_object_weak_ref (G_OBJECT (drawing_area), (GWeakNotify) g_object_unref, gesture);
/* Long press */ /* Long press */
gesture = gtk_gesture_long_press_new (drawing_area); gesture = gtk_gesture_long_press_new (drawing_area);
@ -165,22 +166,25 @@ do_gestures (GtkWidget *do_widget)
G_CALLBACK (long_press_gesture_pressed), drawing_area); G_CALLBACK (long_press_gesture_pressed), drawing_area);
g_signal_connect (gesture, "end", g_signal_connect (gesture, "end",
G_CALLBACK (long_press_gesture_end), drawing_area); G_CALLBACK (long_press_gesture_end), drawing_area);
gtk_gesture_attach (gesture, GTK_PHASE_BUBBLE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture),
g_object_unref (gesture); GTK_PHASE_BUBBLE);
g_object_weak_ref (G_OBJECT (drawing_area), (GWeakNotify) g_object_unref, gesture);
/* Rotate */ /* Rotate */
rotate = gesture = gtk_gesture_rotate_new (drawing_area); rotate = gesture = gtk_gesture_rotate_new (drawing_area);
g_signal_connect (gesture, "angle-changed", g_signal_connect (gesture, "angle-changed",
G_CALLBACK (rotation_angle_changed), drawing_area); G_CALLBACK (rotation_angle_changed), drawing_area);
gtk_gesture_attach (gesture, GTK_PHASE_BUBBLE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture),
g_object_unref (gesture); GTK_PHASE_BUBBLE);
g_object_weak_ref (G_OBJECT (drawing_area), (GWeakNotify) g_object_unref, gesture);
/* Zoom */ /* Zoom */
zoom = gesture = gtk_gesture_zoom_new (drawing_area); zoom = gesture = gtk_gesture_zoom_new (drawing_area);
g_signal_connect (gesture, "scale-changed", g_signal_connect (gesture, "scale-changed",
G_CALLBACK (zoom_scale_changed), drawing_area); G_CALLBACK (zoom_scale_changed), drawing_area);
gtk_gesture_attach (gesture, GTK_PHASE_BUBBLE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture),
g_object_unref (gesture); GTK_PHASE_BUBBLE);
g_object_weak_ref (G_OBJECT (drawing_area), (GWeakNotify) g_object_unref, gesture);
} }
if (!gtk_widget_get_visible (window)) if (!gtk_widget_get_visible (window))

View File

@ -7898,6 +7898,9 @@ gtk_popover_get_modal
<FILE>gtkeventcontroller</FILE> <FILE>gtkeventcontroller</FILE>
<TITLE>GtkEventController</TITLE> <TITLE>GtkEventController</TITLE>
GtkEventController GtkEventController
GtkPropagationPhase
gtk_event_controller_get_propagation_phase
gtk_event_controller_set_propagation_phase
gtk_event_controller_handle_event gtk_event_controller_handle_event
gtk_event_controller_get_widget gtk_event_controller_get_widget
gtk_event_controller_reset gtk_event_controller_reset
@ -7929,9 +7932,6 @@ gtk_gesture_set_sequence_state
gtk_gesture_set_state gtk_gesture_set_state
gtk_gesture_get_sequences gtk_gesture_get_sequences
gtk_gesture_handles_sequence gtk_gesture_handles_sequence
GtkPropagationPhase
gtk_gesture_attach
gtk_gesture_detach
<SUBSECTION> <SUBSECTION>
gtk_gesture_get_last_updated_sequence gtk_gesture_get_last_updated_sequence

View File

@ -653,7 +653,7 @@ gtk_button_init (GtkButton *button)
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (priv->gesture), GDK_BUTTON_PRIMARY); gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (priv->gesture), GDK_BUTTON_PRIMARY);
g_signal_connect (priv->gesture, "pressed", G_CALLBACK (multipress_pressed_cb), button); g_signal_connect (priv->gesture, "pressed", G_CALLBACK (multipress_pressed_cb), button);
g_signal_connect (priv->gesture, "released", G_CALLBACK (multipress_released_cb), button); g_signal_connect (priv->gesture, "released", G_CALLBACK (multipress_released_cb), button);
gtk_gesture_attach (priv->gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->gesture), GTK_PHASE_TARGET);
} }
static void static void
@ -668,12 +668,7 @@ gtk_button_destroy (GtkWidget *widget)
priv->label_text = NULL; priv->label_text = NULL;
} }
if (priv->gesture) g_clear_object (&priv->gesture);
{
gtk_gesture_detach (priv->gesture);
g_object_unref (priv->gesture);
priv->gesture = NULL;
}
GTK_WIDGET_CLASS (gtk_button_parent_class)->destroy (widget); GTK_WIDGET_CLASS (gtk_button_parent_class)->destroy (widget);
} }

View File

@ -409,14 +409,16 @@ gtk_color_plane_init (GtkColorPlane *plane)
G_CALLBACK (plane_drag_gesture_update), plane); G_CALLBACK (plane_drag_gesture_update), plane);
g_signal_connect (plane->priv->drag_gesture, "drag-end", g_signal_connect (plane->priv->drag_gesture, "drag-end",
G_CALLBACK (plane_drag_gesture_end), plane); G_CALLBACK (plane_drag_gesture_end), plane);
gtk_gesture_attach (plane->priv->drag_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (plane->priv->drag_gesture),
GTK_PHASE_TARGET);
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (plane->priv->drag_gesture), gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (plane->priv->drag_gesture),
FALSE); FALSE);
plane->priv->long_press_gesture = gtk_gesture_long_press_new (GTK_WIDGET (plane)); plane->priv->long_press_gesture = gtk_gesture_long_press_new (GTK_WIDGET (plane));
g_signal_connect (plane->priv->long_press_gesture, "pressed", g_signal_connect (plane->priv->long_press_gesture, "pressed",
G_CALLBACK (hold_action), plane); G_CALLBACK (hold_action), plane);
gtk_gesture_attach (plane->priv->long_press_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (plane->priv->long_press_gesture),
GTK_PHASE_TARGET);
} }
static void static void
@ -431,10 +433,7 @@ plane_finalize (GObject *object)
g_clear_object (&plane->priv->s_adj); g_clear_object (&plane->priv->s_adj);
g_clear_object (&plane->priv->v_adj); g_clear_object (&plane->priv->v_adj);
gtk_gesture_detach (plane->priv->drag_gesture);
g_clear_object (&plane->priv->drag_gesture); g_clear_object (&plane->priv->drag_gesture);
gtk_gesture_detach (plane->priv->long_press_gesture);
g_clear_object (&plane->priv->long_press_gesture); g_clear_object (&plane->priv->long_press_gesture);
G_OBJECT_CLASS (gtk_color_plane_parent_class)->finalize (object); G_OBJECT_CLASS (gtk_color_plane_parent_class)->finalize (object);

View File

@ -258,7 +258,8 @@ gtk_color_scale_init (GtkColorScale *scale)
scale->priv->long_press_gesture = gtk_gesture_long_press_new (GTK_WIDGET (scale)); scale->priv->long_press_gesture = gtk_gesture_long_press_new (GTK_WIDGET (scale));
g_signal_connect (scale->priv->long_press_gesture, "pressed", g_signal_connect (scale->priv->long_press_gesture, "pressed",
G_CALLBACK (hold_action), scale); G_CALLBACK (hold_action), scale);
gtk_gesture_attach (scale->priv->long_press_gesture, GTK_PHASE_BUBBLE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (scale->priv->long_press_gesture),
GTK_PHASE_BUBBLE);
} }
static void static void
@ -269,7 +270,6 @@ scale_finalize (GObject *object)
if (scale->priv->surface) if (scale->priv->surface)
cairo_surface_destroy (scale->priv->surface); cairo_surface_destroy (scale->priv->surface);
gtk_gesture_detach (scale->priv->long_press_gesture);
g_clear_object (&scale->priv->long_press_gesture); g_clear_object (&scale->priv->long_press_gesture);
G_OBJECT_CLASS (gtk_color_scale_parent_class)->finalize (object); G_OBJECT_CLASS (gtk_color_scale_parent_class)->finalize (object);

View File

@ -90,13 +90,15 @@ gtk_color_swatch_init (GtkColorSwatch *swatch)
swatch->priv->long_press_gesture = gtk_gesture_long_press_new (GTK_WIDGET (swatch)); swatch->priv->long_press_gesture = gtk_gesture_long_press_new (GTK_WIDGET (swatch));
g_signal_connect (swatch->priv->long_press_gesture, "pressed", g_signal_connect (swatch->priv->long_press_gesture, "pressed",
G_CALLBACK (hold_action), swatch); G_CALLBACK (hold_action), swatch);
gtk_gesture_attach (swatch->priv->long_press_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (swatch->priv->long_press_gesture),
GTK_PHASE_TARGET);
swatch->priv->multipress_gesture = gtk_gesture_multi_press_new (GTK_WIDGET (swatch)); swatch->priv->multipress_gesture = gtk_gesture_multi_press_new (GTK_WIDGET (swatch));
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (swatch->priv->multipress_gesture), FALSE); gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (swatch->priv->multipress_gesture), FALSE);
g_signal_connect (swatch->priv->multipress_gesture, "pressed", g_signal_connect (swatch->priv->multipress_gesture, "pressed",
G_CALLBACK (tap_action), swatch); G_CALLBACK (tap_action), swatch);
gtk_gesture_attach (swatch->priv->multipress_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (swatch->priv->multipress_gesture),
GTK_PHASE_TARGET);
} }
#define INTENSITY(r, g, b) ((r) * 0.30 + (g) * 0.59 + (b) * 0.11) #define INTENSITY(r, g, b) ((r) * 0.30 + (g) * 0.59 + (b) * 0.11)
@ -694,10 +696,7 @@ swatch_finalize (GObject *object)
g_free (swatch->priv->icon); g_free (swatch->priv->icon);
gtk_gesture_detach (swatch->priv->long_press_gesture);
g_object_unref (swatch->priv->long_press_gesture); g_object_unref (swatch->priv->long_press_gesture);
gtk_gesture_detach (swatch->priv->multipress_gesture);
g_object_unref (swatch->priv->multipress_gesture); g_object_unref (swatch->priv->multipress_gesture);
G_OBJECT_CLASS (gtk_color_swatch_parent_class)->finalize (object); G_OBJECT_CLASS (gtk_color_swatch_parent_class)->finalize (object);

View File

@ -2801,7 +2801,6 @@ gtk_drag_source_set (GtkWidget *widget,
site->drag_gesture = gtk_gesture_drag_new (widget); site->drag_gesture = gtk_gesture_drag_new (widget);
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (site->drag_gesture), gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (site->drag_gesture),
FALSE); FALSE);
gtk_gesture_attach (site->drag_gesture, GTK_PHASE_NONE);
g_signal_connect (widget, "button-press-event", g_signal_connect (widget, "button-press-event",
G_CALLBACK (gtk_drag_source_event_cb), G_CALLBACK (gtk_drag_source_event_cb),
@ -2844,7 +2843,6 @@ gtk_drag_source_unset (GtkWidget *widget)
g_signal_handlers_disconnect_by_func (widget, g_signal_handlers_disconnect_by_func (widget,
gtk_drag_source_event_cb, gtk_drag_source_event_cb,
site); site);
gtk_gesture_detach (site->drag_gesture);
g_object_set_data (G_OBJECT (widget), I_("gtk-site-data"), NULL); g_object_set_data (G_OBJECT (widget), I_("gtk-site-data"), NULL);
} }
} }

View File

@ -2701,14 +2701,16 @@ gtk_entry_init (GtkEntry *entry)
G_CALLBACK (gtk_entry_drag_gesture_end), entry); G_CALLBACK (gtk_entry_drag_gesture_end), entry);
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->drag_gesture), FALSE); gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->drag_gesture), FALSE);
gtk_gesture_single_set_exclusive (GTK_GESTURE_SINGLE (priv->drag_gesture), TRUE); gtk_gesture_single_set_exclusive (GTK_GESTURE_SINGLE (priv->drag_gesture), TRUE);
gtk_gesture_attach (priv->drag_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->drag_gesture),
GTK_PHASE_TARGET);
priv->multipress_gesture = gtk_gesture_multi_press_new (GTK_WIDGET (entry)); priv->multipress_gesture = gtk_gesture_multi_press_new (GTK_WIDGET (entry));
g_signal_connect (priv->multipress_gesture, "pressed", g_signal_connect (priv->multipress_gesture, "pressed",
G_CALLBACK (gtk_entry_multipress_gesture_pressed), entry); G_CALLBACK (gtk_entry_multipress_gesture_pressed), entry);
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->multipress_gesture), FALSE); gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->multipress_gesture), FALSE);
gtk_gesture_single_set_exclusive (GTK_GESTURE_SINGLE (priv->multipress_gesture), TRUE); gtk_gesture_single_set_exclusive (GTK_GESTURE_SINGLE (priv->multipress_gesture), TRUE);
gtk_gesture_attach (priv->multipress_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->multipress_gesture),
GTK_PHASE_TARGET);
} }
static void static void
@ -2988,6 +2990,9 @@ gtk_entry_finalize (GObject *object)
g_free (priv->placeholder_text); g_free (priv->placeholder_text);
g_free (priv->im_module); g_free (priv->im_module);
g_clear_object (&priv->drag_gesture);
g_clear_object (&priv->multipress_gesture);
if (priv->tabs) if (priv->tabs)
pango_tab_array_free (priv->tabs); pango_tab_array_free (priv->tabs);

View File

@ -32,6 +32,7 @@
#include "config.h" #include "config.h"
#include "gtkeventcontroller.h" #include "gtkeventcontroller.h"
#include "gtkeventcontrollerprivate.h" #include "gtkeventcontrollerprivate.h"
#include "gtkwidgetprivate.h"
#include "gtktypebuiltins.h" #include "gtktypebuiltins.h"
#include "gtkmarshalers.h" #include "gtkmarshalers.h"
#include "gtkprivate.h" #include "gtkprivate.h"
@ -40,7 +41,8 @@
typedef struct _GtkEventControllerPrivate GtkEventControllerPrivate; typedef struct _GtkEventControllerPrivate GtkEventControllerPrivate;
enum { enum {
PROP_WIDGET = 1 PROP_WIDGET = 1,
PROP_PROPAGATION_PHASE
}; };
enum { enum {
@ -53,6 +55,7 @@ struct _GtkEventControllerPrivate
{ {
GtkWidget *widget; GtkWidget *widget;
guint evmask; guint evmask;
GtkPropagationPhase phase;
}; };
guint signals[N_SIGNALS] = { 0 }; guint signals[N_SIGNALS] = { 0 };
@ -81,6 +84,10 @@ gtk_event_controller_set_property (GObject *object,
case PROP_WIDGET: case PROP_WIDGET:
priv->widget = g_value_get_object (value); priv->widget = g_value_get_object (value);
break; break;
case PROP_PROPAGATION_PHASE:
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (object),
g_value_get_enum (value));
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
} }
@ -101,11 +108,34 @@ gtk_event_controller_get_property (GObject *object,
case PROP_WIDGET: case PROP_WIDGET:
g_value_set_object (value, priv->widget); g_value_set_object (value, priv->widget);
break; break;
case PROP_PROPAGATION_PHASE:
g_value_set_enum (value, priv->phase);
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
} }
} }
static void
gtk_event_controller_constructed (GObject *object)
{
GtkEventController *controller = GTK_EVENT_CONTROLLER (object);
GtkEventControllerPrivate *priv;
priv = gtk_event_controller_get_instance_private (controller);
g_assert (priv->widget != NULL);
_gtk_widget_add_controller (priv->widget, controller);
}
static void
gtk_event_controller_dispose (GObject *object)
{
GtkEventController *controller = GTK_EVENT_CONTROLLER (object);
GtkEventControllerPrivate *priv;
priv = gtk_event_controller_get_instance_private (controller);
_gtk_widget_remove_controller (priv->widget, controller);
}
static void static void
gtk_event_controller_class_init (GtkEventControllerClass *klass) gtk_event_controller_class_init (GtkEventControllerClass *klass)
@ -116,6 +146,8 @@ gtk_event_controller_class_init (GtkEventControllerClass *klass)
object_class->set_property = gtk_event_controller_set_property; object_class->set_property = gtk_event_controller_set_property;
object_class->get_property = gtk_event_controller_get_property; object_class->get_property = gtk_event_controller_get_property;
object_class->constructed = gtk_event_controller_constructed;
object_class->dispose = gtk_event_controller_dispose;
/** /**
* GtkEventController:widget: * GtkEventController:widget:
@ -132,6 +164,21 @@ gtk_event_controller_class_init (GtkEventControllerClass *klass)
GTK_TYPE_WIDGET, GTK_TYPE_WIDGET,
GTK_PARAM_READWRITE | GTK_PARAM_READWRITE |
G_PARAM_CONSTRUCT_ONLY)); G_PARAM_CONSTRUCT_ONLY));
/**
* GtkEventController:propagation-phase:
*
* The propagation phase at which this controller will handle events.
*
* Since: 3.14
*/
g_object_class_install_property (object_class,
PROP_PROPAGATION_PHASE,
g_param_spec_enum ("propagation-phase",
P_("Propagation phase"),
P_("Propagation phase at which this controller is run"),
GTK_TYPE_PROPAGATION_PHASE,
GTK_PHASE_NONE,
GTK_PARAM_READWRITE));
/** /**
* GtkEventController::handle-event: * GtkEventController::handle-event:
* @controller: the object which receives the signal * @controller: the object which receives the signal
@ -269,3 +316,60 @@ gtk_event_controller_reset (GtkEventController *controller)
g_signal_emit (controller, signals[RESET], 0); g_signal_emit (controller, signals[RESET], 0);
} }
/**
* gtk_event_controller_get_propagation_phase:
* @controller: a #GtkEventController
*
* Gets the propagation phase at which @controller handles events.
*
* Returns: the propagation phase
*
* Since: 3.14
**/
GtkPropagationPhase
gtk_event_controller_get_propagation_phase (GtkEventController *controller)
{
GtkEventControllerPrivate *priv;
g_return_val_if_fail (GTK_IS_EVENT_CONTROLLER (controller), GTK_PHASE_NONE);
priv = gtk_event_controller_get_instance_private (controller);
return priv->phase;
}
/**
* gtk_event_controller_set_propagation_phase:
* @controller: a #GtkEventController
* @phase: a propagation phase
*
* Sets the propagation phase at which a controller handles events.
*
* If @phase is %GTK_PHASE_NONE, no automatic event handling will be
* performed, but other additional gesture maintenance will. In that phase,
* the events can be managed by calling gtk_event_controller_handle_event().
*
* Since: 3.14
**/
void
gtk_event_controller_set_propagation_phase (GtkEventController *controller,
GtkPropagationPhase phase)
{
GtkEventControllerPrivate *priv;
g_return_if_fail (GTK_IS_EVENT_CONTROLLER (controller));
g_return_if_fail (phase >= GTK_PHASE_NONE && phase <= GTK_PHASE_TARGET);
priv = gtk_event_controller_get_instance_private (controller);
if (priv->phase == phase)
return;
priv->phase = phase;
if (phase == GTK_PHASE_NONE)
gtk_event_controller_reset (controller);
g_object_notify (G_OBJECT (controller), "propagation-phase");
}

View File

@ -53,6 +53,13 @@ gboolean gtk_event_controller_handle_event (GtkEventController *controller
GDK_AVAILABLE_IN_3_14 GDK_AVAILABLE_IN_3_14
void gtk_event_controller_reset (GtkEventController *controller); void gtk_event_controller_reset (GtkEventController *controller);
GDK_AVAILABLE_IN_3_14
GtkPropagationPhase gtk_event_controller_get_propagation_phase (GtkEventController *controller);
GDK_AVAILABLE_IN_3_14
void gtk_event_controller_set_propagation_phase (GtkEventController *controller,
GtkPropagationPhase phase);
G_END_DECLS G_END_DECLS
#endif /* __GTK_EVENT_CONTROLLER_H__ */ #endif /* __GTK_EVENT_CONTROLLER_H__ */

View File

@ -46,9 +46,9 @@
* *
* ## Event propagation * ## Event propagation
* *
* To receive events, a gesture needs to first be attached to its widget with * In order to receive events, a gesture needs to either set a propagation phase
* gtk_gesture_attach(). The phase passed to this call determines at which * through gtk_event_controller_set_propagation_phase(), or feed those manually
* point in the event processing a gesture operates. * through gtk_event_controller_handle_event().
* *
* In the capture phase, events are propagated from the toplevel down to the * In the capture phase, events are propagated from the toplevel down to the
* target widget, and gestures that are attached to containers above the widget * target widget, and gestures that are attached to containers above the widget
@ -56,17 +56,13 @@
* *
* After the capture phase, GTK+ emits the traditional #GtkWidget::button-press, * After the capture phase, GTK+ emits the traditional #GtkWidget::button-press,
* #GtkWidget::button-release, #GtkWidget::touch-event, etc signals. Gestures * #GtkWidget::button-release, #GtkWidget::touch-event, etc signals. Gestures
* with the target phase are fed events from the default #GtkWidget::event * with the %GTK_PHASE_TARGET phase are fed events from the default #GtkWidget::event
* handlers. * handlers.
* *
* In the bubble phase, events are propagated up from the target widget to the * In the bubble phase, events are propagated up from the target widget to the
* toplevel, and gestures that are attached to containers above the widget get * toplevel, and gestures that are attached to containers above the widget get
* a chance to interact with events that have not been handled yet. * a chance to interact with events that have not been handled yet.
* *
* Gestures attached with the phase 'none' are not receiving any events
* automatically, but events can be passed to them with
* gtk_event_controller_handle_event().
*
* ## States of a sequence # {#touch-sequence-states} * ## States of a sequence # {#touch-sequence-states}
* *
* Whenever input interaction happens, a single event may trigger a cascade of * Whenever input interaction happens, a single event may trigger a cascade of
@ -1547,51 +1543,3 @@ _gtk_gesture_get_pointer_emulating_sequence (GtkGesture *gesture,
return FALSE; return FALSE;
} }
/**
* gtk_gesture_attach:
* @gesture: a #GtkGesture
* @phase: phase at which events are handled
*
* Attaches @gesture to its widget, so @gesture is able to receive
* and manage the events that are emitted on the #GtkWidget. This call
* will also make sure that the gesture state is maintained properly
* whenever input is grabbed elsewhere.
*
* If @phase is %GTK_PHASE_NONE, no automatical event handling will be
* performed, but other additional gesture maintenance will. The events
* can be managed by calling gtk_event_controller_handle_event().
*
* Since: 3.14
**/
void
gtk_gesture_attach (GtkGesture *gesture,
GtkPropagationPhase phase)
{
GtkWidget *widget;
g_return_if_fail (GTK_IS_GESTURE (gesture));
g_return_if_fail (phase >= GTK_PHASE_NONE && phase <= GTK_PHASE_TARGET);
widget = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
_gtk_widget_add_controller (widget, GTK_EVENT_CONTROLLER (gesture), phase);
}
/**
* gtk_gesture_detach:
* @gesture: a #GtkGesture
*
* Detaches @gesture from its widget.
*
* Since: 3.14
**/
void
gtk_gesture_detach (GtkGesture *gesture)
{
GtkWidget *widget;
g_return_if_fail (GTK_IS_GESTURE (gesture));
widget = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
_gtk_widget_remove_controller (widget, GTK_EVENT_CONTROLLER (gesture));
}

View File

@ -109,12 +109,6 @@ GDK_AVAILABLE_IN_3_14
gboolean gtk_gesture_is_grouped_with (GtkGesture *gesture, gboolean gtk_gesture_is_grouped_with (GtkGesture *gesture,
GtkGesture *other); GtkGesture *other);
GDK_AVAILABLE_IN_3_14
void gtk_gesture_attach (GtkGesture *gesture,
GtkPropagationPhase phase);
GDK_AVAILABLE_IN_3_14
void gtk_gesture_detach (GtkGesture *gesture);
G_END_DECLS G_END_DECLS
#endif /* __GTK_GESTURE_H__ */ #endif /* __GTK_GESTURE_H__ */

View File

@ -801,7 +801,8 @@ gtk_paned_init (GtkPaned *paned)
G_CALLBACK (pan_gesture_pan_cb), paned); G_CALLBACK (pan_gesture_pan_cb), paned);
g_signal_connect (gesture, "drag-end", g_signal_connect (gesture, "drag-end",
G_CALLBACK (pan_gesture_drag_end_cb), paned); G_CALLBACK (pan_gesture_drag_end_cb), paned);
gtk_gesture_attach (gesture, GTK_PHASE_CAPTURE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture),
GTK_PHASE_CAPTURE);
priv->pan_gesture = gesture; priv->pan_gesture = gesture;
} }
@ -971,7 +972,6 @@ gtk_paned_finalize (GObject *object)
gtk_paned_set_saved_focus (paned, NULL); gtk_paned_set_saved_focus (paned, NULL);
gtk_paned_set_first_paned (paned, NULL); gtk_paned_set_first_paned (paned, NULL);
gtk_gesture_detach (paned->priv->pan_gesture);
g_clear_object (&paned->priv->pan_gesture); g_clear_object (&paned->priv->pan_gesture);
G_OBJECT_CLASS (gtk_paned_parent_class)->finalize (object); G_OBJECT_CLASS (gtk_paned_parent_class)->finalize (object);

View File

@ -1505,11 +1505,7 @@ gtk_range_destroy (GtkWidget *widget)
gtk_range_remove_step_timer (range); gtk_range_remove_step_timer (range);
if (priv->long_press_gesture)
{
gtk_gesture_detach (priv->long_press_gesture);
g_clear_object (&priv->long_press_gesture); g_clear_object (&priv->long_press_gesture);
}
if (priv->adjustment) if (priv->adjustment)
{ {
@ -2368,10 +2364,7 @@ update_zoom_set (GtkRange *range,
gboolean zoom_set) gboolean zoom_set)
{ {
if (zoom_set) if (zoom_set)
{
gtk_gesture_detach (range->priv->long_press_gesture);
g_clear_object (&range->priv->long_press_gesture); g_clear_object (&range->priv->long_press_gesture);
}
range->priv->zoom_set = zoom_set; range->priv->zoom_set = zoom_set;
} }
@ -2711,7 +2704,6 @@ gtk_range_button_press (GtkWidget *widget,
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->long_press_gesture), gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->long_press_gesture),
FALSE); FALSE);
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (priv->long_press_gesture), 1); gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (priv->long_press_gesture), 1);
gtk_gesture_attach (priv->long_press_gesture, GTK_PHASE_NONE);
g_signal_connect (priv->long_press_gesture, "pressed", g_signal_connect (priv->long_press_gesture, "pressed",
G_CALLBACK (hold_action), range); G_CALLBACK (hold_action), range);

View File

@ -724,6 +724,7 @@ scrolled_window_long_press_cancelled_cb (GtkScrolledWindow *scrolled_window,
static void static void
gtk_scrolled_window_check_attach_pan_gesture (GtkScrolledWindow *sw) gtk_scrolled_window_check_attach_pan_gesture (GtkScrolledWindow *sw)
{ {
GtkPropagationPhase phase = GTK_PHASE_NONE;
GtkScrolledWindowPrivate *priv = sw->priv; GtkScrolledWindowPrivate *priv = sw->priv;
if (priv->kinetic_scrolling && if (priv->kinetic_scrolling &&
@ -739,10 +740,10 @@ gtk_scrolled_window_check_attach_pan_gesture (GtkScrolledWindow *sw)
gtk_gesture_pan_set_orientation (GTK_GESTURE_PAN (priv->pan_gesture), gtk_gesture_pan_set_orientation (GTK_GESTURE_PAN (priv->pan_gesture),
orientation); orientation);
gtk_gesture_attach (priv->pan_gesture, GTK_PHASE_CAPTURE); phase = GTK_PHASE_CAPTURE;
} }
else
gtk_gesture_detach (priv->pan_gesture); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->pan_gesture), phase);
} }
static void static void
@ -1267,6 +1268,7 @@ void
gtk_scrolled_window_set_kinetic_scrolling (GtkScrolledWindow *scrolled_window, gtk_scrolled_window_set_kinetic_scrolling (GtkScrolledWindow *scrolled_window,
gboolean kinetic_scrolling) gboolean kinetic_scrolling)
{ {
GtkPropagationPhase phase = GTK_PHASE_NONE;
GtkScrolledWindowPrivate *priv; GtkScrolledWindowPrivate *priv;
g_return_if_fail (GTK_IS_SCROLLED_WINDOW (scrolled_window)); g_return_if_fail (GTK_IS_SCROLLED_WINDOW (scrolled_window));
@ -1279,19 +1281,15 @@ gtk_scrolled_window_set_kinetic_scrolling (GtkScrolledWindow *scrolled_window,
gtk_scrolled_window_check_attach_pan_gesture (scrolled_window); gtk_scrolled_window_check_attach_pan_gesture (scrolled_window);
if (priv->kinetic_scrolling) if (priv->kinetic_scrolling)
{ phase = GTK_PHASE_CAPTURE;
gtk_gesture_attach (priv->drag_gesture, GTK_PHASE_CAPTURE);
gtk_gesture_attach (priv->swipe_gesture, GTK_PHASE_CAPTURE);
gtk_gesture_attach (priv->long_press_gesture, GTK_PHASE_CAPTURE);
}
else else
{
gtk_gesture_detach (priv->drag_gesture);
gtk_gesture_detach (priv->swipe_gesture);
gtk_gesture_detach (priv->long_press_gesture);
gtk_scrolled_window_cancel_deceleration (scrolled_window); gtk_scrolled_window_cancel_deceleration (scrolled_window);
}
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->drag_gesture), phase);
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->swipe_gesture), phase);
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->long_press_gesture), phase);
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->pan_gesture), phase);
g_object_notify (G_OBJECT (scrolled_window), "kinetic-scrolling"); g_object_notify (G_OBJECT (scrolled_window), "kinetic-scrolling");
} }

View File

@ -710,7 +710,8 @@ gtk_spin_button_init (GtkSpinButton *spin_button)
gtk_widget_add_events (GTK_WIDGET (spin_button), GDK_SCROLL_MASK); gtk_widget_add_events (GTK_WIDGET (spin_button), GDK_SCROLL_MASK);
priv->swipe_gesture = gtk_gesture_swipe_new (GTK_WIDGET (spin_button)); priv->swipe_gesture = gtk_gesture_swipe_new (GTK_WIDGET (spin_button));
gtk_gesture_attach (priv->swipe_gesture, GTK_PHASE_CAPTURE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->swipe_gesture),
GTK_PHASE_CAPTURE);
g_signal_connect (priv->swipe_gesture, "begin", g_signal_connect (priv->swipe_gesture, "begin",
G_CALLBACK (swipe_gesture_begin), spin_button); G_CALLBACK (swipe_gesture_begin), spin_button);
g_signal_connect (priv->swipe_gesture, "update", g_signal_connect (priv->swipe_gesture, "update",
@ -731,7 +732,6 @@ gtk_spin_button_finalize (GObject *object)
if (priv->up_panel_context) if (priv->up_panel_context)
g_object_unref (priv->up_panel_context); g_object_unref (priv->up_panel_context);
gtk_gesture_detach (priv->swipe_gesture);
g_object_unref (priv->swipe_gesture); g_object_unref (priv->swipe_gesture);
G_OBJECT_CLASS (gtk_spin_button_parent_class)->finalize (object); G_OBJECT_CLASS (gtk_spin_button_parent_class)->finalize (object);

View File

@ -749,6 +749,9 @@ gtk_switch_dispose (GObject *object)
priv->action = NULL; priv->action = NULL;
} }
g_clear_object (&priv->pan_gesture);
g_clear_object (&priv->multipress_gesture);
G_OBJECT_CLASS (gtk_switch_parent_class)->dispose (object); G_OBJECT_CLASS (gtk_switch_parent_class)->dispose (object);
} }
@ -924,7 +927,8 @@ gtk_switch_init (GtkSwitch *self)
G_CALLBACK (gtk_switch_multipress_gesture_pressed), self); G_CALLBACK (gtk_switch_multipress_gesture_pressed), self);
g_signal_connect (gesture, "released", g_signal_connect (gesture, "released",
G_CALLBACK (gtk_switch_multipress_gesture_released), self); G_CALLBACK (gtk_switch_multipress_gesture_released), self);
gtk_gesture_attach (gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture),
GTK_PHASE_TARGET);
self->priv->multipress_gesture = gesture; self->priv->multipress_gesture = gesture;
gesture = gtk_gesture_pan_new (GTK_WIDGET (self), gesture = gtk_gesture_pan_new (GTK_WIDGET (self),
@ -935,7 +939,8 @@ gtk_switch_init (GtkSwitch *self)
G_CALLBACK (gtk_switch_pan_gesture_pan), self); G_CALLBACK (gtk_switch_pan_gesture_pan), self);
g_signal_connect (gesture, "drag-end", g_signal_connect (gesture, "drag-end",
G_CALLBACK (gtk_switch_pan_gesture_drag_end), self); G_CALLBACK (gtk_switch_pan_gesture_drag_end), self);
gtk_gesture_attach (gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture),
GTK_PHASE_TARGET);
self->priv->pan_gesture = gesture; self->priv->pan_gesture = gesture;
} }

View File

@ -1555,7 +1555,8 @@ gtk_text_view_init (GtkTextView *text_view)
priv->multipress_gesture = gtk_gesture_multi_press_new (widget); priv->multipress_gesture = gtk_gesture_multi_press_new (widget);
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->multipress_gesture), gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->multipress_gesture),
FALSE); FALSE);
gtk_gesture_attach (priv->multipress_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->multipress_gesture),
GTK_PHASE_TARGET);
g_signal_connect (priv->multipress_gesture, "pressed", g_signal_connect (priv->multipress_gesture, "pressed",
G_CALLBACK (gtk_text_view_multipress_gesture_pressed), G_CALLBACK (gtk_text_view_multipress_gesture_pressed),
widget); widget);
@ -1563,7 +1564,8 @@ gtk_text_view_init (GtkTextView *text_view)
priv->drag_gesture = gtk_gesture_drag_new (widget); priv->drag_gesture = gtk_gesture_drag_new (widget);
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->drag_gesture), gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->drag_gesture),
FALSE); FALSE);
gtk_gesture_attach (priv->drag_gesture, GTK_PHASE_TARGET); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->drag_gesture),
GTK_PHASE_TARGET);
g_signal_connect (priv->drag_gesture, "drag-update", g_signal_connect (priv->drag_gesture, "drag-update",
G_CALLBACK (gtk_text_view_drag_gesture_update), G_CALLBACK (gtk_text_view_drag_gesture_update),
widget); widget);
@ -3258,10 +3260,7 @@ gtk_text_view_finalize (GObject *object)
cancel_pending_scroll (text_view); cancel_pending_scroll (text_view);
gtk_gesture_detach (priv->multipress_gesture);
g_object_unref (priv->multipress_gesture); g_object_unref (priv->multipress_gesture);
gtk_gesture_detach (priv->drag_gesture);
g_object_unref (priv->drag_gesture); g_object_unref (priv->drag_gesture);
if (priv->tabs) if (priv->tabs)

View File

@ -1841,7 +1841,8 @@ gtk_tree_view_init (GtkTreeView *tree_view)
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (tree_view->priv->multipress_gesture), FALSE); gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (tree_view->priv->multipress_gesture), FALSE);
g_signal_connect (tree_view->priv->multipress_gesture, "pressed", g_signal_connect (tree_view->priv->multipress_gesture, "pressed",
G_CALLBACK (_tree_view_multipress_pressed), tree_view); G_CALLBACK (_tree_view_multipress_pressed), tree_view);
gtk_gesture_attach (tree_view->priv->multipress_gesture, GTK_PHASE_BUBBLE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (tree_view->priv->multipress_gesture),
GTK_PHASE_BUBBLE);
} }
@ -2206,11 +2207,7 @@ gtk_tree_view_destroy (GtkWidget *widget)
_gtk_pixel_cache_free (tree_view->priv->pixel_cache); _gtk_pixel_cache_free (tree_view->priv->pixel_cache);
tree_view->priv->pixel_cache = NULL; tree_view->priv->pixel_cache = NULL;
if (tree_view->priv->multipress_gesture)
{
gtk_gesture_detach (tree_view->priv->multipress_gesture);
g_clear_object (&tree_view->priv->multipress_gesture); g_clear_object (&tree_view->priv->multipress_gesture);
}
GTK_WIDGET_CLASS (gtk_tree_view_parent_class)->destroy (widget); GTK_WIDGET_CLASS (gtk_tree_view_parent_class)->destroy (widget);
} }

View File

@ -400,7 +400,6 @@ typedef struct {
typedef struct { typedef struct {
GtkEventController *controller; GtkEventController *controller;
GtkPropagationPhase phase;
guint evmask_notify_id; guint evmask_notify_id;
guint grab_notify_id; guint grab_notify_id;
guint sequence_state_changed_id; guint sequence_state_changed_id;
@ -4206,11 +4205,13 @@ gtk_widget_needs_press_emulation (GtkWidget *widget,
for (l = priv->event_controllers; l; l = l->next) for (l = priv->event_controllers; l; l = l->next)
{ {
EventControllerData *data; EventControllerData *data;
GtkPropagationPhase phase;
GtkGesture *gesture; GtkGesture *gesture;
data = l->data; data = l->data;
phase = gtk_event_controller_get_propagation_phase (data->controller);
if (data->phase != GTK_PHASE_CAPTURE) if (phase != GTK_PHASE_CAPTURE)
continue; continue;
if (!GTK_IS_GESTURE (data->controller)) if (!GTK_IS_GESTURE (data->controller))
continue; continue;
@ -7332,6 +7333,7 @@ _gtk_widget_run_controllers (GtkWidget *widget,
while (l != NULL) while (l != NULL)
{ {
GList *next = l->next; GList *next = l->next;
data = l->data; data = l->data;
if (data->controller == NULL) if (data->controller == NULL)
@ -7339,8 +7341,15 @@ _gtk_widget_run_controllers (GtkWidget *widget,
priv->event_controllers = g_list_delete_link (priv->event_controllers, l); priv->event_controllers = g_list_delete_link (priv->event_controllers, l);
g_free (data); g_free (data);
} }
else if (data->phase == phase) else
{
GtkPropagationPhase controller_phase;
controller_phase = gtk_event_controller_get_propagation_phase (data->controller);
if (controller_phase == phase)
handled |= gtk_event_controller_handle_event (data->controller, event); handled |= gtk_event_controller_handle_event (data->controller, event);
}
l = next; l = next;
} }
@ -16874,10 +16883,12 @@ event_controller_grab_notify (GtkWidget *widget,
EventControllerData *data) EventControllerData *data)
{ {
GtkWidget *grab_widget, *toplevel; GtkWidget *grab_widget, *toplevel;
GtkPropagationPhase phase;
GtkWindowGroup *group; GtkWindowGroup *group;
GdkDevice *device; GdkDevice *device;
device = gtk_gesture_get_device (GTK_GESTURE (data->controller)); device = gtk_gesture_get_device (GTK_GESTURE (data->controller));
phase = gtk_event_controller_get_propagation_phase (data->controller);
if (!device) if (!device)
return; return;
@ -16897,9 +16908,9 @@ event_controller_grab_notify (GtkWidget *widget,
if (!grab_widget || grab_widget == widget) if (!grab_widget || grab_widget == widget)
return; return;
if ((data->phase != GTK_PHASE_CAPTURE && if ((phase != GTK_PHASE_CAPTURE &&
!gtk_widget_is_ancestor (widget, grab_widget)) || !gtk_widget_is_ancestor (widget, grab_widget)) ||
(data->phase == GTK_PHASE_CAPTURE && (phase == GTK_PHASE_CAPTURE &&
!gtk_widget_is_ancestor (widget, grab_widget) && !gtk_widget_is_ancestor (widget, grab_widget) &&
!gtk_widget_is_ancestor (grab_widget, widget))) !gtk_widget_is_ancestor (grab_widget, widget)))
{ {
@ -16988,8 +16999,7 @@ _gtk_widget_has_controller (GtkWidget *widget,
void void
_gtk_widget_add_controller (GtkWidget *widget, _gtk_widget_add_controller (GtkWidget *widget,
GtkEventController *controller, GtkEventController *controller)
GtkPropagationPhase phase)
{ {
EventControllerData *data; EventControllerData *data;
GtkWidgetPrivate *priv; GtkWidgetPrivate *priv;
@ -16997,20 +17007,15 @@ _gtk_widget_add_controller (GtkWidget *widget,
g_return_if_fail (GTK_IS_WIDGET (widget)); g_return_if_fail (GTK_IS_WIDGET (widget));
g_return_if_fail (GTK_IS_EVENT_CONTROLLER (controller)); g_return_if_fail (GTK_IS_EVENT_CONTROLLER (controller));
g_return_if_fail (widget == gtk_event_controller_get_widget (controller)); g_return_if_fail (widget == gtk_event_controller_get_widget (controller));
g_return_if_fail (phase >= GTK_PHASE_NONE && phase <= GTK_PHASE_TARGET);
priv = widget->priv; priv = widget->priv;
data = _gtk_widget_has_controller (widget, controller); data = _gtk_widget_has_controller (widget, controller);
if (data) if (data)
{
data->phase = phase;
return; return;
}
data = g_new0 (EventControllerData, 1); data = g_new0 (EventControllerData, 1);
data->controller = g_object_ref (controller); data->controller = controller;
data->phase = phase;
data->evmask_notify_id = data->evmask_notify_id =
g_signal_connect (controller, "notify::event-mask", g_signal_connect (controller, "notify::event-mask",
G_CALLBACK (event_controller_notify_event_mask), widget); G_CALLBACK (event_controller_notify_event_mask), widget);
@ -17049,8 +17054,6 @@ _gtk_widget_remove_controller (GtkWidget *widget,
g_signal_handler_disconnect (data->controller, data->evmask_notify_id); g_signal_handler_disconnect (data->controller, data->evmask_notify_id);
g_signal_handler_disconnect (data->controller, data->sequence_state_changed_id); g_signal_handler_disconnect (data->controller, data->sequence_state_changed_id);
gtk_event_controller_reset (GTK_EVENT_CONTROLLER (data->controller));
g_object_unref (data->controller);
data->controller = NULL; data->controller = NULL;
} }
@ -17069,7 +17072,9 @@ _gtk_widget_list_controllers (GtkWidget *widget,
for (l = priv->event_controllers; l; l = l->next) for (l = priv->event_controllers; l; l = l->next)
{ {
data = l->data; data = l->data;
if (data->phase == phase && data->controller != NULL)
if (data->controller != NULL &&
phase == gtk_event_controller_get_propagation_phase (data->controller))
retval = g_list_prepend (retval, data->controller); retval = g_list_prepend (retval, data->controller);
} }

View File

@ -156,8 +156,7 @@ GActionGroup * _gtk_widget_get_action_group (GtkWidget *widget
const gchar *prefix); const gchar *prefix);
void _gtk_widget_add_controller (GtkWidget *widget, void _gtk_widget_add_controller (GtkWidget *widget,
GtkEventController *controller, GtkEventController *controller);
GtkPropagationPhase phase);
void _gtk_widget_remove_controller (GtkWidget *widget, void _gtk_widget_remove_controller (GtkWidget *widget,
GtkEventController *controller); GtkEventController *controller);
GList * _gtk_widget_list_controllers (GtkWidget *widget, GList * _gtk_widget_list_controllers (GtkWidget *widget,

View File

@ -1595,7 +1595,8 @@ gtk_window_constructor (GType type,
G_CALLBACK (multipress_gesture_pressed_cb), object); G_CALLBACK (multipress_gesture_pressed_cb), object);
g_signal_connect (priv->multipress_gesture, "stopped", g_signal_connect (priv->multipress_gesture, "stopped",
G_CALLBACK (multipress_gesture_stopped_cb), object); G_CALLBACK (multipress_gesture_stopped_cb), object);
gtk_gesture_attach (priv->multipress_gesture, GTK_PHASE_CAPTURE); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->multipress_gesture),
GTK_PHASE_CAPTURE);
} }
return object; return object;
@ -5482,10 +5483,7 @@ gtk_window_finalize (GObject *object)
} }
if (priv->multipress_gesture) if (priv->multipress_gesture)
{
gtk_gesture_detach (priv->multipress_gesture);
g_object_unref (priv->multipress_gesture); g_object_unref (priv->multipress_gesture);
}
G_OBJECT_CLASS (gtk_window_parent_class)->finalize (object); G_OBJECT_CLASS (gtk_window_parent_class)->finalize (object);
} }