[gimp] app: add new GimpCircle subblass GimpPolar to select polar coordinates



commit ca818d7c88b695fe4cfd660f5d6a7e0015a076ce
Author: Michael Natterer <mitch gimp org>
Date:   Fri May 30 00:31:53 2014 +0200

    app: add new GimpCircle subblass GimpPolar to select polar coordinates

 app/widgets/Makefile.am     |    2 +
 app/widgets/gimppolar.c     |  468 +++++++++++++++++++++++++++++++++++++++++++
 app/widgets/gimppolar.h     |   61 ++++++
 app/widgets/widgets-types.h |    1 +
 4 files changed, 532 insertions(+), 0 deletions(-)
---
diff --git a/app/widgets/Makefile.am b/app/widgets/Makefile.am
index 718d788..3a7df8e 100644
--- a/app/widgets/Makefile.am
+++ b/app/widgets/Makefile.am
@@ -270,6 +270,8 @@ libappwidgets_a_sources = \
        gimppixbuf.h                    \
        gimppluginaction.c              \
        gimppluginaction.h              \
+       gimppolar.c                     \
+       gimppolar.h                     \
        gimpprefsbox.c                  \
        gimpprefsbox.h                  \
        gimpprogressbox.c               \
diff --git a/app/widgets/gimppolar.c b/app/widgets/gimppolar.c
new file mode 100644
index 0000000..91b07b3
--- /dev/null
+++ b/app/widgets/gimppolar.c
@@ -0,0 +1,468 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * gimppolar.c
+ * Copyright (C) 2014 Michael Natterer <mitch gimp org>
+ *
+ * Based on code from the color-rotate plug-in
+ * Copyright (C) 1997-1999 Sven Anders (anderss fmi uni-passau de)
+ *                         Based on code from Pavel Grinfeld (pavel ml com)
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include <gegl.h>
+#include <gtk/gtk.h>
+
+#include "libgimpmath/gimpmath.h"
+#include "libgimpcolor/gimpcolor.h"
+#include "libgimpwidgets/gimpwidgets.h"
+
+#include "widgets-types.h"
+
+#include "core/gimp-cairo.h"
+
+#include "gimppolar.h"
+
+
+#define SEGMENT_FRACTION 0.3
+
+
+enum
+{
+  PROP_0,
+  PROP_ANGLE,
+  PROP_RADIUS
+};
+
+typedef enum
+{
+  POLAR_TARGET_NONE   = 0,
+  POLAR_TARGET_CIRCLE = 1 << 0
+} PolarTarget;
+
+
+struct _GimpPolarPrivate
+{
+  gdouble      angle;
+  gdouble      radius;
+
+  PolarTarget  target;
+  gboolean     has_grab;
+  gboolean     in_widget;
+};
+
+
+static void        gimp_polar_set_property         (GObject            *object,
+                                                    guint               property_id,
+                                                    const GValue       *value,
+                                                    GParamSpec         *pspec);
+static void        gimp_polar_get_property         (GObject            *object,
+                                                    guint               property_id,
+                                                    GValue             *value,
+                                                    GParamSpec         *pspec);
+
+static void        gimp_polar_unmap                (GtkWidget          *widget);
+static gboolean    gimp_polar_expose_event         (GtkWidget          *widget,
+                                                    GdkEventExpose     *event);
+static gboolean    gimp_polar_button_press_event   (GtkWidget          *widget,
+                                                    GdkEventButton     *bevent);
+static gboolean    gimp_polar_button_release_event (GtkWidget          *widget,
+                                                    GdkEventButton     *bevent);
+static gboolean    gimp_polar_motion_notify_event  (GtkWidget          *widget,
+                                                    GdkEventMotion     *mevent);
+static gboolean    gimp_polar_enter_notify_event   (GtkWidget          *widget,
+                                                    GdkEventCrossing   *event);
+static gboolean    gimp_polar_leave_notify_event   (GtkWidget          *widget,
+                                                    GdkEventCrossing   *event);
+
+static void        gimp_polar_set_target           (GimpPolar           *polar,
+                                                    PolarTarget          target);
+
+static void        gimp_polar_draw_circle          (cairo_t            *cr,
+                                                    gint                size,
+                                                    gdouble             angle,
+                                                    gdouble             radius,
+                                                    PolarTarget         highlight);
+
+static gdouble     gimp_polar_normalize_angle      (gdouble             angle);
+static gdouble     gimp_polar_get_angle_distance   (gdouble             alpha,
+                                                    gdouble             beta);
+
+
+G_DEFINE_TYPE (GimpPolar, gimp_polar, GIMP_TYPE_CIRCLE)
+
+#define parent_class gimp_polar_parent_class
+
+
+static void
+gimp_polar_class_init (GimpPolarClass *klass)
+{
+  GObjectClass   *object_class = G_OBJECT_CLASS (klass);
+  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+
+  object_class->get_property         = gimp_polar_get_property;
+  object_class->set_property         = gimp_polar_set_property;
+
+  widget_class->unmap                = gimp_polar_unmap;
+  widget_class->expose_event         = gimp_polar_expose_event;
+  widget_class->button_press_event   = gimp_polar_button_press_event;
+  widget_class->button_release_event = gimp_polar_button_release_event;
+  widget_class->motion_notify_event  = gimp_polar_motion_notify_event;
+  widget_class->enter_notify_event   = gimp_polar_enter_notify_event;
+  widget_class->leave_notify_event   = gimp_polar_leave_notify_event;
+
+  g_object_class_install_property (object_class, PROP_ANGLE,
+                                   g_param_spec_double ("angle",
+                                                        NULL, NULL,
+                                                        0.0, 2 * G_PI, 0.0,
+                                                        GIMP_PARAM_READWRITE |
+                                                        G_PARAM_CONSTRUCT));
+
+  g_object_class_install_property (object_class, PROP_RADIUS,
+                                   g_param_spec_double ("radius",
+                                                        NULL, NULL,
+                                                        0.0, 1.0, 0.0,
+                                                        GIMP_PARAM_READWRITE |
+                                                        G_PARAM_CONSTRUCT));
+
+  g_type_class_add_private (klass, sizeof (GimpPolarPrivate));
+}
+
+static void
+gimp_polar_init (GimpPolar *polar)
+{
+  polar->priv = G_TYPE_INSTANCE_GET_PRIVATE (polar,
+                                             GIMP_TYPE_POLAR,
+                                             GimpPolarPrivate);
+
+  gtk_widget_add_events (GTK_WIDGET (polar),
+                         GDK_POINTER_MOTION_MASK |
+                         GDK_BUTTON_PRESS_MASK   |
+                         GDK_BUTTON_RELEASE_MASK |
+                         GDK_BUTTON1_MOTION_MASK |
+                         GDK_ENTER_NOTIFY_MASK   |
+                         GDK_LEAVE_NOTIFY_MASK);
+}
+
+static void
+gimp_polar_set_property (GObject      *object,
+                         guint         property_id,
+                         const GValue *value,
+                         GParamSpec   *pspec)
+{
+  GimpPolar *polar = GIMP_POLAR (object);
+
+  switch (property_id)
+    {
+    case PROP_ANGLE:
+      polar->priv->angle = g_value_get_double (value);
+      gtk_widget_queue_draw (GTK_WIDGET (polar));
+      break;
+
+    case PROP_RADIUS:
+      polar->priv->radius = g_value_get_double (value);
+      gtk_widget_queue_draw (GTK_WIDGET (polar));
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+      break;
+    }
+}
+
+static void
+gimp_polar_get_property (GObject    *object,
+                         guint       property_id,
+                         GValue     *value,
+                         GParamSpec *pspec)
+{
+  GimpPolar *polar = GIMP_POLAR (object);
+
+  switch (property_id)
+    {
+    case PROP_ANGLE:
+      g_value_set_double (value, polar->priv->angle);
+      break;
+
+    case PROP_RADIUS:
+      g_value_set_double (value, polar->priv->radius);
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+      break;
+    }
+}
+
+static void
+gimp_polar_unmap (GtkWidget *widget)
+{
+  GimpPolar *polar = GIMP_POLAR (widget);
+
+  if (polar->priv->has_grab)
+    {
+      gtk_grab_remove (widget);
+      polar->priv->has_grab = FALSE;
+    }
+
+  GTK_WIDGET_CLASS (parent_class)->unmap (widget);
+}
+
+static gboolean
+gimp_polar_expose_event (GtkWidget      *widget,
+                         GdkEventExpose *event)
+{
+  GimpPolar *polar = GIMP_POLAR (widget);
+
+  GTK_WIDGET_CLASS (parent_class)->expose_event (widget, event);
+
+  if (gtk_widget_is_drawable (widget))
+    {
+      GtkAllocation  allocation;
+      gint           size;
+      cairo_t       *cr;
+
+      g_object_get (widget,
+                    "size", &size,
+                    NULL);
+
+      cr = gdk_cairo_create (event->window);
+      gdk_cairo_region (cr, event->region);
+      cairo_clip (cr);
+
+      gtk_widget_get_allocation (widget, &allocation);
+
+      cairo_translate (cr,
+                       (gdouble) allocation.x + (allocation.width  - size) / 2.0,
+                       (gdouble) allocation.y + (allocation.height - size) / 2.0);
+
+      gimp_polar_draw_circle (cr, size,
+                              polar->priv->angle, polar->priv->radius,
+                              polar->priv->target);
+
+      cairo_destroy (cr);
+    }
+
+  return FALSE;
+}
+
+static gboolean
+gimp_polar_button_press_event (GtkWidget      *widget,
+                               GdkEventButton *bevent)
+{
+  GimpPolar *polar = GIMP_POLAR (widget);
+
+  if (bevent->type == GDK_BUTTON_PRESS &&
+      bevent->button == 1              &&
+      polar->priv->target != POLAR_TARGET_NONE)
+    {
+      gdouble angle;
+      gdouble radius;
+
+      gtk_grab_add (widget);
+      polar->priv->has_grab = TRUE;
+
+      angle = _gimp_circle_get_angle_and_distance (GIMP_CIRCLE (polar),
+                                                   bevent->x, bevent->y,
+                                                   &radius);
+      radius = MIN (radius, 1.0);
+
+      g_object_set (polar,
+                    "angle",  angle,
+                    "radius", radius,
+                    NULL);
+    }
+
+  return FALSE;
+}
+
+static gboolean
+gimp_polar_button_release_event (GtkWidget      *widget,
+                                 GdkEventButton *bevent)
+{
+  GimpPolar *polar = GIMP_POLAR (widget);
+
+  if (bevent->button == 1)
+    {
+      gtk_grab_remove (widget);
+      polar->priv->has_grab = FALSE;
+
+      if (! polar->priv->in_widget)
+        gimp_polar_set_target (polar, POLAR_TARGET_NONE);
+    }
+
+  return FALSE;
+}
+
+static gboolean
+gimp_polar_motion_notify_event (GtkWidget      *widget,
+                                GdkEventMotion *mevent)
+{
+  GimpPolar *polar = GIMP_POLAR (widget);
+  gdouble    angle;
+  gdouble    radius;
+
+  angle = _gimp_circle_get_angle_and_distance (GIMP_CIRCLE (polar),
+                                               mevent->x, mevent->y,
+                                               &radius);
+
+  if (polar->priv->has_grab)
+    {
+      radius = MIN (radius, 1.0);
+
+      g_object_set (polar,
+                    "angle",  angle,
+                    "radius", radius,
+                    NULL);
+    }
+  else
+    {
+      PolarTarget target;
+      gdouble     dist_angle;
+      gdouble     dist_radius;
+
+      dist_angle  = gimp_polar_get_angle_distance (polar->priv->angle, angle);
+      dist_radius = ABS (polar->priv->radius - radius);
+
+      if ((radius < 0.2 && polar->priv->radius < 0.2) ||
+          dist_angle  < (G_PI / 12) && dist_radius < 0.2)
+        {
+          target = POLAR_TARGET_CIRCLE;
+        }
+      else
+        {
+          target = POLAR_TARGET_NONE;
+        }
+
+      gimp_polar_set_target (polar, target);
+    }
+
+  gdk_event_request_motions (mevent);
+
+  return FALSE;
+}
+
+static gboolean
+gimp_polar_enter_notify_event (GtkWidget        *widget,
+                               GdkEventCrossing *event)
+{
+  GimpPolar *polar = GIMP_POLAR (widget);
+
+  polar->priv->in_widget = TRUE;
+
+  return FALSE;
+}
+
+static gboolean
+gimp_polar_leave_notify_event (GtkWidget        *widget,
+                               GdkEventCrossing *event)
+{
+  GimpPolar *polar = GIMP_POLAR (widget);
+
+  polar->priv->in_widget = FALSE;
+
+  if (! polar->priv->has_grab)
+    gimp_polar_set_target (polar, POLAR_TARGET_NONE);
+
+  return FALSE;
+}
+
+
+/*  public functions  */
+
+GtkWidget *
+gimp_polar_new (void)
+{
+  return g_object_new (GIMP_TYPE_POLAR, NULL);
+}
+
+
+/*  private functions  */
+
+static void
+gimp_polar_set_target (GimpPolar   *polar,
+                       PolarTarget  target)
+{
+  if (target != polar->priv->target)
+    {
+      polar->priv->target = target;
+      gtk_widget_queue_draw (GTK_WIDGET (polar));
+    }
+}
+
+static void
+gimp_polar_draw_circle (cairo_t     *cr,
+                        gint         size,
+                        gdouble      angle,
+                        gdouble      radius,
+                        PolarTarget  highlight)
+{
+  gdouble r = size / 2.0 - 2.0; /* half the broad line with and half a px */
+  gdouble x = r + r * radius * cos (angle);
+  gdouble y = r - r * radius * sin (angle);
+
+  cairo_save (cr);
+
+  cairo_translate (cr, 2.0, 2.0); /* half the broad line width and half a px*/
+
+  cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
+  cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
+
+  cairo_arc (cr, x, y, 3, 0, 2 * G_PI);
+
+  if (highlight == POLAR_TARGET_NONE)
+    {
+      cairo_set_line_width (cr, 3.0);
+      cairo_set_source_rgba (cr, 1.0, 1.0, 1.0, 0.6);
+      cairo_stroke_preserve (cr);
+
+      cairo_set_line_width (cr, 1.0);
+      cairo_set_source_rgba (cr, 0.0, 0.0, 0.0, 0.8);
+      cairo_stroke (cr);
+    }
+  else
+    {
+      cairo_set_line_width (cr, 3.0);
+      cairo_set_source_rgba (cr, 0.0, 0.0, 0.0, 0.6);
+      cairo_stroke_preserve (cr);
+
+      cairo_set_line_width (cr, 1.0);
+      cairo_set_source_rgba (cr, 1.0, 1.0, 1.0, 0.8);
+      cairo_stroke (cr);
+    }
+
+  cairo_restore (cr);
+}
+
+static gdouble
+gimp_polar_normalize_angle (gdouble angle)
+{
+  if (angle < 0)
+    return angle + 2 * G_PI;
+  else if (angle > 2 * G_PI)
+    return angle - 2 * G_PI;
+  else
+    return angle;
+}
+
+static gdouble
+gimp_polar_get_angle_distance (gdouble alpha,
+                               gdouble beta)
+{
+  return ABS (MIN (gimp_polar_normalize_angle (alpha - beta),
+                   2 * G_PI - gimp_polar_normalize_angle (alpha - beta)));
+}
diff --git a/app/widgets/gimppolar.h b/app/widgets/gimppolar.h
new file mode 100644
index 0000000..303bbb1
--- /dev/null
+++ b/app/widgets/gimppolar.h
@@ -0,0 +1,61 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * gimppolar.h
+ * Copyright (C) 2014 Michael Natterer <mitch gimp org>
+ *
+ * Based on code from the color-rotate plug-in
+ * Copyright (C) 1997-1999 Sven Anders (anderss fmi uni-passau de)
+ *                         Based on code from Pavel Grinfeld (pavel ml com)
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef __GIMP_POLAR_H__
+#define __GIMP_POLAR_H__
+
+
+#include "gimpcircle.h"
+
+
+#define GIMP_TYPE_POLAR            (gimp_polar_get_type ())
+#define GIMP_POLAR(obj)            (G_TYPE_CHECK_INSTANCE_CAST ((obj), GIMP_TYPE_POLAR, GimpPolar))
+#define GIMP_POLAR_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST ((klass), GIMP_TYPE_POLAR, GimpPolarClass))
+#define GIMP_IS_POLAR(obj)         (G_TYPE_CHECK_INSTANCE_TYPE (obj, GIMP_TYPE_POLAR))
+#define GIMP_IS_POLAR_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GIMP_TYPE_POLAR))
+#define GIMP_POLAR_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS ((obj), GIMP_TYPE_POLAR, GimpPolarClass))
+
+
+typedef struct _GimpPolarPrivate GimpPolarPrivate;
+typedef struct _GimpPolarClass   GimpPolarClass;
+
+struct _GimpPolar
+{
+  GimpCircle       parent_instance;
+
+  GimpPolarPrivate *priv;
+};
+
+struct _GimpPolarClass
+{
+  GimpCircleClass  parent_class;
+};
+
+
+GType       gimp_polar_get_type (void) G_GNUC_CONST;
+
+GtkWidget * gimp_polar_new      (void);
+
+
+#endif /* __GIMP_POLAR_H__ */
diff --git a/app/widgets/widgets-types.h b/app/widgets/widgets-types.h
index 6651b88..d9d2275 100644
--- a/app/widgets/widgets-types.h
+++ b/app/widgets/widgets-types.h
@@ -189,6 +189,7 @@ typedef struct _GimpLanguageStore            GimpLanguageStore;
 typedef struct _GimpMessageBox               GimpMessageBox;
 typedef struct _GimpOverlayBox               GimpOverlayBox;
 typedef struct _GimpPickableButton           GimpPickableButton;
+typedef struct _GimpPolar                    GimpPolar;
 typedef struct _GimpPrefsBox                 GimpPrefsBox;
 typedef struct _GimpProgressBox              GimpProgressBox;
 typedef struct _GimpScaleButton              GimpScaleButton;


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