… this filter is meant to be merged.
In commit 2c066afff9, I made so that we had a "gegl:convert-format" only
at the latest filter over a drawable. While this was already a huge
improvement, it was still "too much".
Basically the only time when we want to convert to drawable format
(which means usually demote the output to lower bit depth) is when the
filter was just created and is meant to be merged.
This also means that from now on, during preview time, when checking and
unchecking the "Merge filter" checkbox, the preview may be slightly
different. A very good test to see the difference more obviously would
be with indexed images, because "Merge filter" preview would map to
allowed colors only.
655 lines
19 KiB
C
655 lines
19 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* gimpapplicator.c
|
|
* Copyright (C) 2012-2013 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gegl.h>
|
|
|
|
#include "gimp-gegl-types.h"
|
|
|
|
#include "gimp-gegl-nodes.h"
|
|
#include "gimpapplicator.h"
|
|
|
|
|
|
static void gimp_applicator_finalize (GObject *object);
|
|
static void gimp_applicator_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_applicator_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
|
|
|
|
G_DEFINE_TYPE (GimpApplicator, gimp_applicator, G_TYPE_OBJECT)
|
|
|
|
#define parent_class gimp_applicator_parent_class
|
|
|
|
|
|
static void
|
|
gimp_applicator_class_init (GimpApplicatorClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = gimp_applicator_finalize;
|
|
object_class->set_property = gimp_applicator_set_property;
|
|
object_class->get_property = gimp_applicator_get_property;
|
|
}
|
|
|
|
static void
|
|
gimp_applicator_init (GimpApplicator *applicator)
|
|
{
|
|
applicator->active = TRUE;
|
|
applicator->opacity = 1.0;
|
|
applicator->paint_mode = GIMP_LAYER_MODE_NORMAL;
|
|
applicator->blend_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
applicator->composite_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
applicator->composite_mode = GIMP_LAYER_COMPOSITE_AUTO;
|
|
applicator->affect = GIMP_COMPONENT_MASK_ALL;
|
|
}
|
|
|
|
static void
|
|
gimp_applicator_finalize (GObject *object)
|
|
{
|
|
GimpApplicator *applicator = GIMP_APPLICATOR (object);
|
|
|
|
g_clear_object (&applicator->node);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gimp_applicator_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id)
|
|
{
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_applicator_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id)
|
|
{
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
GimpApplicator *
|
|
gimp_applicator_new (GeglNode *parent)
|
|
{
|
|
GimpApplicator *applicator;
|
|
|
|
g_return_val_if_fail (parent == NULL || GEGL_IS_NODE (parent), NULL);
|
|
|
|
applicator = g_object_new (GIMP_TYPE_APPLICATOR, NULL);
|
|
|
|
if (parent)
|
|
applicator->node = g_object_ref (parent);
|
|
else
|
|
applicator->node = gegl_node_new ();
|
|
|
|
applicator->input_node =
|
|
gegl_node_get_input_proxy (applicator->node, "input");
|
|
|
|
applicator->aux_node =
|
|
gegl_node_get_input_proxy (applicator->node, "aux");
|
|
|
|
applicator->output_node =
|
|
gegl_node_get_output_proxy (applicator->node, "output");
|
|
|
|
applicator->mode_node = gegl_node_new_child (applicator->node,
|
|
"operation", "gimp:normal",
|
|
NULL);
|
|
|
|
gimp_gegl_mode_node_set_mode (applicator->mode_node,
|
|
applicator->paint_mode,
|
|
applicator->blend_space,
|
|
applicator->composite_space,
|
|
applicator->composite_mode);
|
|
gimp_gegl_mode_node_set_opacity (applicator->mode_node,
|
|
applicator->opacity);
|
|
|
|
gegl_node_link (applicator->input_node, applicator->mode_node);
|
|
|
|
applicator->apply_offset_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:translate",
|
|
NULL);
|
|
|
|
gegl_node_link_many (applicator->aux_node,
|
|
applicator->apply_offset_node,
|
|
NULL);
|
|
|
|
gegl_node_connect (applicator->apply_offset_node, "output",
|
|
applicator->mode_node, "aux");
|
|
|
|
applicator->mask_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:buffer-source",
|
|
NULL);
|
|
|
|
applicator->mask_offset_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:translate",
|
|
NULL);
|
|
|
|
gegl_node_link (applicator->mask_node, applicator->mask_offset_node);
|
|
/* don't connect the the mask offset node to mode's aux2 yet */
|
|
|
|
applicator->affect_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gimp:mask-components",
|
|
"mask", applicator->affect,
|
|
NULL);
|
|
|
|
applicator->convert_format_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:nop",
|
|
NULL);
|
|
|
|
applicator->cache_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:nop",
|
|
NULL);
|
|
|
|
applicator->crop_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:nop",
|
|
NULL);
|
|
|
|
gegl_node_link_many (applicator->input_node,
|
|
applicator->affect_node,
|
|
applicator->convert_format_node,
|
|
applicator->cache_node,
|
|
applicator->crop_node,
|
|
applicator->output_node,
|
|
NULL);
|
|
|
|
gegl_node_connect (applicator->mode_node, "output",
|
|
applicator->affect_node, "aux");
|
|
|
|
return applicator;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_active (GimpApplicator *applicator,
|
|
gboolean active)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (active != applicator->active)
|
|
{
|
|
applicator->active = active;
|
|
|
|
if (active)
|
|
gegl_node_link (applicator->crop_node, applicator->output_node);
|
|
else
|
|
gegl_node_link (applicator->input_node, applicator->output_node);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_src_buffer (GimpApplicator *applicator,
|
|
GeglBuffer *src_buffer)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
g_return_if_fail (src_buffer == NULL || GEGL_IS_BUFFER (src_buffer));
|
|
|
|
if (src_buffer == applicator->src_buffer)
|
|
return;
|
|
|
|
if (src_buffer)
|
|
{
|
|
if (! applicator->src_node)
|
|
{
|
|
applicator->src_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:buffer-source",
|
|
"buffer", src_buffer,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
gegl_node_set (applicator->src_node,
|
|
"buffer", src_buffer,
|
|
NULL);
|
|
}
|
|
|
|
if (! applicator->src_buffer)
|
|
gegl_node_link (applicator->src_node, applicator->input_node);
|
|
}
|
|
else if (applicator->src_buffer)
|
|
{
|
|
gegl_node_disconnect (applicator->input_node, "input");
|
|
|
|
gegl_node_set (applicator->src_node,
|
|
"buffer", NULL,
|
|
NULL);
|
|
}
|
|
|
|
applicator->src_buffer = src_buffer;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_dest_buffer (GimpApplicator *applicator,
|
|
GeglBuffer *dest_buffer)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
g_return_if_fail (dest_buffer == NULL || GEGL_IS_BUFFER (dest_buffer));
|
|
|
|
if (dest_buffer == applicator->dest_buffer)
|
|
return;
|
|
|
|
if (dest_buffer)
|
|
{
|
|
if (! applicator->dest_node)
|
|
{
|
|
applicator->dest_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:write-buffer",
|
|
"buffer", dest_buffer,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
gegl_node_set (applicator->dest_node,
|
|
"buffer", dest_buffer,
|
|
NULL);
|
|
}
|
|
|
|
if (! applicator->dest_buffer)
|
|
gegl_node_link (applicator->affect_node, applicator->dest_node);
|
|
}
|
|
else if (applicator->dest_buffer)
|
|
{
|
|
gegl_node_disconnect (applicator->dest_node, "input");
|
|
|
|
gegl_node_set (applicator->dest_node,
|
|
"buffer", NULL,
|
|
NULL);
|
|
}
|
|
|
|
applicator->dest_buffer = dest_buffer;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_mask_buffer (GimpApplicator *applicator,
|
|
GeglBuffer *mask_buffer)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
g_return_if_fail (mask_buffer == NULL || GEGL_IS_BUFFER (mask_buffer));
|
|
|
|
if (applicator->mask_buffer == mask_buffer)
|
|
return;
|
|
|
|
gegl_node_set (applicator->mask_node,
|
|
"buffer", mask_buffer,
|
|
NULL);
|
|
|
|
if (mask_buffer)
|
|
{
|
|
gegl_node_connect (applicator->mask_offset_node, "output",
|
|
applicator->mode_node, "aux2");
|
|
}
|
|
else
|
|
{
|
|
gegl_node_disconnect (applicator->mode_node, "aux2");
|
|
}
|
|
|
|
applicator->mask_buffer = mask_buffer;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_mask_offset (GimpApplicator *applicator,
|
|
gint mask_offset_x,
|
|
gint mask_offset_y)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (applicator->mask_offset_x != mask_offset_x ||
|
|
applicator->mask_offset_y != mask_offset_y)
|
|
{
|
|
applicator->mask_offset_x = mask_offset_x;
|
|
applicator->mask_offset_y = mask_offset_y;
|
|
|
|
gegl_node_set (applicator->mask_offset_node,
|
|
"x", (gdouble) mask_offset_x,
|
|
"y", (gdouble) mask_offset_y,
|
|
NULL);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_apply_buffer (GimpApplicator *applicator,
|
|
GeglBuffer *apply_buffer)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
g_return_if_fail (apply_buffer == NULL || GEGL_IS_BUFFER (apply_buffer));
|
|
|
|
if (apply_buffer == applicator->apply_buffer)
|
|
return;
|
|
|
|
if (apply_buffer)
|
|
{
|
|
if (! applicator->apply_src_node)
|
|
{
|
|
applicator->apply_src_node =
|
|
gegl_node_new_child (applicator->node,
|
|
"operation", "gegl:buffer-source",
|
|
"buffer", apply_buffer,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
gegl_node_set (applicator->apply_src_node,
|
|
"buffer", apply_buffer,
|
|
NULL);
|
|
}
|
|
|
|
if (! applicator->apply_buffer)
|
|
{
|
|
gegl_node_connect (applicator->apply_src_node, "output",
|
|
applicator->apply_offset_node, "input");
|
|
}
|
|
}
|
|
else if (applicator->apply_buffer)
|
|
{
|
|
gegl_node_link (applicator->aux_node, applicator->apply_offset_node);
|
|
}
|
|
|
|
applicator->apply_buffer = apply_buffer;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_apply_offset (GimpApplicator *applicator,
|
|
gint apply_offset_x,
|
|
gint apply_offset_y)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (applicator->apply_offset_x != apply_offset_x ||
|
|
applicator->apply_offset_y != apply_offset_y)
|
|
{
|
|
applicator->apply_offset_x = apply_offset_x;
|
|
applicator->apply_offset_y = apply_offset_y;
|
|
|
|
gegl_node_set (applicator->apply_offset_node,
|
|
"x", (gdouble) apply_offset_x,
|
|
"y", (gdouble) apply_offset_y,
|
|
NULL);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_opacity (GimpApplicator *applicator,
|
|
gdouble opacity)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (applicator->opacity != opacity)
|
|
{
|
|
applicator->opacity = opacity;
|
|
|
|
gimp_gegl_mode_node_set_opacity (applicator->mode_node,
|
|
opacity);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_mode (GimpApplicator *applicator,
|
|
GimpLayerMode paint_mode,
|
|
GimpLayerColorSpace blend_space,
|
|
GimpLayerColorSpace composite_space,
|
|
GimpLayerCompositeMode composite_mode)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (applicator->paint_mode != paint_mode ||
|
|
applicator->blend_space != blend_space ||
|
|
applicator->composite_space != composite_space ||
|
|
applicator->composite_mode != composite_mode)
|
|
{
|
|
applicator->paint_mode = paint_mode;
|
|
applicator->blend_space = blend_space;
|
|
applicator->composite_space = composite_space;
|
|
applicator->composite_mode = composite_mode;
|
|
|
|
gimp_gegl_mode_node_set_mode (applicator->mode_node,
|
|
paint_mode, blend_space,
|
|
composite_space, composite_mode);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_affect (GimpApplicator *applicator,
|
|
GimpComponentMask affect)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (applicator->affect != affect)
|
|
{
|
|
applicator->affect = affect;
|
|
|
|
gegl_node_set (applicator->affect_node,
|
|
"mask", affect,
|
|
NULL);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_applicator_set_output_format (GimpApplicator *applicator,
|
|
const Babl *format)
|
|
{
|
|
gboolean changed = FALSE;
|
|
|
|
g_return_val_if_fail (GIMP_IS_APPLICATOR (applicator), FALSE);
|
|
|
|
if (applicator->output_format != format)
|
|
{
|
|
changed = TRUE;
|
|
|
|
if (format)
|
|
{
|
|
if (! applicator->output_format)
|
|
{
|
|
gegl_node_set (applicator->convert_format_node,
|
|
"operation", "gegl:convert-format",
|
|
"format", format,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
gegl_node_set (applicator->convert_format_node,
|
|
"format", format,
|
|
NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gegl_node_set (applicator->convert_format_node,
|
|
"operation", "gegl:nop",
|
|
NULL);
|
|
}
|
|
|
|
applicator->output_format = format;
|
|
}
|
|
|
|
return changed;
|
|
}
|
|
|
|
const Babl *
|
|
gimp_applicator_get_output_format (GimpApplicator *applicator)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_APPLICATOR (applicator), NULL);
|
|
|
|
return applicator->output_format;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_cache (GimpApplicator *applicator,
|
|
gboolean enable)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (applicator->cache_enabled != enable)
|
|
{
|
|
if (enable)
|
|
{
|
|
gegl_node_set (applicator->cache_node,
|
|
"operation", "gegl:cache",
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
gegl_node_set (applicator->cache_node,
|
|
"operation", "gegl:nop",
|
|
NULL);
|
|
}
|
|
|
|
applicator->cache_enabled = enable;
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_applicator_get_cache (GimpApplicator *applicator)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_APPLICATOR (applicator), FALSE);
|
|
|
|
return applicator->cache_enabled;
|
|
}
|
|
|
|
gboolean gegl_buffer_list_valid_rectangles (GeglBuffer *buffer,
|
|
GeglRectangle **rectangles,
|
|
gint *n_rectangles);
|
|
|
|
GeglBuffer *
|
|
gimp_applicator_get_cache_buffer (GimpApplicator *applicator,
|
|
GeglRectangle **rectangles,
|
|
gint *n_rectangles)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_APPLICATOR (applicator), NULL);
|
|
g_return_val_if_fail (rectangles != NULL, NULL);
|
|
g_return_val_if_fail (n_rectangles != NULL, NULL);
|
|
|
|
if (applicator->cache_enabled)
|
|
{
|
|
GeglBuffer *cache;
|
|
|
|
gegl_node_get (applicator->cache_node,
|
|
"cache", &cache,
|
|
NULL);
|
|
|
|
if (cache)
|
|
{
|
|
if (gegl_buffer_list_valid_rectangles (cache,
|
|
rectangles, n_rectangles))
|
|
{
|
|
return cache;
|
|
}
|
|
|
|
g_object_unref (cache);
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_set_crop (GimpApplicator *applicator,
|
|
const GeglRectangle *rect)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
if (applicator->crop_enabled != (rect != NULL) ||
|
|
(rect && ! gegl_rectangle_equal (&applicator->crop_rect, rect)))
|
|
{
|
|
if (rect)
|
|
{
|
|
if (! applicator->crop_enabled)
|
|
{
|
|
gegl_node_set (applicator->crop_node,
|
|
"operation", "gimp:compose-crop",
|
|
"x", rect->x,
|
|
"y", rect->y,
|
|
"width", rect->width,
|
|
"height", rect->height,
|
|
NULL);
|
|
|
|
gegl_node_connect (applicator->input_node, "output",
|
|
applicator->crop_node, "aux");
|
|
}
|
|
else
|
|
{
|
|
gegl_node_set (applicator->crop_node,
|
|
"x", rect->x,
|
|
"y", rect->y,
|
|
"width", rect->width,
|
|
"height", rect->height,
|
|
NULL);
|
|
}
|
|
|
|
applicator->crop_enabled = TRUE;
|
|
applicator->crop_rect = *rect;
|
|
}
|
|
else
|
|
{
|
|
gegl_node_disconnect (applicator->crop_node, "aux");
|
|
gegl_node_set (applicator->crop_node,
|
|
"operation", "gegl:nop",
|
|
NULL);
|
|
|
|
applicator->crop_enabled = FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
const GeglRectangle *
|
|
gimp_applicator_get_crop (GimpApplicator *applicator)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_APPLICATOR (applicator), NULL);
|
|
|
|
if (applicator->crop_enabled)
|
|
return &applicator->crop_rect;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
gimp_applicator_blit (GimpApplicator *applicator,
|
|
const GeglRectangle *rect)
|
|
{
|
|
g_return_if_fail (GIMP_IS_APPLICATOR (applicator));
|
|
|
|
gegl_node_blit (applicator->dest_node, 1.0, rect,
|
|
NULL, NULL, 0, GEGL_BLIT_DEFAULT);
|
|
}
|