mirror of https://github.com/GNOME/gimp.git
452 lines
15 KiB
C
452 lines
15 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* gimpcanvasprogress.c
|
|
* Copyright (C) 2010 Michael Natterer <mitch@gimp.org>
|
|
*
|
|
* 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 "libgimpbase/gimpbase.h"
|
|
#include "libgimpmath/gimpmath.h"
|
|
|
|
#include "display-types.h"
|
|
|
|
#include "core/gimpprogress.h"
|
|
|
|
#include "gimpcanvas.h"
|
|
#include "gimpcanvas-style.h"
|
|
#include "gimpcanvasitem-utils.h"
|
|
#include "gimpcanvasprogress.h"
|
|
#include "gimpdisplayshell.h"
|
|
|
|
|
|
#define BORDER 5
|
|
#define RADIUS 20
|
|
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_ANCHOR,
|
|
PROP_X,
|
|
PROP_Y
|
|
};
|
|
|
|
|
|
typedef struct _GimpCanvasProgressPrivate GimpCanvasProgressPrivate;
|
|
|
|
struct _GimpCanvasProgressPrivate
|
|
{
|
|
GimpHandleAnchor anchor;
|
|
gdouble x;
|
|
gdouble y;
|
|
|
|
gchar *text;
|
|
gdouble value;
|
|
};
|
|
|
|
#define GET_PRIVATE(progress) \
|
|
G_TYPE_INSTANCE_GET_PRIVATE (progress, \
|
|
GIMP_TYPE_CANVAS_PROGRESS, \
|
|
GimpCanvasProgressPrivate)
|
|
|
|
|
|
/* local function prototypes */
|
|
|
|
static void gimp_canvas_progress_iface_init (GimpProgressInterface *iface);
|
|
|
|
static void gimp_canvas_progress_finalize (GObject *object);
|
|
static void gimp_canvas_progress_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_canvas_progress_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_canvas_progress_draw (GimpCanvasItem *item,
|
|
cairo_t *cr);
|
|
static cairo_region_t * gimp_canvas_progress_get_extents (GimpCanvasItem *item);
|
|
static gboolean gimp_canvas_progress_hit (GimpCanvasItem *item,
|
|
gdouble x,
|
|
gdouble y);
|
|
|
|
static GimpProgress * gimp_canvas_progress_start (GimpProgress *progress,
|
|
gboolean cancellable,
|
|
const gchar *message);
|
|
static void gimp_canvas_progress_end (GimpProgress *progress);
|
|
static gboolean gimp_canvas_progress_is_active (GimpProgress *progress);
|
|
static void gimp_canvas_progress_set_text (GimpProgress *progress,
|
|
const gchar *message);
|
|
static void gimp_canvas_progress_set_value (GimpProgress *progress,
|
|
gdouble percentage);
|
|
static gdouble gimp_canvas_progress_get_value (GimpProgress *progress);
|
|
static void gimp_canvas_progress_pulse (GimpProgress *progress);
|
|
static gboolean gimp_canvas_progress_message (GimpProgress *progress,
|
|
Gimp *gimp,
|
|
GimpMessageSeverity severity,
|
|
const gchar *domain,
|
|
const gchar *message);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GimpCanvasProgress, gimp_canvas_progress,
|
|
GIMP_TYPE_CANVAS_ITEM,
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_PROGRESS,
|
|
gimp_canvas_progress_iface_init))
|
|
|
|
#define parent_class gimp_canvas_progress_parent_class
|
|
|
|
|
|
static void
|
|
gimp_canvas_progress_class_init (GimpCanvasProgressClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GimpCanvasItemClass *item_class = GIMP_CANVAS_ITEM_CLASS (klass);
|
|
|
|
object_class->finalize = gimp_canvas_progress_finalize;
|
|
object_class->set_property = gimp_canvas_progress_set_property;
|
|
object_class->get_property = gimp_canvas_progress_get_property;
|
|
|
|
item_class->draw = gimp_canvas_progress_draw;
|
|
item_class->get_extents = gimp_canvas_progress_get_extents;
|
|
item_class->hit = gimp_canvas_progress_hit;
|
|
|
|
g_object_class_install_property (object_class, PROP_ANCHOR,
|
|
g_param_spec_enum ("anchor", NULL, NULL,
|
|
GIMP_TYPE_HANDLE_ANCHOR,
|
|
GIMP_HANDLE_ANCHOR_CENTER,
|
|
GIMP_PARAM_READWRITE));
|
|
|
|
g_object_class_install_property (object_class, PROP_X,
|
|
g_param_spec_double ("x", NULL, NULL,
|
|
-GIMP_MAX_IMAGE_SIZE,
|
|
GIMP_MAX_IMAGE_SIZE, 0,
|
|
GIMP_PARAM_READWRITE));
|
|
|
|
g_object_class_install_property (object_class, PROP_Y,
|
|
g_param_spec_double ("y", NULL, NULL,
|
|
-GIMP_MAX_IMAGE_SIZE,
|
|
GIMP_MAX_IMAGE_SIZE, 0,
|
|
GIMP_PARAM_READWRITE));
|
|
|
|
g_type_class_add_private (klass, sizeof (GimpCanvasProgressPrivate));
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_iface_init (GimpProgressInterface *iface)
|
|
{
|
|
iface->start = gimp_canvas_progress_start;
|
|
iface->end = gimp_canvas_progress_end;
|
|
iface->is_active = gimp_canvas_progress_is_active;
|
|
iface->set_text = gimp_canvas_progress_set_text;
|
|
iface->set_value = gimp_canvas_progress_set_value;
|
|
iface->get_value = gimp_canvas_progress_get_value;
|
|
iface->pulse = gimp_canvas_progress_pulse;
|
|
iface->message = gimp_canvas_progress_message;
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_init (GimpCanvasProgress *progress)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_finalize (GObject *object)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (object);
|
|
|
|
if (private->text)
|
|
{
|
|
g_free (private->text);
|
|
private->text = NULL;
|
|
}
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_ANCHOR:
|
|
private->anchor = g_value_get_enum (value);
|
|
break;
|
|
case PROP_X:
|
|
private->x = g_value_get_double (value);
|
|
break;
|
|
case PROP_Y:
|
|
private->y = g_value_get_double (value);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_ANCHOR:
|
|
g_value_set_enum (value, private->anchor);
|
|
break;
|
|
case PROP_X:
|
|
g_value_set_double (value, private->x);
|
|
break;
|
|
case PROP_Y:
|
|
g_value_set_double (value, private->y);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static PangoLayout *
|
|
gimp_canvas_progress_transform (GimpCanvasItem *item,
|
|
gdouble *x,
|
|
gdouble *y,
|
|
gint *width,
|
|
gint *height)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (item);
|
|
GtkWidget *canvas = gimp_canvas_item_get_canvas (item);
|
|
PangoLayout *layout;
|
|
|
|
layout = gimp_canvas_get_layout (GIMP_CANVAS (canvas), "%s",
|
|
private->text);
|
|
|
|
pango_layout_get_pixel_size (layout, width, height);
|
|
|
|
*width = MAX (*width, 2 * RADIUS);
|
|
|
|
*width += 2 * BORDER;
|
|
*height += 3 * BORDER + 2 * RADIUS;
|
|
|
|
gimp_canvas_item_transform_xy_f (item,
|
|
private->x, private->y,
|
|
x, y);
|
|
|
|
gimp_canvas_item_shift_to_north_west (private->anchor,
|
|
*x, *y,
|
|
*width,
|
|
*height,
|
|
x, y);
|
|
|
|
*x = floor (*x);
|
|
*y = floor (*y);
|
|
|
|
return layout;
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_draw (GimpCanvasItem *item,
|
|
cairo_t *cr)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (item);
|
|
GtkWidget *canvas = gimp_canvas_item_get_canvas (item);
|
|
gdouble x, y;
|
|
gint width, height;
|
|
|
|
gimp_canvas_progress_transform (item, &x, &y, &width, &height);
|
|
|
|
cairo_move_to (cr, x, y);
|
|
cairo_line_to (cr, x + width, y);
|
|
cairo_line_to (cr, x + width, y + height - BORDER - 2 * RADIUS);
|
|
cairo_line_to (cr, x + 2 * BORDER + 2 * RADIUS, y + height - BORDER - 2 * RADIUS);
|
|
cairo_arc (cr, x + BORDER + RADIUS, y + height - BORDER - RADIUS,
|
|
BORDER + RADIUS, 0, G_PI);
|
|
cairo_close_path (cr);
|
|
|
|
_gimp_canvas_item_fill (item, cr);
|
|
|
|
cairo_move_to (cr, x + BORDER, y + BORDER);
|
|
cairo_set_source_rgba (cr, 0.0, 0.0, 0.0, 1.0);
|
|
pango_cairo_show_layout (cr,
|
|
gimp_canvas_get_layout (GIMP_CANVAS (canvas),
|
|
"%s", private->text));
|
|
|
|
gimp_canvas_set_tool_bg_style (gimp_canvas_item_get_canvas (item), cr);
|
|
cairo_arc (cr, x + BORDER + RADIUS, y + height - BORDER - RADIUS,
|
|
RADIUS, - G_PI / 2.0, 2 * G_PI - G_PI / 2.0);
|
|
cairo_fill (cr);
|
|
|
|
cairo_set_source_rgba (cr, 0.0, 1.0, 0.0, 1.0);
|
|
cairo_move_to (cr, x + BORDER + RADIUS, y + height - BORDER - RADIUS);
|
|
cairo_arc (cr, x + BORDER + RADIUS, y + height - BORDER - RADIUS,
|
|
RADIUS, - G_PI / 2.0, 2 * G_PI * private->value - G_PI / 2.0);
|
|
cairo_fill (cr);
|
|
}
|
|
|
|
static cairo_region_t *
|
|
gimp_canvas_progress_get_extents (GimpCanvasItem *item)
|
|
{
|
|
cairo_rectangle_int_t rectangle;
|
|
gdouble x, y;
|
|
gint width, height;
|
|
|
|
gimp_canvas_progress_transform (item, &x, &y, &width, &height);
|
|
|
|
/* add 1px on each side because fill()'s default impl does the same */
|
|
rectangle.x = (gint) x - 1;
|
|
rectangle.y = (gint) y - 1;
|
|
rectangle.width = width + 2;
|
|
rectangle.height = height + 2;
|
|
|
|
return cairo_region_create_rectangle (&rectangle);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_canvas_progress_hit (GimpCanvasItem *item,
|
|
gdouble x,
|
|
gdouble y)
|
|
{
|
|
gdouble px, py;
|
|
gint pwidth, pheight;
|
|
gdouble tx, ty;
|
|
|
|
gimp_canvas_progress_transform (item, &px, &py, &pwidth, &pheight);
|
|
gimp_canvas_item_transform_xy_f (item, x, y, &tx, &ty);
|
|
|
|
pheight -= BORDER + 2 * RADIUS;
|
|
|
|
return (tx >= px && tx < (px + pwidth) &&
|
|
ty >= py && ty < (py + pheight));
|
|
}
|
|
|
|
static GimpProgress *
|
|
gimp_canvas_progress_start (GimpProgress *progress,
|
|
gboolean cancellable,
|
|
const gchar *message)
|
|
{
|
|
gimp_canvas_progress_set_text (progress, message);
|
|
|
|
return progress;
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_end (GimpProgress *progress)
|
|
{
|
|
}
|
|
|
|
static gboolean
|
|
gimp_canvas_progress_is_active (GimpProgress *progress)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_set_text (GimpProgress *progress,
|
|
const gchar *message)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (progress);
|
|
cairo_region_t *old_region;
|
|
cairo_region_t *new_region;
|
|
|
|
old_region = gimp_canvas_item_get_extents (GIMP_CANVAS_ITEM (progress));
|
|
|
|
if (private->text)
|
|
g_free (private->text);
|
|
|
|
private->text = g_strdup (message);
|
|
|
|
new_region = gimp_canvas_item_get_extents (GIMP_CANVAS_ITEM (progress));
|
|
|
|
cairo_region_union (new_region, old_region);
|
|
cairo_region_destroy (old_region);
|
|
|
|
_gimp_canvas_item_update (GIMP_CANVAS_ITEM (progress), new_region);
|
|
|
|
cairo_region_destroy (new_region);
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_set_value (GimpProgress *progress,
|
|
gdouble percentage)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (progress);
|
|
|
|
if (percentage != private->value)
|
|
{
|
|
cairo_region_t *region;
|
|
|
|
private->value = percentage;
|
|
|
|
region = gimp_canvas_item_get_extents (GIMP_CANVAS_ITEM (progress));
|
|
|
|
_gimp_canvas_item_update (GIMP_CANVAS_ITEM (progress), region);
|
|
|
|
cairo_region_destroy (region);
|
|
}
|
|
}
|
|
|
|
static gdouble
|
|
gimp_canvas_progress_get_value (GimpProgress *progress)
|
|
{
|
|
GimpCanvasProgressPrivate *private = GET_PRIVATE (progress);
|
|
|
|
return private->value;
|
|
}
|
|
|
|
static void
|
|
gimp_canvas_progress_pulse (GimpProgress *progress)
|
|
{
|
|
}
|
|
|
|
static gboolean
|
|
gimp_canvas_progress_message (GimpProgress *progress,
|
|
Gimp *gimp,
|
|
GimpMessageSeverity severity,
|
|
const gchar *domain,
|
|
const gchar *message)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
GimpCanvasItem *
|
|
gimp_canvas_progress_new (GimpDisplayShell *shell,
|
|
GimpHandleAnchor anchor,
|
|
gdouble x,
|
|
gdouble y)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_DISPLAY_SHELL (shell), NULL);
|
|
|
|
return g_object_new (GIMP_TYPE_CANVAS_PROGRESS,
|
|
"shell", shell,
|
|
"anchor", anchor,
|
|
"x", x,
|
|
"y", y,
|
|
NULL);
|
|
}
|