This prevents some of our generic object implementation tests from working with gesture objects. Instead, add g_return_if_fail checks in all the gesture constructors.
		
			
				
	
	
		
			218 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			218 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* 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>
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * SECTION:gtkgesturezoom
 | 
						|
 * @Short_description: Zoom gesture
 | 
						|
 * @Title: GtkGestureZoom
 | 
						|
 * @See_also: #GtkGestureRotate
 | 
						|
 *
 | 
						|
 * #GtkGestureZoom is a #GtkGesture implementation able to recognize
 | 
						|
 * pinch/zoom gestures, whenever the distance between both tracked
 | 
						|
 * sequences changes, the #GtkGestureZoom::scale-changed signal is
 | 
						|
 * emitted to report the scale factor.
 | 
						|
 */
 | 
						|
 | 
						|
#include "config.h"
 | 
						|
#include <math.h>
 | 
						|
#include "gtkgesturezoom.h"
 | 
						|
#include "gtkgesturezoomprivate.h"
 | 
						|
 | 
						|
typedef struct _GtkGestureZoomPrivate GtkGestureZoomPrivate;
 | 
						|
 | 
						|
enum {
 | 
						|
  SCALE_CHANGED,
 | 
						|
  LAST_SIGNAL
 | 
						|
};
 | 
						|
 | 
						|
struct _GtkGestureZoomPrivate
 | 
						|
{
 | 
						|
  gdouble initial_distance;
 | 
						|
};
 | 
						|
 | 
						|
static guint signals[LAST_SIGNAL] = { 0 };
 | 
						|
 | 
						|
G_DEFINE_TYPE_WITH_PRIVATE (GtkGestureZoom, gtk_gesture_zoom, GTK_TYPE_GESTURE)
 | 
						|
 | 
						|
static void
 | 
						|
gtk_gesture_zoom_init (GtkGestureZoom *gesture)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static GObject *
 | 
						|
