2013-01-17 22:57:06 +00:00
|
|
|
/* GTK - The GIMP Toolkit
|
|
|
|
* Copyright (C) 2012, One Laptop Per Child.
|
|
|
|
* Copyright (C) 2014, 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 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/>.
|
|
|
|
*
|
|
|
|
* Author(s): Carlos Garnacho <carlosg@gnome.org>
|
|
|
|
*/
|
2014-04-09 16:28:12 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* SECTION:gtkgesturelongpress
|
2014-05-20 00:45:42 +00:00
|
|
|
* @Short_description: "Press and Hold" gesture
|
2014-04-09 16:28:12 +00:00
|
|
|
* @Title: GtkGestureLongPress
|
|
|
|
*
|
|
|
|
* #GtkGestureLongPress is a #GtkGesture implementation able to recognize
|
2014-04-10 11:43:41 +00:00
|
|
|
* long presses, triggering the #GtkGestureLongPress::pressed after the
|
2014-04-09 16:28:12 +00:00
|
|
|
* timeout is exceeded.
|
|
|
|
*
|
|
|
|
* If the touchpoint is lifted before the timeout passes, or if it drifts
|
|
|
|
* too far of the initial press point, the #GtkGestureLongPress::cancelled
|
|
|
|
* signal will be emitted.
|
|
|
|
*/
|
|
|
|
|
2013-01-17 22:57:06 +00:00
|
|
|
#include "config.h"
|
2014-05-06 02:40:18 +00:00
|
|
|
#include "gtkgesturelongpress.h"
|
|
|
|
#include "gtkgesturelongpressprivate.h"
|
2014-05-06 16:04:25 +00:00
|
|
|
#include "gtkgestureprivate.h"
|
2013-01-17 22:57:06 +00:00
|
|
|
#include "gtkmarshalers.h"
|
2014-05-06 02:40:18 +00:00
|
|
|
#include "gtkdnd.h"
|
2013-01-17 22:57:06 +00:00
|
|
|
#include "gtkprivate.h"
|
|
|
|
#include "gtkintl.h"
|
|
|
|
|
|
|
|
typedef struct _GtkGestureLongPressPrivate GtkGestureLongPressPrivate;
|
|
|
|
|
|
|
|
enum {
|
|
|
|
PRESSED,
|
|
|
|
CANCELLED,
|
|
|
|
N_SIGNALS
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _GtkGestureLongPressPrivate
|
|
|
|
{
|
|
|
|
gdouble initial_x;
|
|
|
|
gdouble initial_y;
|
|
|
|
|
|
|
|
guint timeout_id;
|
|
|
|
guint delay;
|
|
|
|
guint cancelled : 1;
|
|
|
|
guint triggered : 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
static guint signals[N_SIGNALS] = { 0 };
|
|
|
|
|
2014-03-21 17:57:32 +00:00
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (GtkGestureLongPress, gtk_gesture_long_press, GTK_TYPE_GESTURE_SINGLE)
|
2013-01-17 22:57:06 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_gesture_long_press_init (GtkGestureLongPress *gesture)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gtk_gesture_long_press_check (GtkGesture *gesture)
|
|
|
|
{
|
|
|
|
GtkGestureLongPressPrivate *priv;
|
|
|
|
|
|
|
|
priv = gtk_gesture_long_press_get_instance_private (GTK_GESTURE_LONG_PRESS (gesture));
|
|
|
|
|
|
|
|
if (priv->cancelled)
|
|
|
|
return FALSE;
|
|
|
|
|
2014-04-08 19:11:27 +00:00
|
|
|
return GTK_GESTURE_CLASS (gtk_gesture_long_press_parent_class)->check (gesture);
|
2013-01-17 22:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_gtk_gesture_long_press_timeout (gpointer user_data)
|
|
|
|
{
|
|
|
|
GtkGestureLongPress *gesture = user_data;
|
|
|
|
GtkGestureLongPressPrivate *priv;
|
|
|
|
GdkEventSequence *sequence;
|
|
|
|
gdouble x, y;
|
|
|
|
|
|
|
|
priv = gtk_gesture_long_press_get_instance_private (gesture);
|
|
|
|
sequence = gtk_gesture_get_last_updated_sequence (GTK_GESTURE (gesture));
|
|
|
|
gtk_gesture_get_point (GTK_GESTURE (gesture), sequence, &x, &y);
|
|
|
|
|
|
|
|
priv->timeout_id = 0;
|
|
|
|
priv->triggered = TRUE;
|
|
|
|
g_signal_emit (gesture, signals[PRESSED], 0, x, y);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_gesture_long_press_begin (GtkGesture *gesture,
|
|
|
|
GdkEventSequence *sequence)
|
|
|
|
{
|
|
|
|
GtkGestureLongPressPrivate *priv;
|
2014-03-21 17:57:32 +00:00
|
|
|
const GdkEvent *event;
|
2014-03-31 10:27:43 +00:00
|
|
|
GtkWidget *widget;
|
|
|
|
gint delay;
|
2013-01-17 22:57:06 +00:00
|
|
|
|
|
|
|
priv = gtk_gesture_long_press_get_instance_private (GTK_GESTURE_LONG_PRESS (gesture));
|
2014-03-21 17:57:32 +00:00
|
|
|
sequence = gtk_gesture_single_get_current_sequence (GTK_GESTURE_SINGLE (gesture));
|
2013-01-17 22:57:06 +00:00
|
|
|
event = gtk_gesture_get_last_event (gesture, sequence);
|
|
|
|
|
2014-03-21 17:57:32 +00:00
|
|
|
if (!event ||
|
|
|
|
(event->type != GDK_BUTTON_PRESS &&
|
|
|
|
event->type != GDK_TOUCH_BEGIN))
|
2013-01-17 22:57:06 +00:00
|
|
|
return;
|
|
|
|
|
2014-03-31 10:27:43 +00:00
|
|
|
widget = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
|
|
|
|
g_object_get (gtk_widget_get_settings (widget),
|
|
|
|
"gtk-long-press-time", &delay, NULL);
|
|
|
|
|
2013-01-17 22:57:06 +00:00
|
|
|
gtk_gesture_get_point (gesture, sequence,
|
|
|
|
&priv->initial_x, &priv->initial_y);
|
|
|
|
priv->timeout_id =
|
2014-03-31 10:27:43 +00:00
|
|
|
gdk_threads_add_timeout (delay,
|
2013-01-17 22:57:06 +00:00
|
|
|
_gtk_gesture_long_press_timeout,
|
|
|
|
gesture);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_gesture_long_press_update (GtkGesture *gesture,
|
|
|
|
GdkEventSequence *sequence)
|
|
|
|
{
|
|
|
|
GtkGestureLongPressPrivate *priv;
|
|
|
|
GtkWidget *widget;
|
|
|
|
gdouble x, y;
|
|
|
|
|
|
|
|
widget = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
|
|
|
|
priv = gtk_gesture_long_press_get_instance_private (GTK_GESTURE_LONG_PRESS (gesture));
|
|
|
|
gtk_gesture_get_point (gesture, sequence, &x, &y);
|
|
|
|
|
|
|
|
if (gtk_drag_check_threshold (widget, priv->initial_x, priv->initial_y, x, y))
|
|
|
|
{
|
|
|
|
if (priv->timeout_id)
|
|
|
|
{
|
|
|
|
g_source_remove (priv->timeout_id);
|
|
|
|
priv->timeout_id = 0;
|
2014-04-08 19:10:43 +00:00
|
|
|
g_signal_emit (gesture, signals[CANCELLED], 0);
|
2013-01-17 22:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
priv->cancelled = TRUE;
|
2014-05-06 16:04:25 +00:00
|
|
|
_gtk_gesture_check (gesture);
|
2013-01-17 22:57:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_gesture_long_press_end (GtkGesture *gesture,
|
|
|
|
GdkEventSequence *sequence)
|
|
|
|
{
|
|
|
|
GtkGestureLongPressPrivate *priv;
|
|
|
|
|
|
|
|
priv = gtk_gesture_long_press_get_instance_private (GTK_GESTURE_LONG_PRESS (gesture));
|
|
|
|
|
|
|
|
if (priv->timeout_id)
|
|
|
|
{
|
|
|
|
g_source_remove (priv->timeout_id);
|
|
|
|
priv->timeout_id = 0;
|
2014-04-08 19:10:43 +00:00
|
|
|
g_signal_emit (gesture, signals[CANCELLED], 0);
|
2013-01-17 22:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
priv->cancelled = priv->triggered = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2014-03-21 17:57:32 +00:00
|
|
|
gtk_gesture_long_press_sequence_state_changed (GtkGesture *gesture,
|
|
|
|
GdkEventSequence *sequence,
|
|
|
|
GtkEventSequenceState state)
|
2013-01-17 22:57:06 +00:00
|
|
|
{
|
2014-03-21 17:57:32 +00:00
|
|
|
if (state == GTK_EVENT_SEQUENCE_DENIED)
|
2014-04-08 19:10:43 +00:00
|
|
|
gtk_gesture_long_press_end (gesture, sequence);
|
2013-01-17 22:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2014-03-21 17:57:32 +00:00
|
|
|
gtk_gesture_long_press_finalize (GObject *object)
|
2013-01-17 22:57:06 +00:00
|
|
|
{
|
|
|
|
GtkGestureLongPressPrivate *priv;
|
|
|
|
|
2014-03-21 17:57:32 +00:00
|
|
|
priv = gtk_gesture_long_press_get_instance_private (GTK_GESTURE_LONG_PRESS (object));
|
2013-01-17 22:57:06 +00:00
|
|
|
|
2014-03-21 17:57:32 +00:00
|
|
|
if (priv->timeout_id)
|
|
|
|
g_source_remove (priv->timeout_id);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_gesture_long_press_parent_class)->finalize (object);
|
2013-01-17 22:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_gesture_long_press_class_init (GtkGestureLongPressClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
GtkGestureClass *gesture_class = GTK_GESTURE_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->finalize = gtk_gesture_long_press_finalize;
|
|
|
|
|
|
|
|
gesture_class->check = gtk_gesture_long_press_check;
|
|
|
|
gesture_class->begin = gtk_gesture_long_press_begin;
|
|
|
|
gesture_class->update = gtk_gesture_long_press_update;
|
|
|
|
gesture_class->end = gtk_gesture_long_press_end;
|
2014-03-21 17:57:32 +00:00
|
|
|
gesture_class->cancel = gtk_gesture_long_press_end;
|
|
|
|
gesture_class->sequence_state_changed =
|
|
|
|
gtk_gesture_long_press_sequence_state_changed;
|
2013-01-17 22:57:06 +00:00
|
|
|
|
2014-04-09 16:28:12 +00:00
|
|
|
/**
|
2014-04-10 11:43:41 +00:00
|
|
|
* GtkGestureLongPress::pressed:
|
2014-04-09 16:28:12 +00:00
|
|
|
* @gesture: the object which received the signal
|
|
|
|
* @x: the X coordinate where the press happened, relative to the widget allocation
|
|
|
|
* @y: the Y coordinate where the press happened, relative to the widget allocation
|
|
|
|
*
|
|
|
|
* This signal is emitted whenever a press goes unmoved/unreleased longer than
|
|
|
|
* what the GTK+ defaults tell.
|
|
|
|
*
|
|
|
|
* Since: 3.14
|
|
|
|
*/
|
2013-01-17 22:57:06 +00:00
|
|
|
signals[PRESSED] =
|
|
|
|
g_signal_new ("pressed",
|
|
|
|
G_TYPE_FROM_CLASS (klass),
|
|
|
|
G_SIGNAL_RUN_LAST,
|
|
|
|
G_STRUCT_OFFSET (GtkGestureLongPressClass, pressed),
|
|
|
|
NULL, NULL, NULL,
|
|
|
|
G_TYPE_NONE, 2, G_TYPE_DOUBLE, G_TYPE_DOUBLE);
|
2014-04-09 16:28:12 +00:00
|
|
|
/**
|
2014-04-10 11:43:41 +00:00
|
|
|
* GtkGestureLongPress::cancelled:
|
2014-04-09 16:28:12 +00:00
|
|
|
* @gesture: the object which received the signal
|
|
|
|
*
|
|
|
|
* This signal is emitted whenever a press moved too far, or was released
|
|
|
|
* before #GtkGestureLongPress:pressed happened.
|
|
|
|
*
|
|
|
|
* Since: 3.14
|
|
|
|
*/
|
2013-01-17 22:57:06 +00:00
|
|
|
signals[CANCELLED] =
|
|
|
|
g_signal_new ("cancelled",
|
|
|
|
G_TYPE_FROM_CLASS (klass),
|
|
|
|
G_SIGNAL_RUN_LAST,
|
|
|
|
G_STRUCT_OFFSET (GtkGestureLongPressClass, cancelled),
|
|
|
|
NULL, NULL, NULL,
|
|
|
|
G_TYPE_NONE, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_gesture_long_press_new:
|
|
|
|
* @widget: a #GtkWidget
|
|
|
|
*
|
2014-05-20 02:16:03 +00:00
|
|
|
* Returns a newly created #GtkGesture that recognizes long presses.
|
2013-01-17 22:57:06 +00:00
|
|
|
*
|
|
|
|
* Returns: a newly created #GtkGestureLongPress
|
|
|
|
*
|
|
|
|
* Since: 3.14
|
|
|
|
**/
|
|
|
|
GtkGesture *
|
|
|
|
gtk_gesture_long_press_new (GtkWidget *widget)
|
|
|
|
{
|
2014-05-27 18:30:47 +00:00
|
|
|
g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
|
|
|
|
|
2013-01-17 22:57:06 +00:00
|
|
|
return g_object_new (GTK_TYPE_GESTURE_LONG_PRESS,
|
|
|
|
"widget", widget,
|
|
|
|
NULL);
|
|
|
|
}
|