565 lines
18 KiB
C
565 lines
18 KiB
C
/* PIKA - Photo and Image Kooker Application
|
|
* a rebranding of The GNU Image Manipulation Program (created with heckimp)
|
|
* A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
|
|
*
|
|
* Original copyright, applying to most contents (license remains unchanged):
|
|
* Copyright (C) 1995-1999 Spencer Kimball and Peter Mattis
|
|
*
|
|
* pikapdbcontext.c
|
|
*
|
|
* 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 <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <gegl.h>
|
|
|
|
#include "libpikabase/pikabase.h"
|
|
#include "libpikaconfig/pikaconfig.h"
|
|
|
|
#include "pdb-types.h"
|
|
|
|
#include "config/pikacoreconfig.h"
|
|
|
|
#include "core/pika.h"
|
|
#include "core/pikalist.h"
|
|
#include "core/pikapaintinfo.h"
|
|
#include "core/pikastrokeoptions.h"
|
|
|
|
#include "paint/pikabrushcore.h"
|
|
#include "paint/pikapaintoptions.h"
|
|
|
|
#include "pikapdbcontext.h"
|
|
|
|
#include "pika-intl.h"
|
|
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_ANTIALIAS,
|
|
PROP_FEATHER,
|
|
PROP_FEATHER_RADIUS_X,
|
|
PROP_FEATHER_RADIUS_Y,
|
|
PROP_SAMPLE_MERGED,
|
|
PROP_SAMPLE_CRITERION,
|
|
PROP_SAMPLE_THRESHOLD,
|
|
PROP_SAMPLE_TRANSPARENT,
|
|
PROP_DIAGONAL_NEIGHBORS,
|
|
PROP_INTERPOLATION,
|
|
PROP_TRANSFORM_DIRECTION,
|
|
PROP_TRANSFORM_RESIZE,
|
|
PROP_DISTANCE_METRIC
|
|
};
|
|
|
|
|
|
static void pika_pdb_context_iface_init (PikaConfigInterface *iface);
|
|
|
|
static void pika_pdb_context_constructed (GObject *object);
|
|
static void pika_pdb_context_finalize (GObject *object);
|
|
static void pika_pdb_context_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void pika_pdb_context_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
|
|
static void pika_pdb_context_reset (PikaConfig *config);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (PikaPDBContext, pika_pdb_context, PIKA_TYPE_CONTEXT,
|
|
G_IMPLEMENT_INTERFACE (PIKA_TYPE_CONFIG,
|
|
pika_pdb_context_iface_init))
|
|
|
|
#define parent_class pika_pdb_context_parent_class
|
|
|
|
static PikaConfigInterface *parent_config_iface = NULL;
|
|
|
|
|
|
static void
|
|
pika_pdb_context_class_init (PikaPDBContextClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->constructed = pika_pdb_context_constructed;
|
|
object_class->finalize = pika_pdb_context_finalize;
|
|
object_class->set_property = pika_pdb_context_set_property;
|
|
object_class->get_property = pika_pdb_context_get_property;
|
|
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_ANTIALIAS,
|
|
"antialias",
|
|
_("Antialiasing"),
|
|
_("Smooth edges"),
|
|
TRUE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_FEATHER,
|
|
"feather",
|
|
_("Feather"),
|
|
NULL,
|
|
FALSE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_FEATHER_RADIUS_X,
|
|
"feather-radius-x",
|
|
_("Feather radius X"),
|
|
NULL,
|
|
0.0, 1000.0, 10.0,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_FEATHER_RADIUS_Y,
|
|
"feather-radius-y",
|
|
_("Feather radius Y"),
|
|
NULL,
|
|
0.0, 1000.0, 10.0,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_SAMPLE_MERGED,
|
|
"sample-merged",
|
|
_("Sample merged"),
|
|
NULL,
|
|
FALSE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_SAMPLE_CRITERION,
|
|
"sample-criterion",
|
|
_("Sample criterion"),
|
|
NULL,
|
|
PIKA_TYPE_SELECT_CRITERION,
|
|
PIKA_SELECT_CRITERION_COMPOSITE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_SAMPLE_THRESHOLD,
|
|
"sample-threshold",
|
|
_("Sample threshold"),
|
|
NULL,
|
|
0.0, 1.0, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_SAMPLE_TRANSPARENT,
|
|
"sample-transparent",
|
|
_("Sample transparent"),
|
|
NULL,
|
|
FALSE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_DIAGONAL_NEIGHBORS,
|
|
"diagonal-neighbors",
|
|
_("Diagonal neighbors"),
|
|
NULL,
|
|
FALSE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_INTERPOLATION,
|
|
"interpolation",
|
|
_("Interpolation"),
|
|
NULL,
|
|
PIKA_TYPE_INTERPOLATION_TYPE,
|
|
PIKA_INTERPOLATION_CUBIC,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_TRANSFORM_DIRECTION,
|
|
"transform-direction",
|
|
_("Transform direction"),
|
|
NULL,
|
|
PIKA_TYPE_TRANSFORM_DIRECTION,
|
|
PIKA_TRANSFORM_FORWARD,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_TRANSFORM_RESIZE,
|
|
"transform-resize",
|
|
_("Transform resize"),
|
|
NULL,
|
|
PIKA_TYPE_TRANSFORM_RESIZE,
|
|
PIKA_TRANSFORM_RESIZE_ADJUST,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
/* Legacy blend used "manhattan" metric to compute distance.
|
|
* API needs to stay compatible, hence the default value for this
|
|
* property.
|
|
* Nevertheless Euclidean distance since to render better; for PIKA 3
|
|
* API, we might therefore want to change the defaults to
|
|
* GEGL_DISTANCE_METRIC_EUCLIDEAN. FIXME.
|
|
*/
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_DISTANCE_METRIC,
|
|
"distance-metric",
|
|
_("Distance metric"),
|
|
NULL,
|
|
GEGL_TYPE_DISTANCE_METRIC,
|
|
GEGL_DISTANCE_METRIC_MANHATTAN,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
}
|
|
|
|
static void
|
|
pika_pdb_context_iface_init (PikaConfigInterface *iface)
|
|
{
|
|
parent_config_iface = g_type_interface_peek_parent (iface);
|
|
|
|
if (! parent_config_iface)
|
|
parent_config_iface = g_type_default_interface_peek (PIKA_TYPE_CONFIG);
|
|
|
|
iface->reset = pika_pdb_context_reset;
|
|
}
|
|
|
|
static void
|
|
pika_pdb_context_init (PikaPDBContext *context)
|
|
{
|
|
context->paint_options_list = pika_list_new (PIKA_TYPE_PAINT_OPTIONS,
|
|
FALSE);
|
|
}
|
|
|
|
static void
|
|
pika_pdb_context_constructed (GObject *object)
|
|
{
|
|
PikaPDBContext *context = PIKA_PDB_CONTEXT (object);
|
|
PikaInterpolationType interpolation;
|
|
gint threshold;
|
|
GParamSpec *pspec;
|
|
|
|
G_OBJECT_CLASS (parent_class)->constructed (object);
|
|
|
|
context->stroke_options = pika_stroke_options_new (PIKA_CONTEXT (context)->pika,
|
|
PIKA_CONTEXT (context),
|
|
TRUE);
|
|
|
|
/* keep the stroke options in sync with the context */
|
|
pika_context_define_properties (PIKA_CONTEXT (context->stroke_options),
|
|
PIKA_CONTEXT_PROP_MASK_ALL, FALSE);
|
|
pika_context_set_parent (PIKA_CONTEXT (context->stroke_options),
|
|
PIKA_CONTEXT (context));
|
|
|
|
/* preserve the traditional PDB default */
|
|
g_object_set (context->stroke_options,
|
|
"method", PIKA_STROKE_PAINT_METHOD,
|
|
NULL);
|
|
|
|
g_object_bind_property (G_OBJECT (context), "antialias",
|
|
G_OBJECT (context->stroke_options), "antialias",
|
|
G_BINDING_SYNC_CREATE);
|
|
|
|
/* get default interpolation from pikarc */
|
|
|
|
interpolation = PIKA_CONTEXT (object)->pika->config->interpolation_type;
|
|
|
|
pspec = g_object_class_find_property (G_OBJECT_GET_CLASS (object),
|
|
"interpolation");
|
|
|
|
if (pspec)
|
|
G_PARAM_SPEC_ENUM (pspec)->default_value = interpolation;
|
|
|
|
g_object_set (object, "interpolation", interpolation, NULL);
|
|
|
|
/* get default threshold from pikarc */
|
|
|
|
threshold = PIKA_CONTEXT (object)->pika->config->default_threshold;
|
|
|
|
pspec = g_object_class_find_property (G_OBJECT_GET_CLASS (object),
|
|
"sample-threshold");
|
|
|
|
if (pspec)
|
|
G_PARAM_SPEC_DOUBLE (pspec)->default_value = threshold / 255.0;
|
|
|
|
g_object_set (object, "sample-threshold", threshold / 255.0, NULL);
|
|
}
|
|
|
|
static void
|
|
pika_pdb_context_finalize (GObject *object)
|
|
{
|
|
PikaPDBContext *context = PIKA_PDB_CONTEXT (object);
|
|
|
|
g_clear_object (&context->paint_options_list);
|
|
g_clear_object (&context->stroke_options);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
pika_pdb_context_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
PikaPDBContext *options = PIKA_PDB_CONTEXT (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_ANTIALIAS:
|
|
options->antialias = g_value_get_boolean (value);
|
|
break;
|
|
|
|
case PROP_FEATHER:
|
|
options->feather = g_value_get_boolean (value);
|
|
break;
|
|
|
|
case PROP_FEATHER_RADIUS_X:
|
|
options->feather_radius_x = g_value_get_double (value);
|
|
break;
|
|
|
|
case PROP_FEATHER_RADIUS_Y:
|
|
options->feather_radius_y = g_value_get_double (value);
|
|
break;
|
|
|
|
case PROP_SAMPLE_MERGED:
|
|
options->sample_merged = g_value_get_boolean (value);
|
|
break;
|
|
|
|
case PROP_SAMPLE_CRITERION:
|
|
options->sample_criterion = g_value_get_enum (value);
|
|
break;
|
|
|
|
case PROP_SAMPLE_THRESHOLD:
|
|
options->sample_threshold = g_value_get_double (value);
|
|
break;
|
|
|
|
case PROP_SAMPLE_TRANSPARENT:
|
|
options->sample_transparent = g_value_get_boolean (value);
|
|
break;
|
|
|
|
case PROP_DIAGONAL_NEIGHBORS:
|
|
options->diagonal_neighbors = g_value_get_boolean (value);
|
|
break;
|
|
|
|
case PROP_INTERPOLATION:
|
|
options->interpolation = g_value_get_enum (value);
|
|
break;
|
|
|
|
case PROP_TRANSFORM_DIRECTION:
|
|
options->transform_direction = g_value_get_enum (value);
|
|
break;
|
|
|
|
case PROP_TRANSFORM_RESIZE:
|
|
options->transform_resize = g_value_get_enum (value);
|
|
break;
|
|
|
|
case PROP_DISTANCE_METRIC:
|
|
options->distance_metric = g_value_get_enum (value);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_pdb_context_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
PikaPDBContext *options = PIKA_PDB_CONTEXT (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_ANTIALIAS:
|
|
g_value_set_boolean (value, options->antialias);
|
|
break;
|
|
|
|
case PROP_FEATHER:
|
|
g_value_set_boolean (value, options->feather);
|
|
break;
|
|
|
|
case PROP_FEATHER_RADIUS_X:
|
|
g_value_set_double (value, options->feather_radius_x);
|
|
break;
|
|
|
|
case PROP_FEATHER_RADIUS_Y:
|
|
g_value_set_double (value, options->feather_radius_y);
|
|
break;
|
|
|
|
case PROP_SAMPLE_MERGED:
|
|
g_value_set_boolean (value, options->sample_merged);
|
|
break;
|
|
|
|
case PROP_SAMPLE_CRITERION:
|
|
g_value_set_enum (value, options->sample_criterion);
|
|
break;
|
|
|
|
case PROP_SAMPLE_THRESHOLD:
|
|
g_value_set_double (value, options->sample_threshold);
|
|
break;
|
|
|
|
case PROP_SAMPLE_TRANSPARENT:
|
|
g_value_set_boolean (value, options->sample_transparent);
|
|
break;
|
|
|
|
case PROP_DIAGONAL_NEIGHBORS:
|
|
g_value_set_boolean (value, options->diagonal_neighbors);
|
|
break;
|
|
|
|
case PROP_INTERPOLATION:
|
|
g_value_set_enum (value, options->interpolation);
|
|
break;
|
|
|
|
case PROP_TRANSFORM_DIRECTION:
|
|
g_value_set_enum (value, options->transform_direction);
|
|
break;
|
|
|
|
case PROP_TRANSFORM_RESIZE:
|
|
g_value_set_enum (value, options->transform_resize);
|
|
break;
|
|
|
|
case PROP_DISTANCE_METRIC:
|
|
g_value_set_enum (value, options->distance_metric);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_pdb_context_reset (PikaConfig *config)
|
|
{
|
|
PikaPDBContext *context = PIKA_PDB_CONTEXT (config);
|
|
GList *list;
|
|
|
|
for (list = PIKA_LIST (context->paint_options_list)->queue->head;
|
|
list;
|
|
list = g_list_next (list))
|
|
{
|
|
pika_config_reset (list->data);
|
|
}
|
|
|
|
pika_config_reset (PIKA_CONFIG (context->stroke_options));
|
|
|
|
/* preserve the traditional PDB default */
|
|
g_object_set (context->stroke_options,
|
|
"method", PIKA_STROKE_PAINT_METHOD,
|
|
NULL);
|
|
|
|
parent_config_iface->reset (config);
|
|
|
|
g_object_notify (G_OBJECT (context), "antialias");
|
|
}
|
|
|
|
PikaContext *
|
|
pika_pdb_context_new (Pika *pika,
|
|
PikaContext *parent,
|
|
gboolean set_parent)
|
|
{
|
|
PikaPDBContext *context;
|
|
GList *list;
|
|
|
|
g_return_val_if_fail (PIKA_IS_PIKA (pika), NULL);
|
|
g_return_val_if_fail (PIKA_IS_CONTEXT (parent), NULL);
|
|
|
|
context = g_object_new (PIKA_TYPE_PDB_CONTEXT,
|
|
"pika", pika,
|
|
"name", "PDB Context",
|
|
NULL);
|
|
|
|
if (set_parent)
|
|
{
|
|
pika_context_define_properties (PIKA_CONTEXT (context),
|
|
PIKA_CONTEXT_PROP_MASK_ALL, FALSE);
|
|
pika_context_set_parent (PIKA_CONTEXT (context), parent);
|
|
|
|
for (list = pika_get_paint_info_iter (pika);
|
|
list;
|
|
list = g_list_next (list))
|
|
{
|
|
PikaPaintInfo *info = list->data;
|
|
|
|
pika_container_add (context->paint_options_list,
|
|
PIKA_OBJECT (info->paint_options));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (list = PIKA_LIST (PIKA_PDB_CONTEXT (parent)->paint_options_list)->queue->head;
|
|
list;
|
|
list = g_list_next (list))
|
|
{
|
|
PikaPaintOptions *options = pika_config_duplicate (list->data);
|
|
|
|
pika_container_add (context->paint_options_list,
|
|
PIKA_OBJECT (options));
|
|
g_object_unref (options);
|
|
}
|
|
|
|
pika_config_copy (PIKA_CONFIG (PIKA_PDB_CONTEXT (parent)->stroke_options),
|
|
PIKA_CONFIG (context->stroke_options),
|
|
0);
|
|
}
|
|
|
|
/* copy the context properties last, they might have been
|
|
* overwritten by the above copying of stroke options, which have
|
|
* the pdb context as parent
|
|
*/
|
|
pika_config_sync (G_OBJECT (parent), G_OBJECT (context), 0);
|
|
|
|
/* Reset the proper init name after syncing. */
|
|
g_object_set (G_OBJECT (context),
|
|
"name", "PDB Context",
|
|
NULL);
|
|
|
|
return PIKA_CONTEXT (context);
|
|
}
|
|
|
|
PikaContainer *
|
|
pika_pdb_context_get_paint_options_list (PikaPDBContext *context)
|
|
{
|
|
g_return_val_if_fail (PIKA_IS_PDB_CONTEXT (context), NULL);
|
|
|
|
return context->paint_options_list;
|
|
}
|
|
|
|
PikaPaintOptions *
|
|
pika_pdb_context_get_paint_options (PikaPDBContext *context,
|
|
const gchar *name)
|
|
{
|
|
g_return_val_if_fail (PIKA_IS_PDB_CONTEXT (context), NULL);
|
|
|
|
if (! name)
|
|
name = pika_object_get_name (pika_context_get_paint_info (PIKA_CONTEXT (context)));
|
|
|
|
return (PikaPaintOptions *)
|
|
pika_container_get_child_by_name (context->paint_options_list, name);
|
|
}
|
|
|
|
GList *
|
|
pika_pdb_context_get_brush_options (PikaPDBContext *context)
|
|
{
|
|
GList *brush_options = NULL;
|
|
GList *list;
|
|
|
|
g_return_val_if_fail (PIKA_IS_PDB_CONTEXT (context), NULL);
|
|
|
|
for (list = PIKA_LIST (context->paint_options_list)->queue->head;
|
|
list;
|
|
list = g_list_next (list))
|
|
{
|
|
PikaPaintOptions *options = list->data;
|
|
|
|
if (g_type_is_a (options->paint_info->paint_type, PIKA_TYPE_BRUSH_CORE))
|
|
brush_options = g_list_prepend (brush_options, options);
|
|
}
|
|
|
|
return g_list_reverse (brush_options);
|
|
}
|
|
|
|
PikaStrokeOptions *
|
|
pika_pdb_context_get_stroke_options (PikaPDBContext *context)
|
|
{
|
|
g_return_val_if_fail (PIKA_IS_PDB_CONTEXT (context), NULL);
|
|
|
|
return context->stroke_options;
|
|
}
|