/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Pix * * Copyright (C) 2011 Free Software Foundation, 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, see . */ #include #include #include #include #include "gth-image-line-tool.h" #define MIN4(a,b,c,d) MIN(MIN((a),(b)),MIN((c),(d))) #define MAX4(a,b,c,d) MAX(MAX((a),(b)),MAX((c),(d))) enum { CHANGED, LAST_SIGNAL }; struct _GthImageLineToolPrivate { GthImageViewer *viewer; /* options */ GdkPoint p1; GdkPoint p2; /* utility variables */ int original_width; int original_height; double preview_zoom; cairo_surface_t *preview_image; cairo_rectangle_int_t preview_image_area; GdkPoint preview_center; cairo_rectangle_int_t clip_area; cairo_matrix_t matrix; gboolean first_point_set; GthFit original_fit_mode; gboolean original_zoom_enabled; }; static guint signals[LAST_SIGNAL] = { 0 }; static void gth_image_line_tool_gth_image_tool_interface_init (GthImageViewerToolInterface *iface); G_DEFINE_TYPE_WITH_CODE (GthImageLineTool, gth_image_line_tool, G_TYPE_OBJECT, G_ADD_PRIVATE (GthImageLineTool) G_IMPLEMENT_INTERFACE (GTH_TYPE_IMAGE_VIEWER_TOOL, gth_image_line_tool_gth_image_tool_interface_init)) static void gth_image_line_tool_set_viewer (GthImageViewerTool *base, GthImageViewer *viewer) { GthImageLineTool *self = GTH_IMAGE_LINE_TOOL (base); GdkCursor *cursor; self->priv->viewer = viewer; self->priv->original_fit_mode = gth_image_viewer_get_fit_mode (GTH_IMAGE_VIEWER (viewer)); self->priv->original_zoom_enabled = gth_image_viewer_get_zoom_enabled (GTH_IMAGE_VIEWER (viewer)); gth_image_viewer_set_fit_mode (GTH_IMAGE_VIEWER (viewer), GTH_FIT_SIZE_IF_LARGER); gth_image_viewer_set_zoom_enabled (GTH_IMAGE_VIEWER (viewer), FALSE); self->priv->first_point_set = FALSE; cursor = _gdk_cursor_new_for_widget (GTK_WIDGET (self->priv->viewer), GDK_CROSSHAIR); gth_image_viewer_set_cursor (self->priv->viewer, cursor); g_object_unref (cursor); } static void gth_image_line_tool_unset_viewer (GthImageViewerTool *base, GthImageViewer *viewer) { GthImageLineTool *self = GTH_IMAGE_LINE_TOOL (base); gth_image_viewer_set_fit_mode (GTH_IMAGE_VIEWER (viewer), self->priv->original_fit_mode); gth_image_viewer_set_zoom_enabled (GTH_IMAGE_VIEWER (viewer), self->priv->original_zoom_enabled); self->priv->viewer = NULL; self->priv->first_point_set = FALSE; } static void gth_image_line_tool_realize (GthImageViewerTool *base) { /* void */ } static void gth_image_line_tool_unrealize (GthImageViewerTool *base) { /* void */ } static void update_image_surface (GthImageLineTool *self) { GtkAllocation allocation; cairo_surface_t *image; int max_size; int width; int height; cairo_surface_t *preview_image; if (self->priv->preview_image != NULL) { cairo_surface_destroy (self->priv->preview_image); self->priv->preview_image = NULL; } image = gth_image_viewer_get_current_image (GTH_IMAGE_VIEWER (self->priv->viewer)); if (image == NULL) return; self->priv->original_width = cairo_image_surface_get_width (image); self->priv->original_height = cairo_image_surface_get_height (image); width = self->priv->original_width; height = self->priv->original_height; gtk_widget_get_allocation (GTK_WIDGET (self->priv->viewer), &allocation); max_size = MAX (allocation.width, allocation.height) / G_SQRT2 + 2; if (scale_keeping_ratio (&width, &height, max_size, max_size, FALSE)) preview_image = _cairo_image_surface_scale_fast (image, width, height); else preview_image = cairo_surface_reference (image); self->priv->preview_zoom = (double) width / self->priv->original_width; self->priv->preview_image = preview_image; self->priv->preview_image_area.width = width; self->priv->preview_image_area.height = height; self->priv->preview_image_area.x = MAX ((allocation.width - self->priv->preview_image_area.width) / 2 - 0.5, 0); self->priv->preview_image_area.y = MAX ((allocation.height - self->priv->preview_image_area.height) / 2 - 0.5, 0); } static void gth_image_line_tool_size_allocate (GthImageViewerTool *base, GtkAllocation *allocation) { update_image_surface (GTH_IMAGE_LINE_TOOL (base)); } static void gth_image_line_tool_map (GthImageViewerTool *base) { /* void */ } static void gth_image_line_tool_unmap (GthImageViewerTool *base) { /* void */ } static void paint_image (GthImageLineTool *self, cairo_t *cr) { cairo_save (cr); cairo_set_antialias (cr, CAIRO_ANTIALIAS_NONE); cairo_set_source_surface (cr, self->priv->preview_image, self->priv->preview_image_area.x, self->priv->preview_image_area.y); cairo_pattern_set_filter (cairo_get_source (cr), CAIRO_FILTER_FAST); cairo_rectangle (cr, self->priv->preview_image_area.x, self->priv->preview_image_area.y, self->priv->preview_image_area.width, self->priv->preview_image_area.height); cairo_fill (cr); cairo_restore (cr); } static void gth_image_line_tool_draw (GthImageViewerTool *base, cairo_t *cr) { GthImageLineTool *self = GTH_IMAGE_LINE_TOOL (base); if (self->priv->preview_image == NULL) return; cairo_save (cr); paint_image (self, cr); if (self->priv->first_point_set) { #if CAIRO_VERSION >= CAIRO_VERSION_ENCODE(1, 9, 2) cairo_set_operator (cr, CAIRO_OPERATOR_DIFFERENCE); #endif cairo_set_line_width (cr, 5.0); cairo_set_antialias (cr, CAIRO_ANTIALIAS_DEFAULT); cairo_set_source_rgb (cr, 1.0, 1.0, 1.0); cairo_translate (cr, self->priv->preview_image_area.x, self->priv->preview_image_area.y); cairo_scale (cr, self->priv->preview_zoom, self->priv->preview_zoom); cairo_move_to (cr, self->priv->p1.x, self->priv->p1.y); cairo_line_to (cr, self->priv->p2.x, self->priv->p2.y); cairo_stroke (cr); } cairo_restore (cr); } static gboolean gth_image_line_tool_button_release (GthImageViewerTool *base, GdkEventButton *event) { GthImageLineTool *self = GTH_IMAGE_LINE_TOOL (base); g_signal_emit (self, signals[CHANGED], 0); return FALSE; } static gboolean gth_image_line_tool_button_press (GthImageViewerTool *base, GdkEventButton *event) { GthImageLineTool *self = GTH_IMAGE_LINE_TOOL (base); if (event->type == GDK_BUTTON_PRESS) { self->priv->p1.x = self->priv->p2.x = (event->x - self->priv->preview_image_area.x) / self->priv->preview_zoom; self->priv->p1.y = self->priv->p2.y = (event->y - self->priv->preview_image_area.y) / self->priv->preview_zoom; self->priv->first_point_set = TRUE; } return FALSE; } static gboolean gth_image_line_tool_motion_notify (GthImageViewerTool *base, GdkEventMotion *event) { GthImageLineTool *self = GTH_IMAGE_LINE_TOOL (base); if (! self->priv->first_point_set) return FALSE; self->priv->p2.x = (event->x - self->priv->preview_image_area.x) / self->priv->preview_zoom; self->priv->p2.y = (event->y - self->priv->preview_image_area.y) / self->priv->preview_zoom; gtk_widget_queue_draw (GTK_WIDGET (self->priv->viewer)); return FALSE; } static void gth_image_line_tool_image_changed (GthImageViewerTool *base) { update_image_surface (GTH_IMAGE_LINE_TOOL (base)); } static void gth_image_line_tool_zoom_changed (GthImageViewerTool *base) { update_image_surface (GTH_IMAGE_LINE_TOOL (base)); } static void gth_image_line_tool_gth_image_tool_interface_init (GthImageViewerToolInterface *iface) { iface->set_viewer = gth_image_line_tool_set_viewer; iface->unset_viewer = gth_image_line_tool_unset_viewer; iface->realize = gth_image_line_tool_realize; iface->unrealize = gth_image_line_tool_unrealize; iface->size_allocate = gth_image_line_tool_size_allocate; iface->map = gth_image_line_tool_map; iface->unmap = gth_image_line_tool_unmap; iface->draw = gth_image_line_tool_draw; iface->button_press = gth_image_line_tool_button_press; iface->button_release = gth_image_line_tool_button_release; iface->motion_notify = gth_image_line_tool_motion_notify; iface->image_changed = gth_image_line_tool_image_changed; iface->zoom_changed = gth_image_line_tool_zoom_changed; } static void gth_image_line_tool_finalize (GObject *object) { GthImageLineTool *self; g_return_if_fail (object != NULL); g_return_if_fail (GTH_IS_IMAGE_LINE_TOOL (object)); self = (GthImageLineTool *) object; if (self->priv->preview_image != NULL) cairo_surface_destroy (self->priv->preview_image); /* Chain up */ G_OBJECT_CLASS (gth_image_line_tool_parent_class)->finalize (object); } static void gth_image_line_tool_class_init (GthImageLineToolClass *class) { GObjectClass *gobject_class; gobject_class = (GObjectClass*) class; gobject_class->finalize = gth_image_line_tool_finalize; signals[CHANGED] = g_signal_new ("changed", G_TYPE_FROM_CLASS (class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GthImageLineToolClass, changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } static void gth_image_line_tool_init (GthImageLineTool *self) { self->priv = gth_image_line_tool_get_instance_private (self); self->priv->preview_image = NULL; self->priv->first_point_set = FALSE; } GthImageViewerTool * gth_image_line_tool_new (void) { return (GthImageViewerTool *) g_object_new (GTH_TYPE_IMAGE_LINE_TOOL, NULL); } void gth_image_line_tool_get_points (GthImageLineTool *self, GdkPoint *p1, GdkPoint *p2) { *p1 = self->priv->p1; *p2 = self->priv->p2; }