gtk_gesture_zoom_constructor (GType                  type,
 | 
						|
                              guint                  n_construct_properties,
 | 
						|
                              GObjectConstructParam *construct_properties)
 | 
						|
{
 | 
						|
  GObject *object;
 | 
						|
 | 
						|
  object = G_OBJECT_CLASS (gtk_gesture_zoom_parent_class)->constructor (type,
 | 
						|
                                                                        n_construct_properties,
 | 
						|
                                                                        construct_properties);
 | 
						|
  g_object_set (object, "n-points", 2, NULL);
 | 
						|
 | 
						|
  return object;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
_gtk_gesture_zoom_get_distance (GtkGestureZoom *zoom,
 | 
						|
                                gdouble        *distance)
 | 
						|
{
 | 
						|
  gdouble x1, y1, x2, y2;
 | 
						|
  GtkGesture *gesture;
 | 
						|
  GList *sequences;
 | 
						|
  gdouble dx, dy;
 | 
						|
 | 
						|
  gesture = GTK_GESTURE (zoom);
 | 
						|
 | 
						|
  if (!gtk_gesture_is_recognized (gesture))
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  sequences = gtk_gesture_get_sequences (gesture);
 | 
						|
  g_assert (sequences && sequences->next);
 | 
						|
 | 
						|
  gtk_gesture_get_point (gesture, sequences->data, &x1, &y1);
 | 
						|
  gtk_gesture_get_point (gesture, sequences->next->data, &x2, &y2);
 | 
						|
  g_list_free (sequences);
 | 
						|
 | 
						|
  dx = x1 - x2;
 | 
						|
  dy = y1 - y2;;
 | 
						|
  *distance = sqrt ((dx * dx) + (dy * dy));
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
_gtk_gesture_zoom_check_emit (GtkGestureZoom *gesture)
 | 
						|
{
 | 
						|
  GtkGestureZoomPrivate *priv;
 | 
						|
  gdouble distance, zoom;
 | 
						|
 | 
						|
  if (!_gtk_gesture_zoom_get_distance (gesture, &distance))
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  priv = gtk_gesture_zoom_get_instance_private (gesture);
 | 
						|
 | 
						|
  if (distance == 0 || priv->initial_distance == 0)
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  zoom = distance / priv->initial_distance;
 | 
						|
  g_signal_emit (gesture, signals[SCALE_CHANGED], 0, zoom);
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_gesture_zoom_begin (GtkGesture       *gesture,
 | 
						|
                        GdkEventSequence *sequence)
 | 
						|
{
 | 
						|
  GtkGestureZoom *zoom = GTK_GESTURE_ZOOM (gesture);
 | 
						|
  GtkGestureZoomPrivate *priv;
 | 
						|
 | 
						|
  priv = gtk_gesture_zoom_get_instance_private (zoom);
 | 
						|
  _gtk_gesture_zoom_get_distance (zoom, &priv->initial_distance);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_gesture_zoom_update (GtkGesture       *gesture,
 | 
						|
                         GdkEventSequence *sequence)
 | 
						|
{
 | 
						|
  _gtk_gesture_zoom_check_emit (GTK_GESTURE_ZOOM (gesture));
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_gesture_zoom_class_init (GtkGestureZoomClass *klass)
 | 
						|
{
 | 
						|
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
 | 
						|
  GtkGestureClass *gesture_class = GTK_GESTURE_CLASS (klass);
 | 
						|
 | 
						|
  object_class->constructor = gtk_gesture_zoom_constructor;
 | 
						|
 | 
						|
  gesture_class->begin = gtk_gesture_zoom_begin;
 | 
						|
  gesture_class->update = gtk_gesture_zoom_update;
 | 
						|
 | 
						|
  /**
 | 
						|
   * GtkGestureZoom::scale-changed:
 | 
						|
   * @controller: the object on which the signal is emitted
 | 
						|
   * @scale: Scale delta, taking the initial state as 1:1
 | 
						|
   *
 | 
						|
   * This signal is emitted whenever the distance between both tracked
 | 
						|
   * sequences changes.
 | 
						|
   *
 | 
						|
   * Since: 3.14
 | 
						|
   */
 | 
						|
  signals[SCALE_CHANGED] =
 | 
						|
    g_signal_new ("scale-changed",
 | 
						|
                  GTK_TYPE_GESTURE_ZOOM,
 | 
						|
                  G_SIGNAL_RUN_FIRST,
 | 
						|
                  G_STRUCT_OFFSET (GtkGestureZoomClass, scale_changed),
 | 
						|
                  NULL, NULL, NULL,
 | 
						|
                  G_TYPE_NONE, 1, G_TYPE_DOUBLE);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * gtk_gesture_zoom_new:
 | 
						|
 * @widget: a #GtkWidget
 | 
						|
 *
 | 
						|
 * Returns a newly created #GtkGesture that recognizes zoom
 | 
						|
 * in/out gestures (usually known as pinch/zoom).
 | 
						|
 *
 | 
						|
 * Returns: a newly created #GtkGestureZoom
 | 
						|
 *
 | 
						|
 * Since: 3.14
 | 
						|
 **/
 | 
						|
GtkGesture *
 | 
						|
gtk_gesture_zoom_new (GtkWidget *widget)
 | 
						|
{
 | 
						|
  g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
 | 
						|
 | 
						|
  return g_object_new (GTK_TYPE_GESTURE_ZOOM,
 | 
						|
                       "widget", widget,
 | 
						|
                       NULL);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * gtk_gesture_zoom_get_scale_delta:
 | 
						|
 * @gesture: a #GtkGestureZoom
 | 
						|
 *
 | 
						|
 * If @gesture is active, this function returns the zooming difference
 | 
						|
 * since the gesture was recognized (hence the starting point is
 | 
						|
 * considered 1:1). If @gesture is not active, 1 is returned.
 | 
						|
 *
 | 
						|
 * Returns: the scale delta
 | 
						|
 *
 | 
						|
 * Since: 3.14
 | 
						|
 **/
 | 
						|
gdouble
 | 
						|
gtk_gesture_zoom_get_scale_delta (GtkGestureZoom *gesture)
 | 
						|
{
 | 
						|
  GtkGestureZoomPrivate *priv;
 | 
						|
  gdouble distance;
 | 
						|
 | 
						|
  g_return_val_if_fail (GTK_IS_GESTURE_ZOOM (gesture), 1.0);
 | 
						|
 | 
						|
  if (!_gtk_gesture_zoom_get_distance (gesture, &distance))
 | 
						|
    return 1.0;
 | 
						|
 | 
						|
  priv = gtk_gesture_zoom_get_instance_private (gesture);
 | 
						|
 | 
						|
  return distance / priv->initial_distance;
 | 
						|
}
 |