[gnome-photos/wip/rishi/edit-mode: 20/26] Add PhotosToolCrop
- From: Debarshi Ray <debarshir src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-photos/wip/rishi/edit-mode: 20/26] Add PhotosToolCrop
- Date: Thu, 27 Aug 2015 09:39:21 +0000 (UTC)
commit cb192a1b72501e2cc5266f1a442dc5aac188b05f
Author: Debarshi Ray <debarshir gnome org>
Date: Fri Jun 12 01:35:43 2015 +0200
Add PhotosToolCrop
src/Makefile.am | 2 +
src/photos-icons.h | 1 +
src/photos-tool-crop.c | 690 ++++++++++++++++++++++++++++++++++++++++++++++++
src/photos-tool-crop.h | 45 ++++
src/photos-utils.c | 2 +
5 files changed, 740 insertions(+), 0 deletions(-)
---
diff --git a/src/Makefile.am b/src/Makefile.am
index 310a981..7f9e6be 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -218,6 +218,8 @@ gnome_photos_SOURCES = \
photos-spinner-box.h \
photos-tool.c \
photos-tool.h \
+ photos-tool-crop.c \
+ photos-tool-crop.h \
photos-tool-filters.c \
photos-tool-filters.h \
photos-tool-sharpen.c \
diff --git a/src/photos-icons.h b/src/photos-icons.h
index 80864ed..4e84f6f 100644
--- a/src/photos-icons.h
+++ b/src/photos-icons.h
@@ -40,6 +40,7 @@ G_BEGIN_DECLS
#define PHOTOS_ICON_GO_NEXT_SYMBOLIC "go-next-symbolic"
#define PHOTOS_ICON_GO_PREVIOUS_SYMBOLIC "go-previous-symbolic"
+#define PHOTOS_ICON_IMAGE_CROP_SYMBOLIC "image-crop-symbolic"
#define PHOTOS_ICON_IMAGE_FILTER_SYMBOLIC "image-filter-symbolic"
#define PHOTOS_ICON_IMAGE_SHARPEN_SYMBOLIC "image-sharpen-symbolic"
diff --git a/src/photos-tool-crop.c b/src/photos-tool-crop.c
new file mode 100644
index 0000000..caaec57
--- /dev/null
+++ b/src/photos-tool-crop.c
@@ -0,0 +1,690 @@
+/*
+ * Photos - access, organize and share your photos on GNOME
+ * Copyright © 2015 Red Hat, Inc.
+ *
+ * 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 2
+ * 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, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+ * 02110-1301, USA.
+ */
+
+
+#include "config.h"
+
+#include <cairo.h>
+#include <gdk/gdk.h>
+#include <gio/gio.h>
+#include <glib/gi18n.h>
+#include <gtk/gtk.h>
+
+#include "gegl-gtk-view.h"
+#include "photos-icons.h"
+#include "photos-tool.h"
+#include "photos-tool-crop.h"
+#include "photos-utils.h"
+
+
+typedef enum
+{
+ PHOTOS_TOOL_CROP_LOCATION_NONE,
+ PHOTOS_TOOL_CROP_LOCATION_BOTTOM_LEFT,
+ PHOTOS_TOOL_CROP_LOCATION_BOTTOM_RIGHT,
+ PHOTOS_TOOL_CROP_LOCATION_BOTTOM_SIDE,
+ PHOTOS_TOOL_CROP_LOCATION_CENTER,
+ PHOTOS_TOOL_CROP_LOCATION_LEFT_SIDE,
+ PHOTOS_TOOL_CROP_LOCATION_RIGHT_SIDE,
+ PHOTOS_TOOL_CROP_LOCATION_TOP_LEFT,
+ PHOTOS_TOOL_CROP_LOCATION_TOP_RIGHT,
+ PHOTOS_TOOL_CROP_LOCATION_TOP_SIDE
+} PhotosToolCropLocation;
+
+struct _PhotosToolCrop
+{
+ PhotosTool parent_instance;
+ GeglRectangle bbox_scaled;
+ GeglRectangle bbox_source;
+ GtkListStore *model;
+ GtkWidget *combo_box;
+ GtkWidget *view;
+ PhotosToolCropLocation location;
+ cairo_surface_t *surface;
+ gboolean grabbed;
+ gdouble crop_height;
+ gdouble crop_height0;
+ gdouble crop_width;
+ gdouble crop_width0;
+ gdouble crop_x;
+ gdouble crop_x0;
+ gdouble crop_y;
+ gdouble crop_y0;
+ gdouble event_x0;
+ gdouble event_y0;
+};
+
+struct _PhotosToolCropClass
+{
+ PhotosToolClass parent_class;
+};
+
+
+G_DEFINE_TYPE_WITH_CODE (PhotosToolCrop, photos_tool_crop, PHOTOS_TYPE_TOOL,
+ photos_utils_ensure_extension_points ();
+ g_io_extension_point_implement (PHOTOS_TOOL_EXTENSION_POINT_NAME,
+ g_define_type_id,
+ "crop",
+ 100));
+
+
+typedef enum
+{
+ PHOTOS_TOOL_CROP_ASPECT_RATIO_ANY,
+ PHOTOS_TOOL_CROP_ASPECT_RATIO_BASIS,
+ PHOTOS_TOOL_CROP_ASPECT_RATIO_ORIGINAL,
+ PHOTOS_TOOL_CROP_ASPECT_RATIO_SCREEN
+} PhotosToolCropAspectRatioType;
+
+typedef struct _PhotosToolCropConstraint PhotosToolCropConstraint;
+
+struct _PhotosToolCropConstraint
+{
+ PhotosToolCropAspectRatioType aspect_ratio_type;
+ const gchar *name;
+ guint basis_height;
+ guint basis_width;
+};
+
+enum
+{
+ CONSTRAINT_COLUMN_ASPECT_RATIO = 0,
+ CONSTRAINT_COLUMN_NAME = 1,
+ CONSTRAINT_COLUMN_BASIS_HEIGHT = 2,
+ CONSTRAINT_COLUMN_BASIS_WIDTH = 3
+};
+
+static PhotosToolCropConstraint CONSTRAINTS[] =
+{
+ { PHOTOS_TOOL_CROP_ASPECT_RATIO_ANY, N_("Free"), 0, 0 },
+ { PHOTOS_TOOL_CROP_ASPECT_RATIO_ORIGINAL, N_("Original Size"), 0, 0 },
+ { PHOTOS_TOOL_CROP_ASPECT_RATIO_SCREEN, N_("Screen"), 0, 0 },
+ { PHOTOS_TOOL_CROP_ASPECT_RATIO_BASIS, N_("Square"), 1, 1 }
+};
+
+
+static gdouble
+photos_tool_crop_calculate_aspect_ratio (PhotosToolCrop *self)
+{
+ gdouble ret_val = 1.0;
+ gint active;
+
+ active = gtk_combo_box_get_active (GTK_COMBO_BOX (self->combo_box));
+
+ switch (CONSTRAINTS[active].aspect_ratio_type)
+ {
+ case PHOTOS_TOOL_CROP_ASPECT_RATIO_ANY:
+ if (self->crop_height > 0.0 && self->crop_width > 0.0)
+ ret_val = self->crop_width / self->crop_height;
+ else
+ g_assert_not_reached ();
+ break;
+
+ case PHOTOS_TOOL_CROP_ASPECT_RATIO_BASIS:
+ ret_val = (gdouble) CONSTRAINTS[active].basis_width / CONSTRAINTS[active].basis_height;
+ break;
+
+ case PHOTOS_TOOL_CROP_ASPECT_RATIO_ORIGINAL:
+ ret_val = (gdouble) self->bbox_source.width / self->bbox_source.height;
+ break;
+
+ case PHOTOS_TOOL_CROP_ASPECT_RATIO_SCREEN:
+ {
+ GdkScreen *screen;
+ gint height;
+ gint width;
+
+ screen = gdk_screen_get_default ();
+ height = gdk_screen_get_height (screen);
+ width = gdk_screen_get_width (screen);
+ ret_val = (gdouble) width / height;
+ break;
+ }
+
+ default:
+ g_assert_not_reached ();
+ }
+
+ return ret_val;
+}
+
+
+static void
+photos_tool_crop_set_crop (PhotosToolCrop *self, gdouble event_x, gdouble event_y)
+{
+ gdouble delta_x;
+ gdouble delta_y;
+
+ delta_x = event_x - self->event_x0;
+ delta_y = event_y - self->event_y0;
+
+ switch (self->location)
+ {
+ case PHOTOS_TOOL_CROP_LOCATION_NONE:
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_BOTTOM_LEFT:
+ self->crop_height = self->crop_height0 + delta_y;
+ self->crop_width = self->crop_width0 - delta_x;
+ self->crop_x = self->crop_x0 + delta_x;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_BOTTOM_RIGHT:
+ self->crop_height = self->crop_height0 + delta_y;
+ self->crop_width = self->crop_width0 + delta_x;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_BOTTOM_SIDE:
+ self->crop_height = self->crop_height0 + delta_y;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_CENTER:
+ self->crop_x = self->crop_x0 + delta_x;
+ self->crop_y = self->crop_y0 + delta_y;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_LEFT_SIDE:
+ self->crop_width = self->crop_width0 - delta_x;
+ self->crop_x = self->crop_x0 + delta_x;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_RIGHT_SIDE:
+ self->crop_width = self->crop_width0 + delta_x;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_TOP_LEFT:
+ self->crop_height = self->crop_height0 - delta_y;
+ self->crop_width = self->crop_width0 - delta_x;
+ self->crop_x = self->crop_x0 + delta_x;
+ self->crop_y = self->crop_y0 + delta_y;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_TOP_RIGHT:
+ self->crop_height = self->crop_height0 - delta_y;
+ self->crop_width = self->crop_width0 + delta_x;
+ self->crop_y = self->crop_y0 + delta_y;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_TOP_SIDE:
+ self->crop_height = self->crop_height0 - delta_y;
+ self->crop_y = self->crop_y0 + delta_y;
+ break;
+
+ default:
+ g_assert_not_reached ();
+ break;
+ }
+}
+
+
+static void
+photos_tool_crop_set_cursor (PhotosToolCrop *self)
+{
+ GdkCursor *cursor = NULL;
+ GdkCursorType cursor_type;
+ GdkDisplay *display;
+ GdkWindow *window;
+
+ window = gtk_widget_get_window (self->view);
+
+ switch (self->location)
+ {
+ case PHOTOS_TOOL_CROP_LOCATION_NONE:
+ goto set_cursor;
+
+ case PHOTOS_TOOL_CROP_LOCATION_BOTTOM_LEFT:
+ cursor_type = GDK_BOTTOM_LEFT_CORNER;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_BOTTOM_RIGHT:
+ cursor_type = GDK_BOTTOM_RIGHT_CORNER;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_BOTTOM_SIDE:
+ cursor_type = GDK_BOTTOM_SIDE;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_CENTER:
+ cursor_type = GDK_FLEUR;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_LEFT_SIDE:
+ cursor_type = GDK_LEFT_SIDE;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_RIGHT_SIDE:
+ cursor_type = GDK_RIGHT_SIDE;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_TOP_LEFT:
+ cursor_type = GDK_TOP_LEFT_CORNER;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_TOP_RIGHT:
+ cursor_type = GDK_TOP_RIGHT_CORNER;
+ break;
+
+ case PHOTOS_TOOL_CROP_LOCATION_TOP_SIDE:
+ cursor_type = GDK_TOP_SIDE;
+ break;
+
+ default:
+ g_assert_not_reached ();
+ break;
+ }
+
+ display = gdk_window_get_display (window);
+ cursor = gdk_cursor_new_for_display (display, cursor_type);
+
+ set_cursor:
+ gdk_window_set_cursor (window, cursor);
+ g_clear_object (&cursor);
+}
+
+
+static void
+photos_tool_crop_set_location (PhotosToolCrop *self, gdouble cur_x, gdouble cur_y)
+{
+ const gdouble edge_fuzz = 12.0;
+ gdouble crop_x;
+ gdouble crop_y;
+ gdouble x;
+ gdouble y;
+
+ self->location = PHOTOS_TOOL_CROP_LOCATION_NONE;
+
+ x = (gdouble) gegl_gtk_view_get_x (GEGL_GTK_VIEW (self->view));
+ y = (gdouble) gegl_gtk_view_get_y (GEGL_GTK_VIEW (self->view));
+ crop_x = self->crop_x - x;
+ crop_y = self->crop_y - y;
+
+ if (cur_x > crop_x - edge_fuzz
+ && cur_y > crop_y - edge_fuzz
+ && cur_x < crop_x + self->crop_width + edge_fuzz
+ && cur_y < crop_y + self->crop_height + edge_fuzz)
+ {
+ if (cur_x < crop_x + edge_fuzz && cur_y < crop_y + edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_TOP_LEFT;
+ else if (cur_x > crop_x + self->crop_width - edge_fuzz && cur_y < crop_y + edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_TOP_RIGHT;
+ else if (cur_x > crop_x + self->crop_width - edge_fuzz && cur_y > crop_y + self->crop_height -
edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_BOTTOM_RIGHT;
+ else if (cur_x < crop_x + edge_fuzz && cur_y > crop_y + self->crop_height - edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_BOTTOM_LEFT;
+ else if (cur_y < crop_y + edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_TOP_SIDE;
+ else if (cur_x > crop_x + self->crop_width - edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_RIGHT_SIDE;
+ else if (cur_y > crop_y + self->crop_height - edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_BOTTOM_SIDE;
+ else if (cur_x < crop_x + edge_fuzz)
+ self->location = PHOTOS_TOOL_CROP_LOCATION_LEFT_SIDE;
+ else
+ self->location = PHOTOS_TOOL_CROP_LOCATION_CENTER;
+ }
+}
+
+
+static void
+photos_tool_crop_surface_create (PhotosToolCrop *self)
+{
+ GdkWindow *window;
+ gfloat scale;
+
+ g_clear_pointer (&self->surface, (GDestroyNotify) cairo_surface_destroy);
+
+ window = gtk_widget_get_window (self->view);
+ scale = gegl_gtk_view_get_scale (GEGL_GTK_VIEW (self->view));
+ self->bbox_scaled.height = (gint) (scale * self->bbox_source.height + 0.5);
+ self->bbox_scaled.width = (gint) (scale * self->bbox_source.width + 0.5);
+ self->surface = gdk_window_create_similar_surface (window,
+ CAIRO_CONTENT_COLOR_ALPHA,
+ self->bbox_scaled.width,
+ self->bbox_scaled.height);
+}
+
+
+static void
+photos_tool_crop_surface_draw (PhotosToolCrop *self)
+{
+ cairo_t *cr;
+ const gdouble handle_offset = 3.0;
+ const gdouble handle_radius = 8.0;
+ gdouble one_third_x;
+ gdouble one_third_y;
+
+ cr = cairo_create (self->surface);
+ cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
+ cairo_set_source_rgba (cr, 0.0, 0.0, 0.0, 0.5);
+ cairo_paint (cr);
+
+ cairo_set_source_rgba (cr, 0.0, 0.0, 0.0, 0.0);
+ cairo_rectangle (cr, self->crop_x, self->crop_y, self->crop_width, self->crop_height);
+ cairo_fill (cr);
+
+ cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
+ cairo_set_source_rgba (cr, 0.25, 0.507, 0.828, 1.0);
+
+ cairo_new_sub_path (cr);
+ cairo_arc (cr, self->crop_x - handle_offset, self->crop_y - handle_offset, handle_radius, 0.0, 2.0 * M_PI);
+ cairo_fill (cr);
+
+ cairo_new_sub_path (cr);
+ cairo_arc (cr,
+ self->crop_x + self->crop_width + handle_offset,
+ self->crop_y - handle_offset,
+ handle_radius,
+ 0.0,
+ 2.0 * M_PI);
+ cairo_fill (cr);
+
+ cairo_new_sub_path (cr);
+ cairo_arc (cr,
+ self->crop_x + self->crop_width + handle_offset,
+ self->crop_y + self->crop_height + handle_offset,
+ handle_radius,
+ 0.0,
+ 2.0 * M_PI);
+ cairo_fill (cr);
+
+ cairo_new_sub_path (cr);
+ cairo_arc (cr,
+ self->crop_x - handle_offset,
+ self->crop_y + self->crop_height + handle_offset,
+ handle_radius,
+ 0.0,
+ 2.0 * M_PI);
+ cairo_fill (cr);
+
+ cairo_set_source_rgba (cr, 0.8, 0.8, 0.8, 1.0);
+ cairo_set_line_width (cr, 0.5);
+ one_third_x = self->crop_width / 3.0;
+ one_third_y = self->crop_height / 3.0;
+
+ cairo_move_to (cr, self->crop_x + one_third_x, self->crop_y);
+ cairo_line_to (cr, self->crop_x + one_third_x, self->crop_y + self->crop_height);
+ cairo_stroke (cr);
+
+ cairo_move_to (cr, self->crop_x + 2.0 * one_third_x, self->crop_y);
+ cairo_line_to (cr, self->crop_x + 2.0 * one_third_x, self->crop_y + self->crop_height);
+ cairo_stroke (cr);
+
+ cairo_move_to (cr, self->crop_x, self->crop_y + one_third_y);
+ cairo_line_to (cr, self->crop_x + self->crop_width, self->crop_y + one_third_y);
+ cairo_stroke (cr);
+
+ cairo_move_to (cr, self->crop_x, self->crop_y + 2.0 * one_third_y);
+ cairo_line_to (cr, self->crop_x + self->crop_width, self->crop_y + 2.0 * one_third_y);
+ cairo_stroke (cr);
+
+ cairo_destroy (cr);
+}
+
+
+static void
+photos_tool_crop_surface_reset (PhotosToolCrop *self)
+{
+ gdouble crop_aspect_ratio;
+ gdouble aspect_ratio;
+
+ aspect_ratio = (gdouble) self->bbox_source.width / self->bbox_source.height;
+ crop_aspect_ratio = photos_tool_crop_calculate_aspect_ratio (self);
+
+ if (crop_aspect_ratio < aspect_ratio)
+ {
+ self->crop_height = 0.7 * self->bbox_scaled.height;
+ self->crop_width = self->crop_height * crop_aspect_ratio;
+ }
+ else
+ {
+ self->crop_width = 0.7 * self->bbox_scaled.width;
+ self->crop_height = self->crop_width / crop_aspect_ratio;
+ }
+
+ self->crop_x = ((gdouble) self->bbox_scaled.width - self->crop_width) / 2.0;
+ self->crop_y = ((gdouble) self->bbox_scaled.height - self->crop_height) / 2.0;
+
+ photos_tool_crop_surface_draw (self);
+}
+
+
+static void
+photos_tool_crop_changed (PhotosToolCrop *self)
+{
+ photos_tool_crop_surface_reset (self);
+ gtk_widget_queue_draw (self->view);
+}
+
+
+static void
+photos_tool_crop_surface_resize (PhotosToolCrop *self)
+{
+ gdouble crop_aspect_ratio;
+ gdouble aspect_ratio;
+
+ aspect_ratio = (gdouble) self->bbox_source.width / self->bbox_source.height;
+ crop_aspect_ratio = photos_tool_crop_calculate_aspect_ratio (self);
+
+ if (crop_aspect_ratio < aspect_ratio)
+ {
+ self->crop_height = 0.7 * self->bbox_scaled.height;
+ self->crop_width = self->crop_height * crop_aspect_ratio;
+ }
+ else
+ {
+ self->crop_width = 0.7 * self->bbox_scaled.width;
+ self->crop_height = self->crop_width / crop_aspect_ratio;
+ }
+
+ self->crop_x = ((gdouble) self->bbox_scaled.width - self->crop_width) / 2.0;
+ self->crop_y = ((gdouble) self->bbox_scaled.height - self->crop_height) / 2.0;
+
+ photos_tool_crop_surface_draw (self);
+}
+
+
+static void
+photos_tool_crop_size_allocate (PhotosToolCrop *self, GdkRectangle *allocation)
+{
+ gdouble crop_height_ratio;
+ gdouble crop_width_ratio;
+ gdouble crop_x_ratio;
+ gdouble crop_y_ratio;
+
+ crop_height_ratio = self->crop_height / self->bbox_scaled.height;
+ crop_width_ratio = self->crop_width / self->bbox_scaled.width;
+ crop_x_ratio = self->crop_x / self->bbox_scaled.width;
+ crop_y_ratio = self->crop_y / self->bbox_scaled.height;
+
+ photos_tool_crop_surface_create (self);
+
+ self->crop_height = crop_height_ratio * self->bbox_scaled.height;
+ self->crop_width = crop_width_ratio * self->bbox_scaled.width;
+ self->crop_x = crop_x_ratio * self->bbox_scaled.width;
+ self->crop_y = crop_y_ratio * self->bbox_scaled.height;
+
+ photos_tool_crop_surface_draw (self);
+}
+
+
+static void
+photos_tool_crop_activate (PhotosTool *tool, PhotosBaseItem *item, GeglGtkView *view)
+{
+ PhotosToolCrop *self = PHOTOS_TOOL_CROP (tool);
+
+ if (!photos_base_item_get_bbox_source (item, &self->bbox_source))
+ g_assert_not_reached ();
+
+ self->view = GTK_WIDGET (view);
+ g_signal_connect_swapped (self->view, "size-allocate", G_CALLBACK (photos_tool_crop_size_allocate), self);
+
+ photos_tool_crop_surface_create (self);
+ photos_tool_crop_surface_reset (self);
+}
+
+
+static void
+photos_tool_crop_draw (PhotosTool *tool, cairo_t *cr, GdkRectangle *rect)
+{
+ PhotosToolCrop *self = PHOTOS_TOOL_CROP (tool);
+ gdouble x;
+ gdouble y;
+
+ x = (gdouble) gegl_gtk_view_get_x (GEGL_GTK_VIEW (self->view));
+ x = -x;
+
+ y = (gdouble) gegl_gtk_view_get_y (GEGL_GTK_VIEW (self->view));
+ y = -y;
+
+ cairo_save (cr);
+ cairo_set_source_surface (cr, self->surface, x, y);
+ cairo_paint (cr);
+ cairo_restore (cr);
+}
+
+
+static GtkWidget *
+photos_tool_crop_get_widget (PhotosTool *tool)
+{
+ PhotosToolCrop *self = PHOTOS_TOOL_CROP (tool);
+ return self->combo_box;
+}
+
+
+static gboolean
+photos_tool_crop_left_click_event (PhotosTool *tool, GdkEventButton *event)
+{
+ PhotosToolCrop *self = PHOTOS_TOOL_CROP (tool);
+
+ self->grabbed = TRUE;
+ self->crop_height0 = self->crop_height;
+ self->crop_width0 = self->crop_width;
+ self->crop_x0 = self->crop_x;
+ self->crop_y0 = self->crop_y;
+ self->event_x0 = event->x;
+ self->event_y0 = event->y;
+
+ return GDK_EVENT_PROPAGATE;
+}
+
+
+static gboolean
+photos_tool_crop_left_unclick_event (PhotosTool *tool, GdkEventButton *event)
+{
+ PhotosToolCrop *self = PHOTOS_TOOL_CROP (tool);
+
+ self->grabbed = FALSE;
+ return GDK_EVENT_PROPAGATE;
+}
+
+
+static gboolean
+photos_tool_crop_motion_event (PhotosTool *tool, GdkEventMotion *event)
+{
+ PhotosToolCrop *self = PHOTOS_TOOL_CROP (tool);
+ const gdouble edge_fuzz = 12.0;
+ gdouble crop_x;
+ gdouble crop_y;
+ gdouble x;
+ gdouble y;
+
+ x = (gdouble) gegl_gtk_view_get_x (GEGL_GTK_VIEW (self->view));
+ y = (gdouble) gegl_gtk_view_get_y (GEGL_GTK_VIEW (self->view));
+ crop_x = self->crop_x - x;
+ crop_y = self->crop_y - y;
+
+ if (self->grabbed)
+ {
+ photos_tool_crop_set_crop (self, event->x, event->y);
+ photos_tool_crop_surface_draw (self);
+ gtk_widget_queue_draw (self->view);
+ }
+ else
+ {
+ photos_tool_crop_set_location (self, event->x, event->y);
+ photos_tool_crop_set_cursor (self);
+ }
+
+ return GDK_EVENT_STOP;
+}
+
+
+static void
+photos_tool_crop_dispose (GObject *object)
+{
+ PhotosToolCrop *self = PHOTOS_TOOL_CROP (object);
+
+ g_clear_object (&self->model);
+ g_clear_object (&self->combo_box);
+ g_clear_pointer (&self->surface, (GDestroyNotify) cairo_surface_destroy);
+
+ G_OBJECT_CLASS (photos_tool_crop_parent_class)->dispose (object);
+}
+
+
+static void
+photos_tool_crop_init (PhotosToolCrop *self)
+{
+ GtkCellRenderer *renderer;
+ guint i;
+
+ self->model = gtk_list_store_new (4, G_TYPE_INT, G_TYPE_STRING, G_TYPE_UINT, G_TYPE_UINT);
+
+ for (i = 0; i < G_N_ELEMENTS (CONSTRAINTS); i++)
+ {
+ GtkTreeIter iter;
+
+ gtk_list_store_append (self->model, &iter);
+ gtk_list_store_set (self->model,
+ &iter,
+ CONSTRAINT_COLUMN_ASPECT_RATIO, CONSTRAINTS[i].aspect_ratio_type,
+ CONSTRAINT_COLUMN_NAME, CONSTRAINTS[i].name,
+ CONSTRAINT_COLUMN_BASIS_HEIGHT, CONSTRAINTS[i].basis_height,
+ CONSTRAINT_COLUMN_BASIS_WIDTH, CONSTRAINTS[i].basis_width,
+ -1);
+ }
+
+ self->combo_box = g_object_ref_sink (gtk_combo_box_new_with_model (GTK_TREE_MODEL (self->model)));
+ gtk_combo_box_set_active (GTK_COMBO_BOX (self->combo_box), 1);
+ g_signal_connect_swapped (self->combo_box, "changed", G_CALLBACK (photos_tool_crop_changed), self);
+
+ renderer = gtk_cell_renderer_text_new ();
+ gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (self->combo_box), renderer, TRUE);
+ gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (self->combo_box), renderer, "text",
CONSTRAINT_COLUMN_NAME);
+}
+
+
+static void
+photos_tool_crop_class_init (PhotosToolCropClass *class)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (class);
+ PhotosToolClass *tool_class = PHOTOS_TOOL_CLASS (class);
+
+ tool_class->icon_name = PHOTOS_ICON_IMAGE_CROP_SYMBOLIC;
+ tool_class->name = _("Crop");
+
+ object_class->dispose = photos_tool_crop_dispose;
+ tool_class->activate = photos_tool_crop_activate;
+ tool_class->draw = photos_tool_crop_draw;
+ tool_class->get_widget = photos_tool_crop_get_widget;
+ tool_class->left_click_event = photos_tool_crop_left_click_event;
+ tool_class->left_unclick_event = photos_tool_crop_left_unclick_event;
+ tool_class->motion_event = photos_tool_crop_motion_event;
+}
diff --git a/src/photos-tool-crop.h b/src/photos-tool-crop.h
new file mode 100644
index 0000000..bd921d5
--- /dev/null
+++ b/src/photos-tool-crop.h
@@ -0,0 +1,45 @@
+/*
+ * Photos - access, organize and share your photos on GNOME
+ * Copyright © 2015 Red Hat, Inc.
+ *
+ * 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 2
+ * 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, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+ * 02110-1301, USA.
+ */
+
+#ifndef PHOTOS_TOOL_CROP_H
+#define PHOTOS_TOOL_CROP_H
+
+#include <glib-object.h>
+
+G_BEGIN_DECLS
+
+#define PHOTOS_TYPE_TOOL_CROP (photos_tool_crop_get_type ())
+
+#define PHOTOS_TOOL_CROP(obj) \
+ (G_TYPE_CHECK_INSTANCE_CAST ((obj), \
+ PHOTOS_TYPE_TOOL_CROP, PhotosToolCrop))
+
+#define PHOTOS_IS_TOOL_CROP(obj) \
+ (G_TYPE_CHECK_INSTANCE_TYPE ((obj), \
+ PHOTOS_TYPE_TOOL_CROP))
+
+typedef struct _PhotosToolCrop PhotosToolCrop;
+typedef struct _PhotosToolCropClass PhotosToolCropClass;
+
+GType photos_tool_crop_get_type (void) G_GNUC_CONST;
+
+G_END_DECLS
+
+#endif /* PHOTOS_TOOL_CROP_H */
diff --git a/src/photos-utils.c b/src/photos-utils.c
index 80aab65..8aee7fb 100644
--- a/src/photos-utils.c
+++ b/src/photos-utils.c
@@ -46,6 +46,7 @@
#include "photos-query.h"
#include "photos-source.h"
#include "photos-tool.h"
+#include "photos-tool-crop.h"
#include "photos-tool-filters.h"
#include "photos-tool-sharpen.h"
#include "photos-tracker-queue.h"
@@ -538,6 +539,7 @@ photos_utils_ensure_builtins (void)
g_type_ensure (PHOTOS_TYPE_OPERATION_INSTA_CURVE);
g_type_ensure (PHOTOS_TYPE_OPERATION_INSTA_FILTER);
+ g_type_ensure (PHOTOS_TYPE_TOOL_CROP);
g_type_ensure (PHOTOS_TYPE_TOOL_FILTERS);
g_type_ensure (PHOTOS_TYPE_TOOL_SHARPEN);
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]