PIKApp/app/core/pikafilloptions.c

626 lines
20 KiB
C

/* The PIKA -- an image manipulation program
* Copyright (C) 1995-1999 Spencer Kimball and Peter Mattis
*
* pikafilloptions.c
* Copyright (C) 2003 Simon Budig
*
* 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 <https://www.gnu.org/licenses/>.
*/
#include "config.h"
#include <cairo.h>
#include <gdk-pixbuf/gdk-pixbuf.h>
#include <gegl.h>
#include "libpikabase/pikabase.h"
#include "libpikacolor/pikacolor.h"
#include "libpikaconfig/pikaconfig.h"
#include "core-types.h"
#include "operations/layer-modes/pika-layer-modes.h"
#include "pika.h"
#include "pika-palettes.h"
#include "pikadrawable.h"
#include "pikadrawable-fill.h"
#include "pikaerror.h"
#include "pikafilloptions.h"
#include "pikaimage.h"
#include "pikapattern.h"
#include "pika-intl.h"
enum
{
PROP_0,
PROP_STYLE,
PROP_CUSTOM_STYLE,
PROP_ANTIALIAS,
PROP_FEATHER,
PROP_FEATHER_RADIUS,
PROP_PATTERN_VIEW_TYPE,
PROP_PATTERN_VIEW_SIZE
};
typedef struct _PikaFillOptionsPrivate PikaFillOptionsPrivate;
struct _PikaFillOptionsPrivate
{
PikaFillStyle style;
PikaCustomStyle custom_style;
gboolean antialias;
gboolean feather;
gdouble feather_radius;
PikaViewType pattern_view_type;
PikaViewSize pattern_view_size;
const gchar *undo_desc;
};
#define GET_PRIVATE(options) \
((PikaFillOptionsPrivate *) pika_fill_options_get_instance_private ((PikaFillOptions *) (options)))
static void pika_fill_options_config_init (PikaConfigInterface *iface);
static void pika_fill_options_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec);
static void pika_fill_options_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec);
static gboolean pika_fill_options_serialize (PikaConfig *config,
PikaConfigWriter *writer,
gpointer data);
G_DEFINE_TYPE_WITH_CODE (PikaFillOptions, pika_fill_options, PIKA_TYPE_CONTEXT,
G_ADD_PRIVATE (PikaFillOptions)
G_IMPLEMENT_INTERFACE (PIKA_TYPE_CONFIG,
pika_fill_options_config_init))
static void
pika_fill_options_class_init (PikaFillOptionsClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->set_property = pika_fill_options_set_property;
object_class->get_property = pika_fill_options_get_property;
PIKA_CONFIG_PROP_ENUM (object_class, PROP_STYLE,
"style",
_("Style"),
NULL,
PIKA_TYPE_FILL_STYLE,
PIKA_FILL_STYLE_FG_COLOR,
PIKA_PARAM_STATIC_STRINGS);
PIKA_CONFIG_PROP_ENUM (object_class, PROP_CUSTOM_STYLE,
"custom-style",
_("Custom style"),
NULL,
PIKA_TYPE_CUSTOM_STYLE,
PIKA_CUSTOM_STYLE_SOLID_COLOR,
PIKA_PARAM_STATIC_STRINGS);
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_ANTIALIAS,
"antialias",
_("Antialiasing"),
NULL,
TRUE,
PIKA_PARAM_STATIC_STRINGS);
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_FEATHER,
"feather",
_("Feather edges"),
_("Enable feathering of fill edges"),
FALSE,
PIKA_PARAM_STATIC_STRINGS);
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_FEATHER_RADIUS,
"feather-radius",
_("Radius"),
_("Radius of feathering"),
0.0, 100.0, 10.0,
PIKA_PARAM_STATIC_STRINGS);
g_object_class_install_property (object_class, PROP_PATTERN_VIEW_TYPE,
g_param_spec_enum ("pattern-view-type",
NULL, NULL,
PIKA_TYPE_VIEW_TYPE,
PIKA_VIEW_TYPE_GRID,
G_PARAM_CONSTRUCT |
PIKA_PARAM_READWRITE));
g_object_class_install_property (object_class, PROP_PATTERN_VIEW_SIZE,
g_param_spec_int ("pattern-view-size",
NULL, NULL,
PIKA_VIEW_SIZE_TINY,
PIKA_VIEWABLE_MAX_BUTTON_SIZE,
PIKA_VIEW_SIZE_SMALL,
G_PARAM_CONSTRUCT |
PIKA_PARAM_READWRITE));
}
static void
pika_fill_options_config_init (PikaConfigInterface *iface)
{
iface->serialize = pika_fill_options_serialize;
}
static void
pika_fill_options_init (PikaFillOptions *options)
{
}
static void
pika_fill_options_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec)
{
PikaFillOptionsPrivate *private = GET_PRIVATE (object);
switch (property_id)
{
case PROP_STYLE:
private->style = g_value_get_enum (value);
private->undo_desc = NULL;
break;
case PROP_CUSTOM_STYLE:
private->custom_style = g_value_get_enum (value);
private->undo_desc = NULL;
break;
case PROP_ANTIALIAS:
private->antialias = g_value_get_boolean (value);
break;
case PROP_FEATHER:
private->feather = g_value_get_boolean (value);
break;
case PROP_FEATHER_RADIUS:
private->feather_radius = g_value_get_double (value);
break;
case PROP_PATTERN_VIEW_TYPE:
private->pattern_view_type = g_value_get_enum (value);
break;
case PROP_PATTERN_VIEW_SIZE:
private->pattern_view_size = g_value_get_int (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static void
pika_fill_options_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec)
{
PikaFillOptionsPrivate *private = GET_PRIVATE (object);
switch (property_id)
{
case PROP_STYLE:
g_value_set_enum (value, private->style);
break;
case PROP_CUSTOM_STYLE:
g_value_set_enum (value, private->custom_style);
break;
case PROP_ANTIALIAS:
g_value_set_boolean (value, private->antialias);
break;
case PROP_FEATHER:
g_value_set_boolean (value, private->feather);
break;
case PROP_FEATHER_RADIUS:
g_value_set_double (value, private->feather_radius);
break;
case PROP_PATTERN_VIEW_TYPE:
g_value_set_enum (value, private->pattern_view_type);
break;
case PROP_PATTERN_VIEW_SIZE:
g_value_set_int (value, private->pattern_view_size);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static gboolean
pika_fill_options_serialize (PikaConfig *config,
PikaConfigWriter *writer,
gpointer data)
{
return pika_config_serialize_properties (config, writer);
}
/* public functions */
PikaFillOptions *
pika_fill_options_new (Pika *pika,
PikaContext *context,
gboolean use_context_color)
{
PikaFillOptions *options;
g_return_val_if_fail (PIKA_IS_PIKA (pika), NULL);
g_return_val_if_fail (context == NULL || PIKA_IS_CONTEXT (context), NULL);
g_return_val_if_fail (use_context_color == FALSE || context != NULL, NULL);
options = g_object_new (PIKA_TYPE_FILL_OPTIONS,
"pika", pika,
NULL);
if (use_context_color)
{
pika_context_define_properties (PIKA_CONTEXT (options),
PIKA_CONTEXT_PROP_MASK_FOREGROUND |
PIKA_CONTEXT_PROP_MASK_BACKGROUND |
PIKA_CONTEXT_PROP_MASK_PATTERN,
FALSE);
pika_context_set_parent (PIKA_CONTEXT (options), context);
}
return options;
}
PikaFillStyle
pika_fill_options_get_style (PikaFillOptions *options)
{
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), PIKA_FILL_STYLE_FG_COLOR);
return GET_PRIVATE (options)->style;
}
void
pika_fill_options_set_style (PikaFillOptions *options,
PikaFillStyle style)
{
g_return_if_fail (PIKA_IS_FILL_OPTIONS (options));
g_object_set (options, "style", style, NULL);
}
PikaCustomStyle
pika_fill_options_get_custom_style (PikaFillOptions *options)
{
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options),
PIKA_CUSTOM_STYLE_SOLID_COLOR);
return GET_PRIVATE (options)->custom_style;
}
void
pika_fill_options_set_custom_style (PikaFillOptions *options,
PikaCustomStyle custom_style)
{
g_return_if_fail (PIKA_IS_FILL_OPTIONS (options));
g_object_set (options, "custom-style", custom_style, NULL);
}
gboolean
pika_fill_options_get_antialias (PikaFillOptions *options)
{
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), FALSE);
return GET_PRIVATE (options)->antialias;
}
void
pika_fill_options_set_antialias (PikaFillOptions *options,
gboolean antialias)
{
g_return_if_fail (PIKA_IS_FILL_OPTIONS (options));
g_object_set (options, "antialias", antialias, NULL);
}
gboolean
pika_fill_options_get_feather (PikaFillOptions *options,
gdouble *radius)
{
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), FALSE);
if (radius)
*radius = GET_PRIVATE (options)->feather_radius;
return GET_PRIVATE (options)->feather;
}
void
pika_fill_options_set_feather (PikaFillOptions *options,
gboolean feather,
gdouble radius)
{
g_return_if_fail (PIKA_IS_FILL_OPTIONS (options));
g_object_set (options, "feather", feather, NULL);
g_object_set (options, "feather-radius", radius, NULL);
}
gboolean
pika_fill_options_set_by_fill_type (PikaFillOptions *options,
PikaContext *context,
PikaFillType fill_type,
GError **error)
{
PikaFillOptionsPrivate *private;
PikaRGB color;
const gchar *undo_desc;
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), FALSE);
g_return_val_if_fail (PIKA_IS_CONTEXT (context), FALSE);
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
private = GET_PRIVATE (options);
private->undo_desc = NULL;
switch (fill_type)
{
case PIKA_FILL_FOREGROUND:
pika_context_get_foreground (context, &color);
undo_desc = C_("undo-type", "Fill with Foreground Color");
break;
case PIKA_FILL_BACKGROUND:
pika_context_get_background (context, &color);
undo_desc = C_("undo-type", "Fill with Background Color");
break;
case PIKA_FILL_CIELAB_MIDDLE_GRAY:
{
const float cielab_pixel[3] = {50, 0, 0};
float pixel[3] = {0, 0, 0};
PikaImage *image = pika_context_get_image (context);
PikaImageBaseType base_type;
const Babl *format;
base_type = pika_image_get_base_type (image);
if (base_type == PIKA_INDEXED)
base_type = PIKA_RGB;
format = pika_image_get_format (image, base_type,
PIKA_PRECISION_FLOAT_NON_LINEAR, FALSE,
pika_image_get_layer_space (image));
babl_process (babl_fish (babl_format ("CIE Lab float"), format),
cielab_pixel, pixel, 1);
pika_rgba_set (&color, pixel[0], pixel[1], pixel[2], PIKA_OPACITY_OPAQUE);
undo_desc = C_("undo-type", "Fill with Middle Gray (CIELAB) Color");
}
break;
case PIKA_FILL_WHITE:
pika_rgba_set (&color, 1.0, 1.0, 1.0, PIKA_OPACITY_OPAQUE);
undo_desc = C_("undo-type", "Fill with White");
break;
case PIKA_FILL_TRANSPARENT:
pika_context_get_background (context, &color);
pika_context_set_paint_mode (PIKA_CONTEXT (options),
PIKA_LAYER_MODE_ERASE);
undo_desc = C_("undo-type", "Fill with Transparency");
break;
case PIKA_FILL_PATTERN:
{
PikaPattern *pattern = pika_context_get_pattern (context);
if (! pattern)
{
g_set_error_literal (error, PIKA_ERROR, PIKA_FAILED,
_("No patterns available for this operation."));
return FALSE;
}
pika_fill_options_set_style (options, PIKA_FILL_STYLE_PATTERN);
pika_context_set_pattern (PIKA_CONTEXT (options), pattern);
private->undo_desc = C_("undo-type", "Fill with Pattern");
return TRUE;
}
break;
default:
g_warning ("%s: invalid fill_type %d", G_STRFUNC, fill_type);
return FALSE;
}
pika_fill_options_set_style (options, PIKA_FILL_STYLE_FG_COLOR);
pika_context_set_foreground (PIKA_CONTEXT (options), &color);
private->undo_desc = undo_desc;
return TRUE;
}
gboolean
pika_fill_options_set_by_fill_mode (PikaFillOptions *options,
PikaContext *context,
PikaBucketFillMode fill_mode,
GError **error)
{
PikaFillType fill_type;
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), FALSE);
g_return_val_if_fail (PIKA_IS_CONTEXT (context), FALSE);
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
switch (fill_mode)
{
default:
case PIKA_BUCKET_FILL_FG:
fill_type = PIKA_FILL_FOREGROUND;
break;
case PIKA_BUCKET_FILL_BG:
fill_type = PIKA_FILL_BACKGROUND;
break;
case PIKA_BUCKET_FILL_PATTERN:
fill_type = PIKA_FILL_PATTERN;
break;
}
return pika_fill_options_set_by_fill_type (options, context,
fill_type, error);
}
const gchar *
pika_fill_options_get_undo_desc (PikaFillOptions *options)
{
PikaFillOptionsPrivate *private;
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), NULL);
private = GET_PRIVATE (options);
if (private->undo_desc)
return private->undo_desc;
switch (private->style)
{
case PIKA_FILL_STYLE_FG_COLOR:
return C_("undo-type", "Fill with Foreground Color");
case PIKA_FILL_STYLE_BG_COLOR:
return C_("undo-type", "Fill with Background Color");
case PIKA_FILL_STYLE_PATTERN:
return C_("undo-type", "Fill with Pattern");
}
g_return_val_if_reached (NULL);
}
const Babl *
pika_fill_options_get_format (PikaFillOptions *options,
PikaDrawable *drawable)
{
PikaContext *context;
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), NULL);
g_return_val_if_fail (PIKA_IS_DRAWABLE (drawable), NULL);
context = PIKA_CONTEXT (options);
return pika_layer_mode_get_format (pika_context_get_paint_mode (context),
PIKA_LAYER_COLOR_SPACE_AUTO,
PIKA_LAYER_COLOR_SPACE_AUTO,
pika_layer_mode_get_paint_composite_mode (
pika_context_get_paint_mode (context)),
pika_drawable_get_format (drawable));
}
GeglBuffer *
pika_fill_options_create_buffer (PikaFillOptions *options,
PikaDrawable *drawable,
const GeglRectangle *rect,
gint pattern_offset_x,
gint pattern_offset_y)
{
GeglBuffer *buffer;
g_return_val_if_fail (PIKA_IS_FILL_OPTIONS (options), NULL);
g_return_val_if_fail (pika_fill_options_get_style (options) !=
PIKA_FILL_STYLE_PATTERN ||
pika_context_get_pattern (PIKA_CONTEXT (options)) != NULL,
NULL);
g_return_val_if_fail (PIKA_IS_DRAWABLE (drawable), NULL);
g_return_val_if_fail (rect != NULL, NULL);
buffer = gegl_buffer_new (rect,
pika_fill_options_get_format (options, drawable));
pika_fill_options_fill_buffer (options, drawable, buffer,
pattern_offset_x, pattern_offset_y);
return buffer;
}
void
pika_fill_options_fill_buffer (PikaFillOptions *options,
PikaDrawable *drawable,
GeglBuffer *buffer,
gint pattern_offset_x,
gint pattern_offset_y)
{
g_return_if_fail (PIKA_IS_FILL_OPTIONS (options));
g_return_if_fail (pika_fill_options_get_style (options) !=
PIKA_FILL_STYLE_PATTERN ||
pika_context_get_pattern (PIKA_CONTEXT (options)) != NULL);
g_return_if_fail (PIKA_IS_DRAWABLE (drawable));
g_return_if_fail (GEGL_IS_BUFFER (buffer));
switch (pika_fill_options_get_style (options))
{
case PIKA_FILL_STYLE_FG_COLOR:
{
PikaRGB color;
pika_context_get_foreground (PIKA_CONTEXT (options), &color);
pika_palettes_add_color_history (PIKA_CONTEXT (options)->pika, &color);
pika_drawable_fill_buffer (drawable, buffer,
&color, NULL, 0, 0);
}
break;
case PIKA_FILL_STYLE_BG_COLOR:
{
PikaRGB color;
pika_context_get_background (PIKA_CONTEXT (options), &color);
pika_palettes_add_color_history (PIKA_CONTEXT (options)->pika, &color);
pika_drawable_fill_buffer (drawable, buffer,
&color, NULL, 0, 0);
}
break;
case PIKA_FILL_STYLE_PATTERN:
{
PikaPattern *pattern;
pattern = pika_context_get_pattern (PIKA_CONTEXT (options));
pika_drawable_fill_buffer (drawable, buffer,
NULL, pattern,
pattern_offset_x,
pattern_offset_y);
}
break;
}
}