PIKApp/app/pdb/pikapdbcontext.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;
}