/* -*- 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;
}