mirror of
https://gitlab.gnome.org/GNOME/gimp.git
synced 2025-10-06 01:12:40 +02:00
While there are cases when you want to edit the mask straight away (this is often the case when starting with a white or black mask), in many other cases, the mask may be already as you want it per its initialization (e.g. when initializing with a channel, selection, the alpha channel, etc.). Until now, the Add Mask dialog was always switching to the "Edit Layer Mask" mode by default, which forced an additional unneeded click each time you created a mask (and were in a case where you initialize the mask as you want it directly). Now adding "Edit mask immediately" feature in the "Add Layer Mask dialog": * It's checked by default to keep historical behavior. * As most other dialogs, the last value is remembered, allowing people with repetitive workflow not to have to repeatedly set the settings each and every time. * This default is also visible and settable in Preferences > Interface > Dialog Defaults.
3023 lines
105 KiB
C
3023 lines
105 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* 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 <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <cairo.h>
|
|
#include <gegl.h>
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
|
|
#include "libgimpbase/gimpbase.h"
|
|
#include "libgimpcolor/gimpcolor.h"
|
|
#include "libgimpconfig/gimpconfig.h"
|
|
#include "libgimpmath/gimpmath.h"
|
|
|
|
#include "core-types.h"
|
|
|
|
#include "operations/layer-modes/gimp-layer-modes.h"
|
|
|
|
#include "gegl/gimp-babl.h"
|
|
#include "gegl/gimp-gegl-apply-operation.h"
|
|
#include "gegl/gimp-gegl-loops.h"
|
|
#include "gegl/gimp-gegl-nodes.h"
|
|
|
|
#include "gimpboundary.h"
|
|
#include "gimpchannel-select.h"
|
|
#include "gimpcontext.h"
|
|
#include "gimpcontainer.h"
|
|
#include "gimpdrawable-floating-selection.h"
|
|
#include "gimperror.h"
|
|
#include "gimpgrouplayer.h"
|
|
#include "gimpimage-undo-push.h"
|
|
#include "gimpimage-undo.h"
|
|
#include "gimpimage.h"
|
|
#include "gimpimage-color-profile.h"
|
|
#include "gimplayer-floating-selection.h"
|
|
#include "gimplayer.h"
|
|
#include "gimplayermask.h"
|
|
#include "gimpobjectqueue.h"
|
|
#include "gimppickable.h"
|
|
#include "gimpprogress.h"
|
|
|
|
#include "gimp-intl.h"
|
|
|
|
|
|
enum
|
|
{
|
|
OPACITY_CHANGED,
|
|
MODE_CHANGED,
|
|
BLEND_SPACE_CHANGED,
|
|
COMPOSITE_SPACE_CHANGED,
|
|
COMPOSITE_MODE_CHANGED,
|
|
EFFECTIVE_MODE_CHANGED,
|
|
EXCLUDES_BACKDROP_CHANGED,
|
|
LOCK_ALPHA_CHANGED,
|
|
MASK_CHANGED,
|
|
APPLY_MASK_CHANGED,
|
|
EDIT_MASK_CHANGED,
|
|
SHOW_MASK_CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_OPACITY,
|
|
PROP_MODE,
|
|
PROP_BLEND_SPACE,
|
|
PROP_COMPOSITE_SPACE,
|
|
PROP_COMPOSITE_MODE,
|
|
PROP_EXCLUDES_BACKDROP,
|
|
PROP_LOCK_ALPHA,
|
|
PROP_MASK,
|
|
PROP_FLOATING_SELECTION
|
|
};
|
|
|
|
|
|
static void gimp_color_managed_iface_init (GimpColorManagedInterface *iface);
|
|
static void gimp_pickable_iface_init (GimpPickableInterface *iface);
|
|
|
|
static void gimp_layer_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_layer_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_layer_dispose (GObject *object);
|
|
static void gimp_layer_finalize (GObject *object);
|
|
static void gimp_layer_notify (GObject *object,
|
|
GParamSpec *pspec);
|
|
|
|
static void gimp_layer_name_changed (GimpObject *object);
|
|
static gint64 gimp_layer_get_memsize (GimpObject *object,
|
|
gint64 *gui_size);
|
|
|
|
static void gimp_layer_invalidate_preview (GimpViewable *viewable);
|
|
static gchar * gimp_layer_get_description (GimpViewable *viewable,
|
|
gchar **tooltip);
|
|
|
|
static GeglNode * gimp_layer_get_node (GimpFilter *filter);
|
|
|
|
static void gimp_layer_removed (GimpItem *item);
|
|
static void gimp_layer_unset_removed (GimpItem *item);
|
|
static gboolean gimp_layer_is_attached (GimpItem *item);
|
|
static GimpItemTree * gimp_layer_get_tree (GimpItem *item);
|
|
static GimpItem * gimp_layer_duplicate (GimpItem *item,
|
|
GType new_type);
|
|
static void gimp_layer_convert (GimpItem *item,
|
|
GimpImage *dest_image,
|
|
GType old_type);
|
|
static gboolean gimp_layer_rename (GimpItem *item,
|
|
const gchar *new_name,
|
|
const gchar *undo_desc,
|
|
GError **error);
|
|
static void gimp_layer_start_move (GimpItem *item,
|
|
gboolean push_undo);
|
|
static void gimp_layer_end_move (GimpItem *item,
|
|
gboolean push_undo);
|
|
static void gimp_layer_translate (GimpItem *item,
|
|
gdouble offset_x,
|
|
gdouble offset_y,
|
|
gboolean push_undo);
|
|
static void gimp_layer_scale (GimpItem *item,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint new_offset_x,
|
|
gint new_offset_y,
|
|
GimpInterpolationType interp_type,
|
|
GimpProgress *progress);
|
|
static void gimp_layer_resize (GimpItem *item,
|
|
GimpContext *context,
|
|
GimpFillType fill_type,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint offset_x,
|
|
gint offset_y);
|
|
static void gimp_layer_flip (GimpItem *item,
|
|
GimpContext *context,
|
|
GimpOrientationType flip_type,
|
|
gdouble axis,
|
|
gboolean clip_result);
|
|
static void gimp_layer_rotate (GimpItem *item,
|
|
GimpContext *context,
|
|
GimpRotationType rotate_type,
|
|
gdouble center_x,
|
|
gdouble center_y,
|
|
gboolean clip_result);
|
|
static void gimp_layer_transform (GimpItem *item,
|
|
GimpContext *context,
|
|
const GimpMatrix3 *matrix,
|
|
GimpTransformDirection direction,
|
|
GimpInterpolationType interpolation_type,
|
|
GimpTransformResize clip_result,
|
|
GimpProgress *progress,
|
|
gboolean push_undo);
|
|
static void gimp_layer_to_selection (GimpItem *item,
|
|
GimpChannelOps op,
|
|
gboolean antialias,
|
|
gboolean feather,
|
|
gdouble feather_radius_x,
|
|
gdouble feather_radius_y);
|
|
|
|
static void gimp_layer_alpha_changed (GimpDrawable *drawable);
|
|
static gint64 gimp_layer_estimate_memsize (GimpDrawable *drawable,
|
|
GimpComponentType component_type,
|
|
gint width,
|
|
gint height);
|
|
static gboolean gimp_layer_supports_alpha (GimpDrawable *drawable);
|
|
static void gimp_layer_convert_type (GimpDrawable *drawable,
|
|
GimpImage *dest_image,
|
|
const Babl *new_format,
|
|
GimpColorProfile *src_profile,
|
|
GimpColorProfile *dest_profile,
|
|
GeglDitherMethod layer_dither_type,
|
|
GeglDitherMethod mask_dither_type,
|
|
gboolean push_undo,
|
|
GimpProgress *progress);
|
|
static void gimp_layer_invalidate_boundary (GimpDrawable *drawable);
|
|
static void gimp_layer_get_active_components (GimpDrawable *drawable,
|
|
gboolean *active);
|
|
static GimpComponentMask
|
|
gimp_layer_get_active_mask (GimpDrawable *drawable);
|
|
static void gimp_layer_set_buffer (GimpDrawable *drawable,
|
|
gboolean push_undo,
|
|
const gchar *undo_desc,
|
|
GeglBuffer *buffer,
|
|
const GeglRectangle *bounds);
|
|
static GeglRectangle
|
|
gimp_layer_get_bounding_box (GimpDrawable *drawable);
|
|
|
|
static GimpColorProfile *
|
|
gimp_layer_get_color_profile (GimpColorManaged *managed);
|
|
|
|
static gdouble gimp_layer_get_opacity_at (GimpPickable *pickable,
|
|
gint x,
|
|
gint y);
|
|
|
|
static gboolean gimp_layer_real_is_alpha_locked (GimpLayer *layer,
|
|
GimpLayer **locked_layer);
|
|
static void gimp_layer_real_translate (GimpLayer *layer,
|
|
gint offset_x,
|
|
gint offset_y);
|
|
static void gimp_layer_real_scale (GimpLayer *layer,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint new_offset_x,
|
|
gint new_offset_y,
|
|
GimpInterpolationType interp_type,
|
|
GimpProgress *progress);
|
|
static void gimp_layer_real_resize (GimpLayer *layer,
|
|
GimpContext *context,
|
|
GimpFillType fill_type,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint offset_x,
|
|
gint offset_y);
|
|
static void gimp_layer_real_flip (GimpLayer *layer,
|
|
GimpContext *context,
|
|
GimpOrientationType flip_type,
|
|
gdouble axis,
|
|
gboolean clip_result);
|
|
static void gimp_layer_real_rotate (GimpLayer *layer,
|
|
GimpContext *context,
|
|
GimpRotationType rotate_type,
|
|
gdouble center_x,
|
|
gdouble center_y,
|
|
gboolean clip_result);
|
|
static void gimp_layer_real_transform (GimpLayer *layer,
|
|
GimpContext *context,
|
|
const GimpMatrix3 *matrix,
|
|
GimpTransformDirection direction,
|
|
GimpInterpolationType interpolation_type,
|
|
GimpTransformResize clip_result,
|
|
GimpProgress *progress,
|
|
gboolean push_undo);
|
|
static void gimp_layer_real_convert_type (GimpLayer *layer,
|
|
GimpImage *dest_image,
|
|
const Babl *new_format,
|
|
GimpColorProfile *src_profile,
|
|
GimpColorProfile *dest_profile,
|
|
GeglDitherMethod layer_dither_type,
|
|
GeglDitherMethod mask_dither_type,
|
|
gboolean push_undo,
|
|
GimpProgress *progress);
|
|
static GeglRectangle
|
|
gimp_layer_real_get_bounding_box (GimpLayer *layer);
|
|
static void gimp_layer_real_get_effective_mode (GimpLayer *layer,
|
|
GimpLayerMode *mode,
|
|
GimpLayerColorSpace *blend_space,
|
|
GimpLayerColorSpace *composite_space,
|
|
GimpLayerCompositeMode *composite_mode);
|
|
static gboolean
|
|
gimp_layer_real_get_excludes_backdrop (GimpLayer *layer);
|
|
|
|
static void gimp_layer_layer_mask_update (GimpDrawable *layer_mask,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height,
|
|
GimpLayer *layer);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GimpLayer, gimp_layer, GIMP_TYPE_DRAWABLE,
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_COLOR_MANAGED,
|
|
gimp_color_managed_iface_init)
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_PICKABLE,
|
|
gimp_pickable_iface_init))
|
|
|
|
#define parent_class gimp_layer_parent_class
|
|
|
|
static guint layer_signals[LAST_SIGNAL] = { 0 };
|
|
|
|
|
|
static void
|
|
gimp_layer_class_init (GimpLayerClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GimpObjectClass *gimp_object_class = GIMP_OBJECT_CLASS (klass);
|
|
GimpViewableClass *viewable_class = GIMP_VIEWABLE_CLASS (klass);
|
|
GimpFilterClass *filter_class = GIMP_FILTER_CLASS (klass);
|
|
GimpItemClass *item_class = GIMP_ITEM_CLASS (klass);
|
|
GimpDrawableClass *drawable_class = GIMP_DRAWABLE_CLASS (klass);
|
|
|
|
layer_signals[OPACITY_CHANGED] =
|
|
g_signal_new ("opacity-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, opacity_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[MODE_CHANGED] =
|
|
g_signal_new ("mode-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, mode_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[BLEND_SPACE_CHANGED] =
|
|
g_signal_new ("blend-space-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, blend_space_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[COMPOSITE_SPACE_CHANGED] =
|
|
g_signal_new ("composite-space-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, composite_space_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[COMPOSITE_MODE_CHANGED] =
|
|
g_signal_new ("composite-mode-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, composite_mode_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[EFFECTIVE_MODE_CHANGED] =
|
|
g_signal_new ("effective-mode-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, effective_mode_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[EXCLUDES_BACKDROP_CHANGED] =
|
|
g_signal_new ("excludes-backdrop-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, excludes_backdrop_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[LOCK_ALPHA_CHANGED] =
|
|
g_signal_new ("lock-alpha-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, lock_alpha_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[MASK_CHANGED] =
|
|
g_signal_new ("mask-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, mask_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[APPLY_MASK_CHANGED] =
|
|
g_signal_new ("apply-mask-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, apply_mask_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[EDIT_MASK_CHANGED] =
|
|
g_signal_new ("edit-mask-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, edit_mask_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
layer_signals[SHOW_MASK_CHANGED] =
|
|
g_signal_new ("show-mask-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpLayerClass, show_mask_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
object_class->set_property = gimp_layer_set_property;
|
|
object_class->get_property = gimp_layer_get_property;
|
|
object_class->dispose = gimp_layer_dispose;
|
|
object_class->finalize = gimp_layer_finalize;
|
|
object_class->notify = gimp_layer_notify;
|
|
|
|
gimp_object_class->name_changed = gimp_layer_name_changed;
|
|
gimp_object_class->get_memsize = gimp_layer_get_memsize;
|
|
|
|
viewable_class->default_icon_name = "gimp-layer";
|
|
viewable_class->default_name = _("Layer");
|
|
viewable_class->invalidate_preview = gimp_layer_invalidate_preview;
|
|
viewable_class->get_description = gimp_layer_get_description;
|
|
|
|
filter_class->get_node = gimp_layer_get_node;
|
|
|
|
item_class->removed = gimp_layer_removed;
|
|
item_class->unset_removed = gimp_layer_unset_removed;
|
|
item_class->is_attached = gimp_layer_is_attached;
|
|
item_class->get_tree = gimp_layer_get_tree;
|
|
item_class->duplicate = gimp_layer_duplicate;
|
|
item_class->convert = gimp_layer_convert;
|
|
item_class->rename = gimp_layer_rename;
|
|
item_class->start_move = gimp_layer_start_move;
|
|
item_class->end_move = gimp_layer_end_move;
|
|
item_class->translate = gimp_layer_translate;
|
|
item_class->scale = gimp_layer_scale;
|
|
item_class->resize = gimp_layer_resize;
|
|
item_class->flip = gimp_layer_flip;
|
|
item_class->rotate = gimp_layer_rotate;
|
|
item_class->transform = gimp_layer_transform;
|
|
item_class->to_selection = gimp_layer_to_selection;
|
|
item_class->rename_desc = C_("undo-type", "Rename Layer");
|
|
item_class->translate_desc = C_("undo-type", "Move Layer");
|
|
item_class->scale_desc = C_("undo-type", "Scale Layer");
|
|
item_class->resize_desc = C_("undo-type", "Resize Layer");
|
|
item_class->flip_desc = C_("undo-type", "Flip Layer");
|
|
item_class->rotate_desc = C_("undo-type", "Rotate Layer");
|
|
item_class->transform_desc = C_("undo-type", "Transform Layer");
|
|
item_class->to_selection_desc = C_("undo-type", "Alpha to Selection");
|
|
item_class->reorder_desc = C_("undo-type", "Reorder Layer");
|
|
item_class->raise_desc = C_("undo-type", "Raise Layer");
|
|
item_class->raise_to_top_desc = C_("undo-type", "Raise Layer to Top");
|
|
item_class->lower_desc = C_("undo-type", "Lower Layer");
|
|
item_class->lower_to_bottom_desc = C_("undo-type", "Lower Layer to Bottom");
|
|
item_class->raise_failed = _("Layer cannot be raised higher.");
|
|
item_class->lower_failed = _("Layer cannot be lowered more.");
|
|
|
|
drawable_class->alpha_changed = gimp_layer_alpha_changed;
|
|
drawable_class->estimate_memsize = gimp_layer_estimate_memsize;
|
|
drawable_class->supports_alpha = gimp_layer_supports_alpha;
|
|
drawable_class->convert_type = gimp_layer_convert_type;
|
|
drawable_class->invalidate_boundary = gimp_layer_invalidate_boundary;
|
|
drawable_class->get_active_components = gimp_layer_get_active_components;
|
|
drawable_class->get_active_mask = gimp_layer_get_active_mask;
|
|
drawable_class->set_buffer = gimp_layer_set_buffer;
|
|
drawable_class->get_bounding_box = gimp_layer_get_bounding_box;
|
|
|
|
klass->opacity_changed = NULL;
|
|
klass->mode_changed = NULL;
|
|
klass->blend_space_changed = NULL;
|
|
klass->composite_space_changed = NULL;
|
|
klass->composite_mode_changed = NULL;
|
|
klass->excludes_backdrop_changed = NULL;
|
|
klass->lock_alpha_changed = NULL;
|
|
klass->mask_changed = NULL;
|
|
klass->apply_mask_changed = NULL;
|
|
klass->edit_mask_changed = NULL;
|
|
klass->show_mask_changed = NULL;
|
|
klass->is_alpha_locked = gimp_layer_real_is_alpha_locked;
|
|
klass->translate = gimp_layer_real_translate;
|
|
klass->scale = gimp_layer_real_scale;
|
|
klass->resize = gimp_layer_real_resize;
|
|
klass->flip = gimp_layer_real_flip;
|
|
klass->rotate = gimp_layer_real_rotate;
|
|
klass->transform = gimp_layer_real_transform;
|
|
klass->convert_type = gimp_layer_real_convert_type;
|
|
klass->get_bounding_box = gimp_layer_real_get_bounding_box;
|
|
klass->get_effective_mode = gimp_layer_real_get_effective_mode;
|
|
klass->get_excludes_backdrop = gimp_layer_real_get_excludes_backdrop;
|
|
|
|
g_object_class_install_property (object_class, PROP_OPACITY,
|
|
g_param_spec_double ("opacity", NULL, NULL,
|
|
GIMP_OPACITY_TRANSPARENT,
|
|
GIMP_OPACITY_OPAQUE,
|
|
GIMP_OPACITY_OPAQUE,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_MODE,
|
|
g_param_spec_enum ("mode", NULL, NULL,
|
|
GIMP_TYPE_LAYER_MODE,
|
|
GIMP_LAYER_MODE_NORMAL,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_BLEND_SPACE,
|
|
g_param_spec_enum ("blend-space",
|
|
NULL, NULL,
|
|
GIMP_TYPE_LAYER_COLOR_SPACE,
|
|
GIMP_LAYER_COLOR_SPACE_AUTO,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_COMPOSITE_SPACE,
|
|
g_param_spec_enum ("composite-space",
|
|
NULL, NULL,
|
|
GIMP_TYPE_LAYER_COLOR_SPACE,
|
|
GIMP_LAYER_COLOR_SPACE_AUTO,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_COMPOSITE_MODE,
|
|
g_param_spec_enum ("composite-mode",
|
|
NULL, NULL,
|
|
GIMP_TYPE_LAYER_COMPOSITE_MODE,
|
|
GIMP_LAYER_COMPOSITE_AUTO,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_EXCLUDES_BACKDROP,
|
|
g_param_spec_boolean ("excludes-backdrop",
|
|
NULL, NULL,
|
|
FALSE,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_LOCK_ALPHA,
|
|
g_param_spec_boolean ("lock-alpha",
|
|
NULL, NULL,
|
|
FALSE,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_MASK,
|
|
g_param_spec_object ("mask",
|
|
NULL, NULL,
|
|
GIMP_TYPE_LAYER_MASK,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_FLOATING_SELECTION,
|
|
g_param_spec_boolean ("floating-selection",
|
|
NULL, NULL,
|
|
FALSE,
|
|
GIMP_PARAM_READABLE));
|
|
}
|
|
|
|
static void
|
|
gimp_layer_init (GimpLayer *layer)
|
|
{
|
|
layer->opacity = GIMP_OPACITY_OPAQUE;
|
|
layer->mode = GIMP_LAYER_MODE_NORMAL;
|
|
layer->blend_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
layer->composite_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
layer->composite_mode = GIMP_LAYER_COMPOSITE_AUTO;
|
|
layer->effective_mode = layer->mode;
|
|
layer->effective_blend_space = gimp_layer_get_real_blend_space (layer);
|
|
layer->effective_composite_space = gimp_layer_get_real_composite_space (layer);
|
|
layer->effective_composite_mode = gimp_layer_get_real_composite_mode (layer);
|
|
layer->excludes_backdrop = FALSE;
|
|
layer->lock_alpha = FALSE;
|
|
|
|
layer->mask = NULL;
|
|
layer->apply_mask = TRUE;
|
|
layer->edit_mask = TRUE;
|
|
layer->show_mask = FALSE;
|
|
|
|
/* floating selection */
|
|
layer->fs.drawable = NULL;
|
|
layer->fs.boundary_known = FALSE;
|
|
layer->fs.segs = NULL;
|
|
layer->fs.num_segs = 0;
|
|
}
|
|
|
|
static void
|
|
gimp_color_managed_iface_init (GimpColorManagedInterface *iface)
|
|
{
|
|
iface->get_color_profile = gimp_layer_get_color_profile;
|
|
}
|
|
|
|
static void
|
|
gimp_pickable_iface_init (GimpPickableInterface *iface)
|
|
{
|
|
iface->get_opacity_at = gimp_layer_get_opacity_at;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_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_layer_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_OPACITY:
|
|
g_value_set_double (value, gimp_layer_get_opacity (layer));
|
|
break;
|
|
case PROP_MODE:
|
|
g_value_set_enum (value, gimp_layer_get_mode (layer));
|
|
break;
|
|
case PROP_BLEND_SPACE:
|
|
g_value_set_enum (value, gimp_layer_get_blend_space (layer));
|
|
break;
|
|
case PROP_COMPOSITE_SPACE:
|
|
g_value_set_enum (value, gimp_layer_get_composite_space (layer));
|
|
break;
|
|
case PROP_COMPOSITE_MODE:
|
|
g_value_set_enum (value, gimp_layer_get_composite_mode (layer));
|
|
break;
|
|
case PROP_EXCLUDES_BACKDROP:
|
|
g_value_set_boolean (value, gimp_layer_get_excludes_backdrop (layer));
|
|
break;
|
|
case PROP_LOCK_ALPHA:
|
|
g_value_set_boolean (value, gimp_layer_get_lock_alpha (layer));
|
|
break;
|
|
case PROP_MASK:
|
|
g_value_set_object (value, gimp_layer_get_mask (layer));
|
|
break;
|
|
case PROP_FLOATING_SELECTION:
|
|
g_value_set_boolean (value, gimp_layer_is_floating_sel (layer));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_layer_dispose (GObject *object)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (object);
|
|
|
|
if (layer->mask)
|
|
g_signal_handlers_disconnect_by_func (layer->mask,
|
|
gimp_layer_layer_mask_update,
|
|
layer);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
{
|
|
GimpDrawable *fs_drawable = gimp_layer_get_floating_sel_drawable (layer);
|
|
|
|
/* only detach if this is actually the drawable's fs because the
|
|
* layer might be on the undo stack and not attached to anything
|
|
*/
|
|
if (gimp_drawable_get_floating_sel (fs_drawable) == layer)
|
|
gimp_drawable_detach_floating_sel (fs_drawable);
|
|
|
|
gimp_layer_set_floating_sel_drawable (layer, NULL);
|
|
}
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_finalize (GObject *object)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (object);
|
|
|
|
g_clear_object (&layer->mask);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_update_mode_node (GimpLayer *layer)
|
|
{
|
|
GeglNode *mode_node;
|
|
GimpLayerMode visible_mode;
|
|
GimpLayerColorSpace visible_blend_space;
|
|
GimpLayerColorSpace visible_composite_space;
|
|
GimpLayerCompositeMode visible_composite_mode;
|
|
|
|
mode_node = gimp_drawable_get_mode_node (GIMP_DRAWABLE (layer));
|
|
|
|
if (layer->mask && layer->show_mask)
|
|
{
|
|
visible_mode = GIMP_LAYER_MODE_NORMAL;
|
|
visible_blend_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
visible_composite_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
visible_composite_mode = GIMP_LAYER_COMPOSITE_AUTO;
|
|
|
|
/* This makes sure that masks of LEGACY-mode layers are
|
|
* composited in PERCEPTUAL space, and non-LEGACY layers in
|
|
* LINEAR space, or whatever composite space was chosen in the
|
|
* layer attributes dialog
|
|
*/
|
|
visible_composite_space = gimp_layer_get_real_composite_space (layer);
|
|
}
|
|
else
|
|
{
|
|
visible_mode = layer->effective_mode;
|
|
visible_blend_space = layer->effective_blend_space;
|
|
visible_composite_space = layer->effective_composite_space;
|
|
visible_composite_mode = layer->effective_composite_mode;
|
|
}
|
|
|
|
gimp_gegl_mode_node_set_mode (mode_node,
|
|
visible_mode,
|
|
visible_blend_space,
|
|
visible_composite_space,
|
|
visible_composite_mode);
|
|
gimp_gegl_mode_node_set_opacity (mode_node, layer->opacity);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_notify (GObject *object,
|
|
GParamSpec *pspec)
|
|
{
|
|
if (! strcmp (pspec->name, "is-last-node") &&
|
|
gimp_filter_peek_node (GIMP_FILTER (object)))
|
|
{
|
|
gimp_layer_update_mode_node (GIMP_LAYER (object));
|
|
|
|
gimp_drawable_update (GIMP_DRAWABLE (object), 0, 0, -1, -1);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_layer_name_changed (GimpObject *object)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (object);
|
|
|
|
if (GIMP_OBJECT_CLASS (parent_class)->name_changed)
|
|
GIMP_OBJECT_CLASS (parent_class)->name_changed (object);
|
|
|
|
if (layer->mask)
|
|
{
|
|
gchar *mask_name = g_strdup_printf (_("%s mask"),
|
|
gimp_object_get_name (object));
|
|
|
|
gimp_object_take_name (GIMP_OBJECT (layer->mask), mask_name);
|
|
}
|
|
}
|
|
|
|
static gint64
|
|
gimp_layer_get_memsize (GimpObject *object,
|
|
gint64 *gui_size)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (object);
|
|
gint64 memsize = 0;
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (layer->mask), gui_size);
|
|
|
|
*gui_size += layer->fs.num_segs * sizeof (GimpBoundSeg);
|
|
|
|
return memsize + GIMP_OBJECT_CLASS (parent_class)->get_memsize (object,
|
|
gui_size);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_invalidate_preview (GimpViewable *viewable)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (viewable);
|
|
|
|
GIMP_VIEWABLE_CLASS (parent_class)->invalidate_preview (viewable);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
floating_sel_invalidate (layer);
|
|
}
|
|
|
|
static gchar *
|
|
gimp_layer_get_description (GimpViewable *viewable,
|
|
gchar **tooltip)
|
|
{
|
|
if (gimp_layer_is_floating_sel (GIMP_LAYER (viewable)))
|
|
{
|
|
GimpDrawable *drawable;
|
|
const gchar *header = _("Floating Selection");
|
|
|
|
drawable = gimp_layer_get_floating_sel_drawable (GIMP_LAYER (viewable));
|
|
if (GIMP_IS_LAYER_MASK (drawable))
|
|
header = _("Floating Mask");
|
|
else if (GIMP_IS_LAYER (drawable))
|
|
header = _("Floating Layer");
|
|
/* TRANSLATORS: the first %s will be the type of floating item, i.e.
|
|
* either a "Floating Layer" or "Floating Mask" usually. The second will
|
|
* be a layer name.
|
|
*/
|
|
return g_strdup_printf (_("%s\n(%s)"),
|
|
header, gimp_object_get_name (viewable));
|
|
}
|
|
|
|
return GIMP_VIEWABLE_CLASS (parent_class)->get_description (viewable,
|
|
tooltip);
|
|
}
|
|
|
|
static GeglNode *
|
|
gimp_layer_get_node (GimpFilter *filter)
|
|
{
|
|
GimpDrawable *drawable = GIMP_DRAWABLE (filter);
|
|
GimpLayer *layer = GIMP_LAYER (filter);
|
|
GeglNode *node;
|
|
GeglNode *input;
|
|
GeglNode *source;
|
|
GeglNode *mode_node;
|
|
gboolean source_node_hijacked = FALSE;
|
|
|
|
node = GIMP_FILTER_CLASS (parent_class)->get_node (filter);
|
|
|
|
input = gegl_node_get_input_proxy (node, "input");
|
|
|
|
source = gimp_drawable_get_source_node (drawable);
|
|
|
|
/* if the source node already has a parent, we are a floating
|
|
* selection and the source node has been hijacked by the fs'
|
|
* drawable
|
|
*/
|
|
if (gegl_node_get_parent (source))
|
|
source_node_hijacked = TRUE;
|
|
|
|
if (! source_node_hijacked)
|
|
gegl_node_add_child (node, source);
|
|
|
|
gegl_node_connect (input, "output", source, "input");
|
|
|
|
g_warn_if_fail (layer->layer_offset_node == NULL);
|
|
g_warn_if_fail (layer->mask_offset_node == NULL);
|
|
|
|
/* the mode node connects it all, and has aux and aux2 inputs for
|
|
* the layer and its mask
|
|
*/
|
|
mode_node = gimp_drawable_get_mode_node (drawable);
|
|
gimp_layer_update_mode_node (layer);
|
|
|
|
/* the layer's offset node */
|
|
layer->layer_offset_node = gegl_node_new_child (node,
|
|
"operation", "gegl:translate",
|
|
NULL);
|
|
gimp_item_add_offset_node (GIMP_ITEM (layer), layer->layer_offset_node);
|
|
|
|
/* the layer mask's offset node */
|
|
layer->mask_offset_node = gegl_node_new_child (node,
|
|
"operation", "gegl:translate",
|
|
NULL);
|
|
gimp_item_add_offset_node (GIMP_ITEM (layer), layer->mask_offset_node);
|
|
|
|
if (! source_node_hijacked)
|
|
{
|
|
gegl_node_connect (source, "output",
|
|
layer->layer_offset_node, "input");
|
|
}
|
|
|
|
if (! (layer->mask && gimp_layer_get_show_mask (layer)))
|
|
{
|
|
gegl_node_connect (layer->layer_offset_node, "output",
|
|
mode_node, "aux");
|
|
}
|
|
|
|
if (layer->mask)
|
|
{
|
|
GeglNode *mask;
|
|
|
|
mask = gimp_drawable_get_source_node (GIMP_DRAWABLE (layer->mask));
|
|
|
|
gegl_node_connect (mask, "output",
|
|
layer->mask_offset_node, "input");
|
|
|
|
if (gimp_layer_get_show_mask (layer))
|
|
{
|
|
gegl_node_connect (layer->mask_offset_node, "output",
|
|
mode_node, "aux");
|
|
}
|
|
else if (gimp_layer_get_apply_mask (layer))
|
|
{
|
|
gegl_node_connect (layer->mask_offset_node, "output",
|
|
mode_node, "aux2");
|
|
}
|
|
}
|
|
|
|
return node;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_removed (GimpItem *item)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
|
|
if (layer->mask)
|
|
gimp_item_removed (GIMP_ITEM (layer->mask));
|
|
|
|
if (GIMP_ITEM_CLASS (parent_class)->removed)
|
|
GIMP_ITEM_CLASS (parent_class)->removed (item);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_unset_removed (GimpItem *item)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
|
|
if (layer->mask)
|
|
gimp_item_unset_removed (GIMP_ITEM (layer->mask));
|
|
|
|
if (GIMP_ITEM_CLASS (parent_class)->unset_removed)
|
|
GIMP_ITEM_CLASS (parent_class)->unset_removed (item);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_layer_is_attached (GimpItem *item)
|
|
{
|
|
GimpImage *image = gimp_item_get_image (item);
|
|
|
|
return (GIMP_IS_IMAGE (image) &&
|
|
gimp_container_have (gimp_image_get_layers (image),
|
|
GIMP_OBJECT (item)));
|
|
}
|
|
|
|
static GimpItemTree *
|
|
gimp_layer_get_tree (GimpItem *item)
|
|
{
|
|
if (gimp_item_is_attached (item))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (item);
|
|
|
|
return gimp_image_get_layer_tree (image);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static GimpItem *
|
|
gimp_layer_duplicate (GimpItem *item,
|
|
GType new_type)
|
|
{
|
|
GimpItem *new_item;
|
|
|
|
g_return_val_if_fail (g_type_is_a (new_type, GIMP_TYPE_DRAWABLE), NULL);
|
|
|
|
new_item = GIMP_ITEM_CLASS (parent_class)->duplicate (item, new_type);
|
|
|
|
if (GIMP_IS_LAYER (new_item))
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GimpLayer *new_layer = GIMP_LAYER (new_item);
|
|
|
|
/* PASS_THROUGH mode is invalid for regular layers.
|
|
* We used to change the mode to NORMAL *before* duplicating (see
|
|
* #793714 on bugzilla) but it would change the image's render.
|
|
* Instead we first duplicate so that the group's render is used
|
|
* as-is for the non-group duplicate layer. Then we set NORMAL
|
|
* mode.
|
|
*/
|
|
if (gimp_layer_get_mode (layer) == GIMP_LAYER_MODE_PASS_THROUGH &&
|
|
! GIMP_IS_GROUP_LAYER (new_item))
|
|
{
|
|
GimpLayerColorSpace blend_space;
|
|
GimpLayerColorSpace composite_space;
|
|
GimpLayerCompositeMode composite_mode;
|
|
|
|
/* keep the group's current blend space, composite space, and composite
|
|
* mode.
|
|
*/
|
|
blend_space = gimp_layer_get_blend_space (layer);
|
|
composite_space = gimp_layer_get_composite_space (layer);
|
|
composite_mode = gimp_layer_get_composite_mode (layer);
|
|
|
|
gimp_layer_set_mode (new_layer, GIMP_LAYER_MODE_NORMAL, FALSE);
|
|
gimp_layer_set_blend_space (new_layer, blend_space, FALSE);
|
|
gimp_layer_set_composite_space (new_layer, composite_space, FALSE);
|
|
gimp_layer_set_composite_mode (new_layer, composite_mode, FALSE);
|
|
gimp_layer_set_opacity (new_layer, 1.0, FALSE);
|
|
}
|
|
else
|
|
{
|
|
gimp_layer_set_mode (new_layer,
|
|
gimp_layer_get_mode (layer), FALSE);
|
|
gimp_layer_set_blend_space (new_layer,
|
|
gimp_layer_get_blend_space (layer), FALSE);
|
|
gimp_layer_set_composite_space (new_layer,
|
|
gimp_layer_get_composite_space (layer), FALSE);
|
|
gimp_layer_set_composite_mode (new_layer,
|
|
gimp_layer_get_composite_mode (layer), FALSE);
|
|
gimp_layer_set_opacity (new_layer,
|
|
gimp_layer_get_opacity (layer), FALSE);
|
|
}
|
|
|
|
if (gimp_layer_can_lock_alpha (new_layer))
|
|
gimp_layer_set_lock_alpha (new_layer,
|
|
gimp_layer_get_lock_alpha (layer), FALSE);
|
|
|
|
/* duplicate the layer mask if necessary */
|
|
if (layer->mask)
|
|
{
|
|
GimpItem *mask;
|
|
|
|
mask = gimp_item_duplicate (GIMP_ITEM (layer->mask),
|
|
G_TYPE_FROM_INSTANCE (layer->mask));
|
|
gimp_layer_add_mask (new_layer, GIMP_LAYER_MASK (mask),
|
|
layer->edit_mask, FALSE, NULL);
|
|
|
|
new_layer->apply_mask = layer->apply_mask;
|
|
new_layer->show_mask = layer->show_mask;
|
|
}
|
|
}
|
|
|
|
return new_item;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_convert (GimpItem *item,
|
|
GimpImage *dest_image,
|
|
GType old_type)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GimpDrawable *drawable = GIMP_DRAWABLE (item);
|
|
GimpImageBaseType old_base_type;
|
|
GimpImageBaseType new_base_type;
|
|
GimpPrecision old_precision;
|
|
GimpPrecision new_precision;
|
|
GimpColorProfile *src_profile = NULL;
|
|
GimpColorProfile *dest_profile = NULL;
|
|
|
|
old_base_type = gimp_drawable_get_base_type (drawable);
|
|
new_base_type = gimp_image_get_base_type (dest_image);
|
|
|
|
old_precision = gimp_drawable_get_precision (drawable);
|
|
new_precision = gimp_image_get_precision (dest_image);
|
|
|
|
if (g_type_is_a (old_type, GIMP_TYPE_LAYER))
|
|
{
|
|
src_profile =
|
|
gimp_color_managed_get_color_profile (GIMP_COLOR_MANAGED (item));
|
|
|
|
dest_profile =
|
|
gimp_color_managed_get_color_profile (GIMP_COLOR_MANAGED (dest_image));
|
|
|
|
if (gimp_color_profile_is_equal (dest_profile, src_profile))
|
|
{
|
|
src_profile = NULL;
|
|
dest_profile = NULL;
|
|
}
|
|
}
|
|
|
|
if (old_base_type != new_base_type ||
|
|
old_precision != new_precision ||
|
|
dest_profile)
|
|
{
|
|
gimp_drawable_convert_type (drawable, dest_image,
|
|
new_base_type,
|
|
new_precision,
|
|
gimp_drawable_has_alpha (drawable),
|
|
src_profile,
|
|
dest_profile,
|
|
GEGL_DITHER_NONE, GEGL_DITHER_NONE,
|
|
FALSE, NULL);
|
|
}
|
|
|
|
if (layer->mask)
|
|
gimp_item_set_image (GIMP_ITEM (layer->mask), dest_image);
|
|
|
|
GIMP_ITEM_CLASS (parent_class)->convert (item, dest_image, old_type);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_layer_rename (GimpItem *item,
|
|
const gchar *new_name,
|
|
const gchar *undo_desc,
|
|
GError **error)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GimpImage *image = gimp_item_get_image (item);
|
|
gboolean attached;
|
|
gboolean floating_sel;
|
|
|
|
attached = gimp_item_is_attached (item);
|
|
floating_sel = gimp_layer_is_floating_sel (layer);
|
|
|
|
if (floating_sel)
|
|
{
|
|
if (GIMP_IS_CHANNEL (gimp_layer_get_floating_sel_drawable (layer)))
|
|
{
|
|
g_set_error_literal (error, GIMP_ERROR, GIMP_FAILED,
|
|
_("Cannot create a new layer from the floating "
|
|
"selection because it belongs to a layer mask "
|
|
"or channel."));
|
|
return FALSE;
|
|
}
|
|
|
|
if (attached)
|
|
{
|
|
gimp_image_undo_group_start (image,
|
|
GIMP_UNDO_GROUP_ITEM_PROPERTIES,
|
|
undo_desc);
|
|
|
|
floating_sel_to_layer (layer, NULL);
|
|
}
|
|
}
|
|
|
|
GIMP_ITEM_CLASS (parent_class)->rename (item, new_name, undo_desc, error);
|
|
|
|
if (attached && floating_sel)
|
|
gimp_image_undo_group_end (image);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_start_move (GimpItem *item,
|
|
gboolean push_undo)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GimpLayer *ancestor = layer;
|
|
GSList *ancestors = NULL;
|
|
|
|
/* suspend mask cropping for all of the layer's ancestors */
|
|
while ((ancestor = gimp_layer_get_parent (ancestor)))
|
|
{
|
|
gimp_group_layer_suspend_mask (GIMP_GROUP_LAYER (ancestor), push_undo);
|
|
|
|
ancestors = g_slist_prepend (ancestors, g_object_ref (ancestor));
|
|
}
|
|
|
|
/* we keep the ancestor list around, so that we can resume mask cropping for
|
|
* the same set of groups in gimp_layer_end_move(). note that
|
|
* gimp_image_remove_layer() calls start_move() before removing the layer,
|
|
* while it's still part of the layer tree, and end_move() afterwards, when
|
|
* it's no longer part of the layer tree, and hence we can't use get_parent()
|
|
* in end_move() to get the same set of ancestors.
|
|
*/
|
|
layer->move_stack = g_slist_prepend (layer->move_stack, ancestors);
|
|
|
|
if (GIMP_ITEM_CLASS (parent_class)->start_move)
|
|
GIMP_ITEM_CLASS (parent_class)->start_move (item, push_undo);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_end_move (GimpItem *item,
|
|
gboolean push_undo)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GSList *ancestors;
|
|
GSList *iter;
|
|
|
|
g_return_if_fail (layer->move_stack != NULL);
|
|
|
|
if (GIMP_ITEM_CLASS (parent_class)->end_move)
|
|
GIMP_ITEM_CLASS (parent_class)->end_move (item, push_undo);
|
|
|
|
ancestors = layer->move_stack->data;
|
|
|
|
layer->move_stack = g_slist_remove (layer->move_stack, ancestors);
|
|
|
|
/* resume mask cropping for all of the layer's ancestors */
|
|
for (iter = ancestors; iter; iter = g_slist_next (iter))
|
|
{
|
|
GimpGroupLayer *ancestor = iter->data;
|
|
|
|
gimp_group_layer_resume_mask (ancestor, push_undo);
|
|
|
|
g_object_unref (ancestor);
|
|
}
|
|
|
|
g_slist_free (ancestors);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_translate (GimpItem *item,
|
|
gdouble offset_x,
|
|
gdouble offset_y,
|
|
gboolean push_undo)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_item_displace (gimp_item_get_image (item), NULL, item);
|
|
|
|
GIMP_LAYER_GET_CLASS (layer)->translate (layer,
|
|
SIGNED_ROUND (offset_x),
|
|
SIGNED_ROUND (offset_y));
|
|
|
|
if (layer->mask)
|
|
{
|
|
gint off_x, off_y;
|
|
|
|
gimp_item_get_offset (item, &off_x, &off_y);
|
|
gimp_item_set_offset (GIMP_ITEM (layer->mask), off_x, off_y);
|
|
|
|
gimp_viewable_invalidate_preview (GIMP_VIEWABLE (layer->mask));
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_layer_scale (GimpItem *item,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint new_offset_x,
|
|
gint new_offset_y,
|
|
GimpInterpolationType interpolation_type,
|
|
GimpProgress *progress)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GimpObjectQueue *queue = NULL;
|
|
|
|
if (progress && layer->mask)
|
|
{
|
|
GimpLayerMask *mask;
|
|
|
|
queue = gimp_object_queue_new (progress);
|
|
progress = GIMP_PROGRESS (queue);
|
|
|
|
/* temporarily set layer->mask to NULL, so that its size won't be counted
|
|
* when pushing the layer to the queue.
|
|
*/
|
|
mask = layer->mask;
|
|
layer->mask = NULL;
|
|
|
|
gimp_object_queue_push (queue, layer);
|
|
gimp_object_queue_push (queue, mask);
|
|
|
|
layer->mask = mask;
|
|
}
|
|
|
|
if (queue)
|
|
gimp_object_queue_pop (queue);
|
|
|
|
GIMP_LAYER_GET_CLASS (layer)->scale (layer, new_width, new_height,
|
|
new_offset_x, new_offset_y,
|
|
interpolation_type, progress);
|
|
|
|
if (layer->mask)
|
|
{
|
|
if (queue)
|
|
gimp_object_queue_pop (queue);
|
|
|
|
gimp_item_scale (GIMP_ITEM (layer->mask),
|
|
new_width, new_height,
|
|
new_offset_x, new_offset_y,
|
|
interpolation_type, progress);
|
|
}
|
|
|
|
g_clear_object (&queue);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_resize (GimpItem *item,
|
|
GimpContext *context,
|
|
GimpFillType fill_type,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint offset_x,
|
|
gint offset_y)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
|
|
GIMP_LAYER_GET_CLASS (layer)->resize (layer, context, fill_type,
|
|
new_width, new_height,
|
|
offset_x, offset_y);
|
|
|
|
if (layer->mask)
|
|
gimp_item_resize (GIMP_ITEM (layer->mask), context, GIMP_FILL_TRANSPARENT,
|
|
new_width, new_height, offset_x, offset_y);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_flip (GimpItem *item,
|
|
GimpContext *context,
|
|
GimpOrientationType flip_type,
|
|
gdouble axis,
|
|
gboolean clip_result)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
|
|
GIMP_LAYER_GET_CLASS (layer)->flip (layer, context, flip_type, axis,
|
|
clip_result);
|
|
|
|
if (layer->mask)
|
|
gimp_item_flip (GIMP_ITEM (layer->mask), context,
|
|
flip_type, axis, clip_result);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_rotate (GimpItem *item,
|
|
GimpContext *context,
|
|
GimpRotationType rotate_type,
|
|
gdouble center_x,
|
|
gdouble center_y,
|
|
gboolean clip_result)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
|
|
GIMP_LAYER_GET_CLASS (layer)->rotate (layer, context,
|
|
rotate_type, center_x, center_y,
|
|
clip_result);
|
|
|
|
if (layer->mask)
|
|
gimp_item_rotate (GIMP_ITEM (layer->mask), context,
|
|
rotate_type, center_x, center_y, clip_result);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_transform (GimpItem *item,
|
|
GimpContext *context,
|
|
const GimpMatrix3 *matrix,
|
|
GimpTransformDirection direction,
|
|
GimpInterpolationType interpolation_type,
|
|
GimpTransformResize clip_result,
|
|
GimpProgress *progress,
|
|
gboolean push_undo)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GimpObjectQueue *queue = NULL;
|
|
|
|
if (progress && layer->mask)
|
|
{
|
|
GimpLayerMask *mask;
|
|
|
|
queue = gimp_object_queue_new (progress);
|
|
progress = GIMP_PROGRESS (queue);
|
|
|
|
/* temporarily set layer->mask to NULL, so that its size won't be counted
|
|
* when pushing the layer to the queue.
|
|
*/
|
|
mask = layer->mask;
|
|
layer->mask = NULL;
|
|
|
|
gimp_object_queue_push (queue, layer);
|
|
gimp_object_queue_push (queue, mask);
|
|
|
|
layer->mask = mask;
|
|
}
|
|
|
|
if (queue)
|
|
gimp_object_queue_pop (queue);
|
|
|
|
GIMP_LAYER_GET_CLASS (layer)->transform (layer, context, matrix, direction,
|
|
interpolation_type,
|
|
clip_result,
|
|
progress, push_undo);
|
|
|
|
if (layer->mask)
|
|
{
|
|
if (queue)
|
|
gimp_object_queue_pop (queue);
|
|
|
|
gimp_item_transform (GIMP_ITEM (layer->mask), context,
|
|
matrix, direction,
|
|
interpolation_type,
|
|
clip_result, progress);
|
|
}
|
|
|
|
g_clear_object (&queue);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_to_selection (GimpItem *item,
|
|
GimpChannelOps op,
|
|
gboolean antialias,
|
|
gboolean feather,
|
|
gdouble feather_radius_x,
|
|
gdouble feather_radius_y)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (item);
|
|
GimpImage *image = gimp_item_get_image (item);
|
|
|
|
gimp_channel_select_alpha (gimp_image_get_mask (image),
|
|
GIMP_DRAWABLE (layer),
|
|
op,
|
|
feather, feather_radius_x, feather_radius_y);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_alpha_changed (GimpDrawable *drawable)
|
|
{
|
|
if (GIMP_DRAWABLE_CLASS (parent_class)->alpha_changed)
|
|
GIMP_DRAWABLE_CLASS (parent_class)->alpha_changed (drawable);
|
|
|
|
/* When we add/remove alpha, whatever cached color transforms in
|
|
* view renderers need to be recreated because they cache the wrong
|
|
* lcms formats. See bug 478528.
|
|
*/
|
|
gimp_color_managed_profile_changed (GIMP_COLOR_MANAGED (drawable));
|
|
}
|
|
|
|
static gint64
|
|
gimp_layer_estimate_memsize (GimpDrawable *drawable,
|
|
GimpComponentType component_type,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (drawable);
|
|
gint64 memsize = 0;
|
|
|
|
if (layer->mask)
|
|
memsize += gimp_drawable_estimate_memsize (GIMP_DRAWABLE (layer->mask),
|
|
component_type,
|
|
width, height);
|
|
|
|
return memsize +
|
|
GIMP_DRAWABLE_CLASS (parent_class)->estimate_memsize (drawable,
|
|
component_type,
|
|
width, height);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_layer_supports_alpha (GimpDrawable *drawable)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_convert_type (GimpDrawable *drawable,
|
|
GimpImage *dest_image,
|
|
const Babl *new_format,
|
|
GimpColorProfile *src_profile,
|
|
GimpColorProfile *dest_profile,
|
|
GeglDitherMethod layer_dither_type,
|
|
GeglDitherMethod mask_dither_type,
|
|
gboolean push_undo,
|
|
GimpProgress *progress)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (drawable);
|
|
GimpObjectQueue *queue = NULL;
|
|
const Babl *dest_space = NULL;
|
|
const Babl *space_format;
|
|
gboolean convert_mask;
|
|
|
|
convert_mask = layer->mask &&
|
|
gimp_babl_format_get_precision (new_format) !=
|
|
gimp_drawable_get_precision (GIMP_DRAWABLE (layer->mask));
|
|
|
|
if (progress && convert_mask)
|
|
{
|
|
GimpLayerMask *mask;
|
|
|
|
queue = gimp_object_queue_new (progress);
|
|
progress = GIMP_PROGRESS (queue);
|
|
|
|
/* temporarily set layer->mask to NULL, so that its size won't be counted
|
|
* when pushing the layer to the queue.
|
|
*/
|
|
mask = layer->mask;
|
|
layer->mask = NULL;
|
|
|
|
gimp_object_queue_push (queue, layer);
|
|
gimp_object_queue_push (queue, mask);
|
|
|
|
layer->mask = mask;
|
|
}
|
|
|
|
if (queue)
|
|
gimp_object_queue_pop (queue);
|
|
|
|
/* when called with a dest_profile, always use the space from that
|
|
* profile, we can't use gimp_image_get_layer_space() during an
|
|
* image type or precision conversion
|
|
*/
|
|
if (dest_profile)
|
|
{
|
|
dest_space =
|
|
gimp_color_profile_get_space (dest_profile,
|
|
GIMP_COLOR_RENDERING_INTENT_RELATIVE_COLORIMETRIC,
|
|
NULL);
|
|
}
|
|
|
|
space_format = babl_format_with_space ((const gchar *) new_format, dest_space);
|
|
|
|
GIMP_LAYER_GET_CLASS (layer)->convert_type (layer, dest_image, space_format,
|
|
src_profile, dest_profile,
|
|
layer_dither_type,
|
|
mask_dither_type, push_undo,
|
|
progress);
|
|
|
|
if (convert_mask)
|
|
{
|
|
if (queue)
|
|
gimp_object_queue_pop (queue);
|
|
|
|
gimp_drawable_convert_type (GIMP_DRAWABLE (layer->mask), dest_image,
|
|
GIMP_GRAY,
|
|
gimp_babl_format_get_precision (new_format),
|
|
gimp_drawable_has_alpha (GIMP_DRAWABLE (layer->mask)),
|
|
NULL, NULL,
|
|
layer_dither_type, mask_dither_type,
|
|
push_undo, progress);
|
|
}
|
|
|
|
g_clear_object (&queue);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_invalidate_boundary (GimpDrawable *drawable)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (drawable);
|
|
|
|
if (gimp_item_is_attached (GIMP_ITEM (drawable)) &&
|
|
gimp_item_is_visible (GIMP_ITEM (drawable)))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (drawable));
|
|
GimpChannel *mask = gimp_image_get_mask (image);
|
|
|
|
/* Turn the current selection off */
|
|
gimp_image_selection_invalidate (image);
|
|
|
|
/* Only bother with the bounds if there is a selection */
|
|
if (! gimp_channel_is_empty (mask))
|
|
{
|
|
mask->bounds_known = FALSE;
|
|
mask->boundary_known = FALSE;
|
|
}
|
|
}
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
floating_sel_invalidate (layer);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_get_active_components (GimpDrawable *drawable,
|
|
gboolean *active)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (drawable);
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (drawable));
|
|
const Babl *format = gimp_drawable_get_format (drawable);
|
|
|
|
/* first copy the image active channels */
|
|
gimp_image_get_active_array (image, active);
|
|
|
|
if (gimp_drawable_has_alpha (drawable) &&
|
|
gimp_layer_is_alpha_locked (layer, NULL))
|
|
active[babl_format_get_n_components (format) - 1] = FALSE;
|
|
}
|
|
|
|
static GimpComponentMask
|
|
gimp_layer_get_active_mask (GimpDrawable *drawable)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (drawable);
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (drawable));
|
|
GimpComponentMask mask = gimp_image_get_active_mask (image);
|
|
|
|
if (gimp_drawable_has_alpha (drawable) &&
|
|
gimp_layer_is_alpha_locked (layer, NULL))
|
|
mask &= ~GIMP_COMPONENT_MASK_ALPHA;
|
|
|
|
return mask;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_set_buffer (GimpDrawable *drawable,
|
|
gboolean push_undo,
|
|
const gchar *undo_desc,
|
|
GeglBuffer *buffer,
|
|
const GeglRectangle *bounds)
|
|
{
|
|
GeglBuffer *old_buffer = gimp_drawable_get_buffer (drawable);
|
|
gint old_trc = -1;
|
|
|
|
if (old_buffer)
|
|
old_trc = gimp_drawable_get_trc (drawable);
|
|
|
|
GIMP_DRAWABLE_CLASS (parent_class)->set_buffer (drawable,
|
|
push_undo, undo_desc,
|
|
buffer, bounds);
|
|
|
|
if (gimp_filter_peek_node (GIMP_FILTER (drawable)))
|
|
{
|
|
if (gimp_drawable_get_trc (drawable) != old_trc)
|
|
gimp_layer_update_mode_node (GIMP_LAYER (drawable));
|
|
}
|
|
}
|
|
|
|
static GeglRectangle
|
|
gimp_layer_get_bounding_box (GimpDrawable *drawable)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (drawable);
|
|
GimpLayerMask *mask = gimp_layer_get_mask (layer);
|
|
GeglRectangle bounding_box;
|
|
|
|
if (mask && gimp_layer_get_show_mask (layer))
|
|
{
|
|
bounding_box = gimp_drawable_get_bounding_box (GIMP_DRAWABLE (mask));
|
|
}
|
|
else
|
|
{
|
|
bounding_box = GIMP_LAYER_GET_CLASS (layer)->get_bounding_box (layer);
|
|
|
|
if (mask && gimp_layer_get_apply_mask (layer))
|
|
{
|
|
GeglRectangle mask_bounding_box;
|
|
|
|
mask_bounding_box = gimp_drawable_get_bounding_box (
|
|
GIMP_DRAWABLE (mask));
|
|
|
|
gegl_rectangle_intersect (&bounding_box,
|
|
&bounding_box, &mask_bounding_box);
|
|
}
|
|
}
|
|
|
|
return bounding_box;
|
|
}
|
|
|
|
static GimpColorProfile *
|
|
gimp_layer_get_color_profile (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (managed));
|
|
|
|
return gimp_color_managed_get_color_profile (GIMP_COLOR_MANAGED (image));
|
|
}
|
|
|
|
static gdouble
|
|
gimp_layer_get_opacity_at (GimpPickable *pickable,
|
|
gint x,
|
|
gint y)
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (pickable);
|
|
gdouble value = GIMP_OPACITY_TRANSPARENT;
|
|
|
|
if (x >= 0 && x < gimp_item_get_width (GIMP_ITEM (layer)) &&
|
|
y >= 0 && y < gimp_item_get_height (GIMP_ITEM (layer)) &&
|
|
gimp_item_is_visible (GIMP_ITEM (layer)))
|
|
{
|
|
if (! gimp_drawable_has_alpha (GIMP_DRAWABLE (layer)))
|
|
{
|
|
value = GIMP_OPACITY_OPAQUE;
|
|
}
|
|
else
|
|
{
|
|
gegl_buffer_sample (gimp_drawable_get_buffer (GIMP_DRAWABLE (layer)),
|
|
x, y, NULL, &value, babl_format ("A double"),
|
|
GEGL_SAMPLER_NEAREST, GEGL_ABYSS_NONE);
|
|
}
|
|
|
|
if (gimp_layer_get_mask (layer) &&
|
|
gimp_layer_get_apply_mask (layer))
|
|
{
|
|
gdouble mask_value;
|
|
|
|
mask_value = gimp_pickable_get_opacity_at (GIMP_PICKABLE (layer->mask),
|
|
x, y);
|
|
|
|
value *= mask_value;
|
|
}
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
static gboolean
|
|
gimp_layer_real_is_alpha_locked (GimpLayer *layer,
|
|
GimpLayer **locked_layer)
|
|
{
|
|
GimpItem *parent = gimp_item_get_parent (GIMP_ITEM (layer));
|
|
|
|
if (layer->lock_alpha)
|
|
{
|
|
if (locked_layer)
|
|
*locked_layer = layer;
|
|
}
|
|
else if (parent &&
|
|
gimp_layer_is_alpha_locked (GIMP_LAYER (parent), locked_layer))
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
return layer->lock_alpha;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_translate (GimpLayer *layer,
|
|
gint offset_x,
|
|
gint offset_y)
|
|
{
|
|
/* update the old region */
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
|
|
/* invalidate the selection boundary because of a layer modification */
|
|
gimp_drawable_invalidate_boundary (GIMP_DRAWABLE (layer));
|
|
|
|
GIMP_ITEM_CLASS (parent_class)->translate (GIMP_ITEM (layer),
|
|
offset_x, offset_y,
|
|
FALSE);
|
|
|
|
/* update the new region */
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_scale (GimpLayer *layer,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint new_offset_x,
|
|
gint new_offset_y,
|
|
GimpInterpolationType interpolation_type,
|
|
GimpProgress *progress)
|
|
{
|
|
GIMP_ITEM_CLASS (parent_class)->scale (GIMP_ITEM (layer),
|
|
new_width, new_height,
|
|
new_offset_x, new_offset_y,
|
|
interpolation_type, progress);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_resize (GimpLayer *layer,
|
|
GimpContext *context,
|
|
GimpFillType fill_type,
|
|
gint new_width,
|
|
gint new_height,
|
|
gint offset_x,
|
|
gint offset_y)
|
|
{
|
|
if (fill_type == GIMP_FILL_TRANSPARENT &&
|
|
! gimp_drawable_has_alpha (GIMP_DRAWABLE (layer)))
|
|
{
|
|
fill_type = GIMP_FILL_BACKGROUND;
|
|
}
|
|
|
|
GIMP_ITEM_CLASS (parent_class)->resize (GIMP_ITEM (layer),
|
|
context, fill_type,
|
|
new_width, new_height,
|
|
offset_x, offset_y);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_flip (GimpLayer *layer,
|
|
GimpContext *context,
|
|
GimpOrientationType flip_type,
|
|
gdouble axis,
|
|
gboolean clip_result)
|
|
{
|
|
GIMP_ITEM_CLASS (parent_class)->flip (GIMP_ITEM (layer),
|
|
context, flip_type, axis, clip_result);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_rotate (GimpLayer *layer,
|
|
GimpContext *context,
|
|
GimpRotationType rotate_type,
|
|
gdouble center_x,
|
|
gdouble center_y,
|
|
gboolean clip_result)
|
|
{
|
|
GIMP_ITEM_CLASS (parent_class)->rotate (GIMP_ITEM (layer),
|
|
context, rotate_type,
|
|
center_x, center_y,
|
|
clip_result);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_transform (GimpLayer *layer,
|
|
GimpContext *context,
|
|
const GimpMatrix3 *matrix,
|
|
GimpTransformDirection direction,
|
|
GimpInterpolationType interpolation_type,
|
|
GimpTransformResize clip_result,
|
|
GimpProgress *progress,
|
|
gboolean push_undo)
|
|
{
|
|
if (! gimp_matrix3_is_simple (matrix) &&
|
|
! gimp_drawable_has_alpha (GIMP_DRAWABLE (layer)))
|
|
gimp_layer_add_alpha (GIMP_LAYER (layer));
|
|
|
|
GIMP_ITEM_CLASS (parent_class)->transform (GIMP_ITEM (layer),
|
|
context, matrix, direction,
|
|
interpolation_type,
|
|
clip_result,
|
|
progress, push_undo);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_convert_type (GimpLayer *layer,
|
|
GimpImage *dest_image,
|
|
const Babl *new_format,
|
|
GimpColorProfile *src_profile,
|
|
GimpColorProfile *dest_profile,
|
|
GeglDitherMethod layer_dither_type,
|
|
GeglDitherMethod mask_dither_type,
|
|
gboolean push_undo,
|
|
GimpProgress *progress)
|
|
{
|
|
GimpDrawable *drawable = GIMP_DRAWABLE (layer);
|
|
GeglBuffer *src_buffer;
|
|
GeglBuffer *dest_buffer;
|
|
|
|
if (layer_dither_type == GEGL_DITHER_NONE)
|
|
{
|
|
src_buffer = g_object_ref (gimp_drawable_get_buffer (drawable));
|
|
}
|
|
else
|
|
{
|
|
gint bits;
|
|
|
|
src_buffer =
|
|
gegl_buffer_new (GEGL_RECTANGLE (0, 0,
|
|
gimp_item_get_width (GIMP_ITEM (layer)),
|
|
gimp_item_get_height (GIMP_ITEM (layer))),
|
|
gimp_drawable_get_format (drawable));
|
|
|
|
bits = (babl_format_get_bytes_per_pixel (new_format) * 8 /
|
|
babl_format_get_n_components (new_format));
|
|
|
|
gimp_gegl_apply_dither (gimp_drawable_get_buffer (drawable),
|
|
NULL, NULL,
|
|
src_buffer, 1 << bits, layer_dither_type);
|
|
}
|
|
|
|
dest_buffer =
|
|
gegl_buffer_new (GEGL_RECTANGLE (0, 0,
|
|
gimp_item_get_width (GIMP_ITEM (layer)),
|
|
gimp_item_get_height (GIMP_ITEM (layer))),
|
|
new_format);
|
|
|
|
if (dest_profile)
|
|
{
|
|
if (! src_profile)
|
|
src_profile =
|
|
gimp_color_managed_get_color_profile (GIMP_COLOR_MANAGED (layer));
|
|
|
|
gimp_gegl_convert_color_profile (src_buffer, NULL, src_profile,
|
|
dest_buffer, NULL, dest_profile,
|
|
GIMP_COLOR_RENDERING_INTENT_PERCEPTUAL,
|
|
TRUE, progress);
|
|
}
|
|
else
|
|
{
|
|
gimp_gegl_buffer_copy (src_buffer, NULL, GEGL_ABYSS_NONE,
|
|
dest_buffer, NULL);
|
|
}
|
|
|
|
gimp_drawable_set_buffer (drawable, push_undo, NULL, dest_buffer);
|
|
|
|
g_object_unref (src_buffer);
|
|
g_object_unref (dest_buffer);
|
|
}
|
|
|
|
static GeglRectangle
|
|
gimp_layer_real_get_bounding_box (GimpLayer *layer)
|
|
{
|
|
return GIMP_DRAWABLE_CLASS (parent_class)->get_bounding_box (
|
|
GIMP_DRAWABLE (layer));
|
|
}
|
|
|
|
static void
|
|
gimp_layer_real_get_effective_mode (GimpLayer *layer,
|
|
GimpLayerMode *mode,
|
|
GimpLayerColorSpace *blend_space,
|
|
GimpLayerColorSpace *composite_space,
|
|
GimpLayerCompositeMode *composite_mode)
|
|
{
|
|
*mode = gimp_layer_get_mode (layer);
|
|
*blend_space = gimp_layer_get_real_blend_space (layer);
|
|
*composite_space = gimp_layer_get_real_composite_space (layer);
|
|
*composite_mode = gimp_layer_get_real_composite_mode (layer);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_layer_real_get_excludes_backdrop (GimpLayer *layer)
|
|
{
|
|
GimpLayerCompositeRegion included_region;
|
|
|
|
included_region = gimp_layer_mode_get_included_region (layer->mode,
|
|
layer->effective_composite_mode);
|
|
|
|
return ! (included_region & GIMP_LAYER_COMPOSITE_REGION_DESTINATION);
|
|
}
|
|
|
|
static void
|
|
gimp_layer_layer_mask_update (GimpDrawable *drawable,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height,
|
|
GimpLayer *layer)
|
|
{
|
|
if (gimp_layer_get_apply_mask (layer) ||
|
|
gimp_layer_get_show_mask (layer))
|
|
{
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer),
|
|
x, y, width, height);
|
|
}
|
|
}
|
|
|
|
|
|
/* public functions */
|
|
|
|
void
|
|
gimp_layer_fix_format_space (GimpLayer *layer,
|
|
gboolean copy_buffer,
|
|
gboolean push_undo)
|
|
{
|
|
GimpDrawable *drawable;
|
|
const Babl *format;
|
|
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (push_undo == FALSE || copy_buffer == TRUE);
|
|
|
|
drawable = GIMP_DRAWABLE (layer);
|
|
|
|
format = gimp_image_get_layer_format (gimp_item_get_image (GIMP_ITEM (layer)),
|
|
gimp_drawable_has_alpha (drawable));
|
|
|
|
if (format != gimp_drawable_get_format (drawable))
|
|
{
|
|
gimp_drawable_set_format (drawable, format, copy_buffer, push_undo);
|
|
}
|
|
}
|
|
|
|
GimpLayer *
|
|
gimp_layer_get_parent (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), NULL);
|
|
|
|
return GIMP_LAYER (gimp_viewable_get_parent (GIMP_VIEWABLE (layer)));
|
|
}
|
|
|
|
GimpLayerMask *
|
|
gimp_layer_get_mask (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), NULL);
|
|
|
|
return layer->mask;
|
|
}
|
|
|
|
GimpLayerMask *
|
|
gimp_layer_add_mask (GimpLayer *layer,
|
|
GimpLayerMask *mask,
|
|
gboolean edit_mask,
|
|
gboolean push_undo,
|
|
GError **error)
|
|
{
|
|
GimpImage *image;
|
|
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), NULL);
|
|
g_return_val_if_fail (GIMP_IS_LAYER_MASK (mask), NULL);
|
|
g_return_val_if_fail (gimp_item_get_image (GIMP_ITEM (layer)) ==
|
|
gimp_item_get_image (GIMP_ITEM (mask)), NULL);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, NULL);
|
|
|
|
if (! gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
push_undo = FALSE;
|
|
|
|
image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
if (layer->mask)
|
|
{
|
|
g_set_error_literal (error, GIMP_ERROR, GIMP_FAILED,
|
|
_("Unable to add a layer mask since "
|
|
"the layer already has one."));
|
|
return NULL;
|
|
}
|
|
|
|
if ((gimp_item_get_width (GIMP_ITEM (layer)) !=
|
|
gimp_item_get_width (GIMP_ITEM (mask))) ||
|
|
(gimp_item_get_height (GIMP_ITEM (layer)) !=
|
|
gimp_item_get_height (GIMP_ITEM (mask))))
|
|
{
|
|
g_set_error_literal (error, GIMP_ERROR, GIMP_FAILED,
|
|
_("Cannot add layer mask of different "
|
|
"dimensions than specified layer."));
|
|
return NULL;
|
|
}
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_layer_mask_add (image, C_("undo-type", "Add Layer Masks"),
|
|
layer, mask);
|
|
|
|
layer->mask = g_object_ref_sink (mask);
|
|
layer->apply_mask = TRUE;
|
|
layer->edit_mask = edit_mask;
|
|
layer->show_mask = FALSE;
|
|
|
|
gimp_layer_mask_set_layer (mask, layer);
|
|
|
|
if (gimp_filter_peek_node (GIMP_FILTER (layer)))
|
|
{
|
|
GeglNode *mode_node;
|
|
GeglNode *mask;
|
|
|
|
mode_node = gimp_drawable_get_mode_node (GIMP_DRAWABLE (layer));
|
|
|
|
mask = gimp_drawable_get_source_node (GIMP_DRAWABLE (layer->mask));
|
|
|
|
gegl_node_connect (mask, "output",
|
|
layer->mask_offset_node, "input");
|
|
|
|
if (layer->show_mask)
|
|
{
|
|
gegl_node_connect (layer->mask_offset_node, "output",
|
|
mode_node, "aux");
|
|
}
|
|
else
|
|
{
|
|
gegl_node_connect (layer->mask_offset_node, "output",
|
|
mode_node, "aux2");
|
|
}
|
|
|
|
gimp_layer_update_mode_node (layer);
|
|
}
|
|
|
|
gimp_drawable_update_bounding_box (GIMP_DRAWABLE (layer));
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
gimp_layer_update_excludes_backdrop (layer);
|
|
|
|
if (gimp_layer_get_apply_mask (layer) ||
|
|
gimp_layer_get_show_mask (layer))
|
|
{
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
}
|
|
|
|
g_signal_connect (mask, "update",
|
|
G_CALLBACK (gimp_layer_layer_mask_update),
|
|
layer);
|
|
|
|
g_signal_emit (layer, layer_signals[MASK_CHANGED], 0);
|
|
|
|
g_object_notify (G_OBJECT (layer), "mask");
|
|
|
|
/* if the mask came from the undo stack, reset its "removed" state */
|
|
if (gimp_item_is_removed (GIMP_ITEM (mask)))
|
|
gimp_item_unset_removed (GIMP_ITEM (mask));
|
|
|
|
return layer->mask;
|
|
}
|
|
|
|
GimpLayerMask *
|
|
gimp_layer_create_mask (GimpLayer *layer,
|
|
GimpAddMaskType add_mask_type,
|
|
GimpChannel *channel)
|
|
{
|
|
GimpDrawable *drawable;
|
|
GimpItem *item;
|
|
GimpLayerMask *mask;
|
|
GimpImage *image;
|
|
gchar *mask_name;
|
|
GeglColor *black = gegl_color_new ("black");
|
|
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), NULL);
|
|
g_return_val_if_fail (add_mask_type != GIMP_ADD_MASK_CHANNEL ||
|
|
GIMP_IS_CHANNEL (channel), NULL);
|
|
|
|
drawable = GIMP_DRAWABLE (layer);
|
|
item = GIMP_ITEM (layer);
|
|
image = gimp_item_get_image (item);
|
|
|
|
mask_name = g_strdup_printf (_("%s mask"),
|
|
gimp_object_get_name (layer));
|
|
|
|
mask = gimp_layer_mask_new (image,
|
|
gimp_item_get_width (item),
|
|
gimp_item_get_height (item),
|
|
mask_name, black);
|
|
|
|
g_free (mask_name);
|
|
g_object_unref (black);
|
|
|
|
switch (add_mask_type)
|
|
{
|
|
case GIMP_ADD_MASK_WHITE:
|
|
gimp_channel_all (GIMP_CHANNEL (mask), FALSE);
|
|
break;
|
|
|
|
case GIMP_ADD_MASK_BLACK:
|
|
gimp_channel_clear (GIMP_CHANNEL (mask), NULL, FALSE);
|
|
break;
|
|
|
|
case GIMP_ADD_MASK_ALPHA:
|
|
case GIMP_ADD_MASK_ALPHA_TRANSFER:
|
|
if (gimp_drawable_has_alpha (drawable))
|
|
{
|
|
GeglBuffer *dest_buffer;
|
|
const Babl *component_format;
|
|
|
|
dest_buffer = gimp_drawable_get_buffer (GIMP_DRAWABLE (mask));
|
|
|
|
component_format =
|
|
gimp_image_get_component_format (image, GIMP_CHANNEL_ALPHA);
|
|
|
|
gegl_buffer_set_format (dest_buffer, component_format);
|
|
gimp_gegl_buffer_copy (gimp_drawable_get_buffer (drawable), NULL,
|
|
GEGL_ABYSS_NONE,
|
|
dest_buffer, NULL);
|
|
gegl_buffer_set_format (dest_buffer, NULL);
|
|
|
|
if (add_mask_type == GIMP_ADD_MASK_ALPHA_TRANSFER)
|
|
{
|
|
gimp_drawable_push_undo (drawable,
|
|
C_("undo-type", "Transfer Alpha to Mask"),
|
|
NULL,
|
|
0, 0,
|
|
gimp_item_get_width (item),
|
|
gimp_item_get_height (item));
|
|
|
|
gimp_gegl_apply_set_alpha (gimp_drawable_get_buffer (drawable),
|
|
NULL, NULL,
|
|
gimp_drawable_get_buffer (drawable),
|
|
1.0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gimp_channel_all (GIMP_CHANNEL (mask), FALSE);
|
|
}
|
|
break;
|
|
|
|
case GIMP_ADD_MASK_SELECTION:
|
|
case GIMP_ADD_MASK_CHANNEL:
|
|
{
|
|
gboolean channel_empty;
|
|
gint offset_x, offset_y;
|
|
gint copy_x, copy_y;
|
|
gint copy_width, copy_height;
|
|
|
|
if (add_mask_type == GIMP_ADD_MASK_SELECTION)
|
|
channel = GIMP_CHANNEL (gimp_image_get_mask (image));
|
|
|
|
channel_empty = gimp_channel_is_empty (channel);
|
|
|
|
gimp_item_get_offset (item, &offset_x, &offset_y);
|
|
|
|
gimp_rectangle_intersect (0, 0,
|
|
gimp_image_get_width (image),
|
|
gimp_image_get_height (image),
|
|
offset_x, offset_y,
|
|
gimp_item_get_width (item),
|
|
gimp_item_get_height (item),
|
|
©_x, ©_y,
|
|
©_width, ©_height);
|
|
|
|
if (copy_width < gimp_item_get_width (item) ||
|
|
copy_height < gimp_item_get_height (item) ||
|
|
channel_empty)
|
|
gimp_channel_clear (GIMP_CHANNEL (mask), NULL, FALSE);
|
|
|
|
if ((copy_width || copy_height) && ! channel_empty)
|
|
{
|
|
GeglBuffer *src;
|
|
GeglBuffer *dest;
|
|
const Babl *format;
|
|
|
|
src = gimp_drawable_get_buffer (GIMP_DRAWABLE (channel));
|
|
dest = gimp_drawable_get_buffer (GIMP_DRAWABLE (mask));
|
|
|
|
format = gegl_buffer_get_format (src);
|
|
|
|
/* make sure no gamma conversion happens */
|
|
gegl_buffer_set_format (dest, format);
|
|
gimp_gegl_buffer_copy (src,
|
|
GEGL_RECTANGLE (copy_x, copy_y,
|
|
copy_width, copy_height),
|
|
GEGL_ABYSS_NONE,
|
|
dest,
|
|
GEGL_RECTANGLE (copy_x - offset_x,
|
|
copy_y - offset_y,
|
|
0, 0));
|
|
gegl_buffer_set_format (dest, NULL);
|
|
|
|
GIMP_CHANNEL (mask)->bounds_known = FALSE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case GIMP_ADD_MASK_COPY:
|
|
{
|
|
GeglBuffer *src_buffer;
|
|
GeglBuffer *dest_buffer;
|
|
|
|
if (! gimp_drawable_is_gray (drawable))
|
|
{
|
|
const Babl *copy_format =
|
|
gimp_image_get_format (image, GIMP_GRAY,
|
|
gimp_drawable_get_precision (drawable),
|
|
gimp_drawable_has_alpha (drawable),
|
|
NULL);
|
|
|
|
src_buffer =
|
|
gegl_buffer_new (GEGL_RECTANGLE (0, 0,
|
|
gimp_item_get_width (item),
|
|
gimp_item_get_height (item)),
|
|
copy_format);
|
|
|
|
gimp_gegl_buffer_copy (gimp_drawable_get_buffer (drawable), NULL,
|
|
GEGL_ABYSS_NONE,
|
|
src_buffer, NULL);
|
|
}
|
|
else
|
|
{
|
|
src_buffer = gimp_drawable_get_buffer (drawable);
|
|
g_object_ref (src_buffer);
|
|
}
|
|
|
|
dest_buffer = gimp_drawable_get_buffer (GIMP_DRAWABLE (mask));
|
|
|
|
if (gimp_drawable_has_alpha (drawable))
|
|
{
|
|
GeglColor *background = gegl_color_new ("transparent");
|
|
|
|
gimp_gegl_apply_flatten (src_buffer, NULL, NULL,
|
|
dest_buffer, background,
|
|
GIMP_LAYER_COLOR_SPACE_RGB_LINEAR);
|
|
|
|
g_object_unref (background);
|
|
}
|
|
else
|
|
{
|
|
gimp_gegl_buffer_copy (src_buffer, NULL, GEGL_ABYSS_NONE,
|
|
dest_buffer, NULL);
|
|
}
|
|
|
|
g_object_unref (src_buffer);
|
|
}
|
|
|
|
GIMP_CHANNEL (mask)->bounds_known = FALSE;
|
|
break;
|
|
}
|
|
|
|
return mask;
|
|
}
|
|
|
|
void
|
|
gimp_layer_apply_mask (GimpLayer *layer,
|
|
GimpMaskApplyMode mode,
|
|
gboolean push_undo)
|
|
{
|
|
GimpItem *item;
|
|
GimpImage *image;
|
|
GimpLayerMask *mask;
|
|
gboolean view_changed = FALSE;
|
|
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
mask = gimp_layer_get_mask (layer);
|
|
|
|
if (! mask)
|
|
return;
|
|
|
|
/* APPLY can not be done to group layers */
|
|
g_return_if_fail (! gimp_viewable_get_children (GIMP_VIEWABLE (layer)) ||
|
|
mode == GIMP_MASK_DISCARD);
|
|
|
|
/* APPLY can only be done to layers with an alpha channel */
|
|
g_return_if_fail (gimp_drawable_has_alpha (GIMP_DRAWABLE (layer)) ||
|
|
mode == GIMP_MASK_DISCARD || push_undo == TRUE);
|
|
|
|
item = GIMP_ITEM (layer);
|
|
image = gimp_item_get_image (item);
|
|
|
|
if (! image)
|
|
return;
|
|
|
|
if (push_undo)
|
|
{
|
|
gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_LAYER_APPLY_MASK,
|
|
(mode == GIMP_MASK_APPLY) ?
|
|
C_("undo-type", "Apply Layer Masks") :
|
|
C_("undo-type", "Delete Layer Masks"));
|
|
|
|
gimp_image_undo_push_layer_mask_show (image, NULL, layer);
|
|
gimp_image_undo_push_layer_mask_apply (image, NULL, layer);
|
|
gimp_image_undo_push_layer_mask_remove (image, NULL, layer, mask);
|
|
|
|
if (mode == GIMP_MASK_APPLY &&
|
|
! gimp_drawable_has_alpha (GIMP_DRAWABLE (layer)))
|
|
{
|
|
gimp_layer_add_alpha (layer);
|
|
}
|
|
}
|
|
|
|
/* check if applying the mask changes the projection */
|
|
if (gimp_layer_get_show_mask (layer) ||
|
|
(mode == GIMP_MASK_APPLY && ! gimp_layer_get_apply_mask (layer)) ||
|
|
(mode == GIMP_MASK_DISCARD && gimp_layer_get_apply_mask (layer)))
|
|
{
|
|
view_changed = TRUE;
|
|
}
|
|
|
|
if (mode == GIMP_MASK_APPLY)
|
|
{
|
|
GeglBuffer *mask_buffer;
|
|
GeglBuffer *dest_buffer;
|
|
|
|
if (push_undo)
|
|
gimp_drawable_push_undo (GIMP_DRAWABLE (layer), NULL,
|
|
NULL,
|
|
0, 0,
|
|
gimp_item_get_width (item),
|
|
gimp_item_get_height (item));
|
|
|
|
/* Combine the current layer's alpha channel and the mask */
|
|
mask_buffer = gimp_drawable_get_buffer (GIMP_DRAWABLE (mask));
|
|
dest_buffer = gimp_drawable_get_buffer (GIMP_DRAWABLE (layer));
|
|
|
|
gimp_gegl_apply_opacity (gimp_drawable_get_buffer (GIMP_DRAWABLE (layer)),
|
|
NULL, NULL, dest_buffer,
|
|
mask_buffer, 0, 0, 1.0);
|
|
}
|
|
|
|
g_signal_handlers_disconnect_by_func (mask,
|
|
gimp_layer_layer_mask_update,
|
|
layer);
|
|
|
|
gimp_item_removed (GIMP_ITEM (mask));
|
|
g_object_unref (mask);
|
|
layer->mask = NULL;
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_end (image);
|
|
|
|
if (gimp_filter_peek_node (GIMP_FILTER (layer)))
|
|
{
|
|
GeglNode *mode_node;
|
|
|
|
mode_node = gimp_drawable_get_mode_node (GIMP_DRAWABLE (layer));
|
|
|
|
if (layer->show_mask)
|
|
{
|
|
gegl_node_connect (layer->layer_offset_node, "output",
|
|
mode_node, "aux");
|
|
}
|
|
else
|
|
{
|
|
gegl_node_disconnect (mode_node, "aux2");
|
|
}
|
|
|
|
gimp_layer_update_mode_node (layer);
|
|
}
|
|
|
|
gimp_drawable_update_bounding_box (GIMP_DRAWABLE (layer));
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
gimp_layer_update_excludes_backdrop (layer);
|
|
|
|
/* If applying actually changed the view */
|
|
if (view_changed)
|
|
{
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
}
|
|
else
|
|
{
|
|
gimp_viewable_invalidate_preview (GIMP_VIEWABLE (layer));
|
|
}
|
|
|
|
g_signal_emit (layer, layer_signals[MASK_CHANGED], 0);
|
|
|
|
g_object_notify (G_OBJECT (layer), "mask");
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_apply_mask (GimpLayer *layer,
|
|
gboolean apply,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (layer->mask != NULL);
|
|
|
|
if (layer->apply_mask != apply)
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
if (push_undo && gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
gimp_image_undo_push_layer_mask_apply (image,
|
|
apply ?
|
|
C_("undo-type", "Enable Layer Masks") :
|
|
C_("undo-type", "Disable Layer Masks"),
|
|
layer);
|
|
|
|
layer->apply_mask = apply ? TRUE : FALSE;
|
|
|
|
if (gimp_filter_peek_node (GIMP_FILTER (layer)) &&
|
|
! gimp_layer_get_show_mask (layer))
|
|
{
|
|
GeglNode *mode_node;
|
|
|
|
mode_node = gimp_drawable_get_mode_node (GIMP_DRAWABLE (layer));
|
|
|
|
if (layer->apply_mask)
|
|
{
|
|
gegl_node_connect (layer->mask_offset_node, "output",
|
|
mode_node, "aux2");
|
|
}
|
|
else
|
|
{
|
|
gegl_node_disconnect (mode_node, "aux2");
|
|
}
|
|
}
|
|
|
|
gimp_drawable_update_bounding_box (GIMP_DRAWABLE (layer));
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
gimp_layer_update_excludes_backdrop (layer);
|
|
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
|
|
g_signal_emit (layer, layer_signals[APPLY_MASK_CHANGED], 0);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_get_apply_mask (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
g_return_val_if_fail (layer->mask, FALSE);
|
|
|
|
return layer->apply_mask;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_edit_mask (GimpLayer *layer,
|
|
gboolean edit)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (layer->mask != NULL);
|
|
|
|
if (layer->edit_mask != edit)
|
|
{
|
|
layer->edit_mask = edit ? TRUE : FALSE;
|
|
|
|
g_signal_emit (layer, layer_signals[EDIT_MASK_CHANGED], 0);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_get_edit_mask (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
g_return_val_if_fail (layer->mask, FALSE);
|
|
|
|
return layer->edit_mask;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_show_mask (GimpLayer *layer,
|
|
gboolean show,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (layer->mask != NULL);
|
|
|
|
if (layer->show_mask != show)
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_layer_mask_show (image,
|
|
C_("undo-type", "Show Layer Masks"),
|
|
layer);
|
|
|
|
layer->show_mask = show ? TRUE : FALSE;
|
|
|
|
if (gimp_filter_peek_node (GIMP_FILTER (layer)))
|
|
{
|
|
GeglNode *mode_node;
|
|
|
|
mode_node = gimp_drawable_get_mode_node (GIMP_DRAWABLE (layer));
|
|
|
|
if (layer->show_mask)
|
|
{
|
|
gegl_node_disconnect (mode_node, "aux2");
|
|
|
|
gegl_node_connect (layer->mask_offset_node, "output",
|
|
mode_node, "aux");
|
|
}
|
|
else
|
|
{
|
|
gegl_node_connect (layer->layer_offset_node, "output",
|
|
mode_node, "aux");
|
|
|
|
if (gimp_layer_get_apply_mask (layer))
|
|
{
|
|
gegl_node_connect (layer->mask_offset_node, "output",
|
|
mode_node, "aux2");
|
|
}
|
|
}
|
|
|
|
gimp_layer_update_mode_node (layer);
|
|
}
|
|
|
|
gimp_drawable_update_bounding_box (GIMP_DRAWABLE (layer));
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
gimp_layer_update_excludes_backdrop (layer);
|
|
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
|
|
g_signal_emit (layer, layer_signals[SHOW_MASK_CHANGED], 0);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_get_show_mask (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
g_return_val_if_fail (layer->mask, FALSE);
|
|
|
|
return layer->show_mask;
|
|
}
|
|
|
|
void
|
|
gimp_layer_add_alpha (GimpLayer *layer)
|
|
{
|
|
GimpItem *item;
|
|
GimpDrawable *drawable;
|
|
GeglBuffer *new_buffer;
|
|
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
if (gimp_drawable_has_alpha (GIMP_DRAWABLE (layer)))
|
|
return;
|
|
|
|
item = GIMP_ITEM (layer);
|
|
drawable = GIMP_DRAWABLE (layer);
|
|
|
|
new_buffer = gegl_buffer_new (GEGL_RECTANGLE (0, 0,
|
|
gimp_item_get_width (item),
|
|
gimp_item_get_height (item)),
|
|
gimp_drawable_get_format_with_alpha (drawable));
|
|
|
|
gimp_gegl_buffer_copy (
|
|
gimp_drawable_get_buffer (drawable), NULL, GEGL_ABYSS_NONE,
|
|
new_buffer, NULL);
|
|
|
|
gimp_drawable_set_buffer (GIMP_DRAWABLE (layer),
|
|
gimp_item_is_attached (GIMP_ITEM (layer)),
|
|
C_("undo-type", "Add Alpha Channel"),
|
|
new_buffer);
|
|
g_object_unref (new_buffer);
|
|
}
|
|
|
|
void
|
|
gimp_layer_remove_alpha (GimpLayer *layer,
|
|
GimpContext *context)
|
|
{
|
|
GeglBuffer *new_buffer;
|
|
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (GIMP_IS_CONTEXT (context));
|
|
|
|
if (! gimp_drawable_has_alpha (GIMP_DRAWABLE (layer)))
|
|
return;
|
|
|
|
new_buffer =
|
|
gegl_buffer_new (GEGL_RECTANGLE (0, 0,
|
|
gimp_item_get_width (GIMP_ITEM (layer)),
|
|
gimp_item_get_height (GIMP_ITEM (layer))),
|
|
gimp_drawable_get_format_without_alpha (GIMP_DRAWABLE (layer)));
|
|
|
|
gimp_gegl_apply_flatten (gimp_drawable_get_buffer (GIMP_DRAWABLE (layer)),
|
|
NULL, NULL, new_buffer,
|
|
gimp_context_get_background (context),
|
|
gimp_layer_get_real_composite_space (layer));
|
|
|
|
gimp_drawable_set_buffer (GIMP_DRAWABLE (layer),
|
|
gimp_item_is_attached (GIMP_ITEM (layer)),
|
|
C_("undo-type", "Remove Alpha Channel"),
|
|
new_buffer);
|
|
g_object_unref (new_buffer);
|
|
}
|
|
|
|
void
|
|
gimp_layer_resize_to_image (GimpLayer *layer,
|
|
GimpContext *context,
|
|
GimpFillType fill_type)
|
|
{
|
|
GimpImage *image;
|
|
gint offset_x;
|
|
gint offset_y;
|
|
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (gimp_item_is_attached (GIMP_ITEM (layer)));
|
|
g_return_if_fail (GIMP_IS_CONTEXT (context));
|
|
|
|
image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_ITEM_RESIZE,
|
|
C_("undo-type", "Layer to Image Size"));
|
|
|
|
gimp_item_get_offset (GIMP_ITEM (layer), &offset_x, &offset_y);
|
|
gimp_item_resize (GIMP_ITEM (layer), context, fill_type,
|
|
gimp_image_get_width (image),
|
|
gimp_image_get_height (image),
|
|
offset_x, offset_y);
|
|
|
|
gimp_image_undo_group_end (image);
|
|
}
|
|
|
|
/**********************/
|
|
/* access functions */
|
|
/**********************/
|
|
|
|
GimpDrawable *
|
|
gimp_layer_get_floating_sel_drawable (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), NULL);
|
|
|
|
return layer->fs.drawable;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_floating_sel_drawable (GimpLayer *layer,
|
|
GimpDrawable *drawable)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (drawable == NULL || GIMP_IS_DRAWABLE (drawable));
|
|
|
|
if (g_set_object (&layer->fs.drawable, drawable))
|
|
{
|
|
if (layer->fs.segs)
|
|
{
|
|
g_clear_pointer (&layer->fs.segs, g_free);
|
|
layer->fs.num_segs = 0;
|
|
}
|
|
|
|
g_object_notify (G_OBJECT (layer), "floating-selection");
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_is_floating_sel (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
|
|
return (gimp_layer_get_floating_sel_drawable (layer) != NULL);
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_opacity (GimpLayer *layer,
|
|
gdouble opacity,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
opacity = CLAMP (opacity, GIMP_OPACITY_TRANSPARENT, GIMP_OPACITY_OPAQUE);
|
|
|
|
if (layer->opacity != opacity)
|
|
{
|
|
if (push_undo && gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_image_undo_push_layer_opacity (image, NULL, layer);
|
|
}
|
|
|
|
layer->opacity = opacity;
|
|
|
|
g_signal_emit (layer, layer_signals[OPACITY_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "opacity");
|
|
|
|
if (gimp_filter_peek_node (GIMP_FILTER (layer)))
|
|
gimp_layer_update_mode_node (layer);
|
|
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
}
|
|
}
|
|
|
|
gdouble
|
|
gimp_layer_get_opacity (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_OPACITY_OPAQUE);
|
|
|
|
return layer->opacity;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_mode (GimpLayer *layer,
|
|
GimpLayerMode mode,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
if (gimp_viewable_get_children (GIMP_VIEWABLE (layer)) == NULL)
|
|
{
|
|
g_return_if_fail (gimp_layer_mode_get_context (mode) &
|
|
GIMP_LAYER_MODE_CONTEXT_LAYER);
|
|
}
|
|
else
|
|
{
|
|
g_return_if_fail (gimp_layer_mode_get_context (mode) &
|
|
GIMP_LAYER_MODE_CONTEXT_GROUP);
|
|
}
|
|
|
|
if (layer->mode != mode)
|
|
{
|
|
if (gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_image_unset_default_new_layer_mode (image);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_layer_mode (image, NULL, layer);
|
|
}
|
|
|
|
g_object_freeze_notify (G_OBJECT (layer));
|
|
|
|
layer->mode = mode;
|
|
|
|
g_signal_emit (layer, layer_signals[MODE_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "mode");
|
|
|
|
/* when changing modes, we always switch to AUTO blend and
|
|
* composite in order to avoid confusion
|
|
*/
|
|
if (layer->blend_space != GIMP_LAYER_COLOR_SPACE_AUTO)
|
|
{
|
|
layer->blend_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
|
|
g_signal_emit (layer, layer_signals[BLEND_SPACE_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "blend-space");
|
|
}
|
|
|
|
if (layer->composite_space != GIMP_LAYER_COLOR_SPACE_AUTO)
|
|
{
|
|
layer->composite_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
|
|
g_signal_emit (layer, layer_signals[COMPOSITE_SPACE_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "composite-space");
|
|
}
|
|
|
|
if (layer->composite_mode != GIMP_LAYER_COMPOSITE_AUTO)
|
|
{
|
|
layer->composite_mode = GIMP_LAYER_COMPOSITE_AUTO;
|
|
|
|
g_signal_emit (layer, layer_signals[COMPOSITE_MODE_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "composite-mode");
|
|
}
|
|
|
|
g_object_thaw_notify (G_OBJECT (layer));
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
gimp_layer_update_excludes_backdrop (layer);
|
|
}
|
|
}
|
|
|
|
GimpLayerMode
|
|
gimp_layer_get_mode (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_LAYER_MODE_NORMAL);
|
|
|
|
return layer->mode;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_blend_space (GimpLayer *layer,
|
|
GimpLayerColorSpace blend_space,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
if (! gimp_layer_mode_is_blend_space_mutable (layer->mode))
|
|
return;
|
|
|
|
if (layer->blend_space != blend_space)
|
|
{
|
|
if (push_undo && gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_image_undo_push_layer_mode (image, _("Set layer's blend space"), layer);
|
|
}
|
|
|
|
layer->blend_space = blend_space;
|
|
|
|
g_signal_emit (layer, layer_signals[BLEND_SPACE_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "blend-space");
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
}
|
|
}
|
|
|
|
GimpLayerColorSpace
|
|
gimp_layer_get_blend_space (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_LAYER_COLOR_SPACE_AUTO);
|
|
|
|
return layer->blend_space;
|
|
}
|
|
|
|
GimpLayerColorSpace
|
|
gimp_layer_get_real_blend_space (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_LAYER_COLOR_SPACE_RGB_LINEAR);
|
|
|
|
if (layer->blend_space == GIMP_LAYER_COLOR_SPACE_AUTO)
|
|
return gimp_layer_mode_get_blend_space (layer->mode);
|
|
else
|
|
return layer->blend_space;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_composite_space (GimpLayer *layer,
|
|
GimpLayerColorSpace composite_space,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
if (! gimp_layer_mode_is_composite_space_mutable (layer->mode))
|
|
return;
|
|
|
|
if (layer->composite_space != composite_space)
|
|
{
|
|
if (push_undo && gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_image_undo_push_layer_mode (image, _("Set layer's composite space"), layer);
|
|
}
|
|
|
|
layer->composite_space = composite_space;
|
|
|
|
g_signal_emit (layer, layer_signals[COMPOSITE_SPACE_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "composite-space");
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
}
|
|
}
|
|
|
|
GimpLayerColorSpace
|
|
gimp_layer_get_composite_space (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_LAYER_COLOR_SPACE_AUTO);
|
|
|
|
return layer->composite_space;
|
|
}
|
|
|
|
GimpLayerColorSpace
|
|
gimp_layer_get_real_composite_space (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_LAYER_COLOR_SPACE_RGB_LINEAR);
|
|
|
|
if (layer->composite_space == GIMP_LAYER_COLOR_SPACE_AUTO)
|
|
return gimp_layer_mode_get_composite_space (layer->mode);
|
|
else
|
|
return layer->composite_space;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_composite_mode (GimpLayer *layer,
|
|
GimpLayerCompositeMode composite_mode,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
if (! gimp_layer_mode_is_composite_mode_mutable (layer->mode))
|
|
return;
|
|
|
|
if (layer->composite_mode != composite_mode)
|
|
{
|
|
if (push_undo && gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_image_undo_push_layer_mode (image, _("Set layer's composite mode"), layer);
|
|
}
|
|
|
|
layer->composite_mode = composite_mode;
|
|
|
|
g_signal_emit (layer, layer_signals[COMPOSITE_MODE_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "composite-mode");
|
|
|
|
gimp_layer_update_effective_mode (layer);
|
|
gimp_layer_update_excludes_backdrop (layer);
|
|
}
|
|
}
|
|
|
|
GimpLayerCompositeMode
|
|
gimp_layer_get_composite_mode (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_LAYER_COMPOSITE_AUTO);
|
|
|
|
return layer->composite_mode;
|
|
}
|
|
|
|
GimpLayerCompositeMode
|
|
gimp_layer_get_real_composite_mode (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), GIMP_LAYER_COMPOSITE_UNION);
|
|
|
|
if (layer->composite_mode == GIMP_LAYER_COMPOSITE_AUTO)
|
|
return gimp_layer_mode_get_composite_mode (layer->mode);
|
|
else
|
|
return layer->composite_mode;
|
|
}
|
|
|
|
void
|
|
gimp_layer_get_effective_mode (GimpLayer *layer,
|
|
GimpLayerMode *mode,
|
|
GimpLayerColorSpace *blend_space,
|
|
GimpLayerColorSpace *composite_space,
|
|
GimpLayerCompositeMode *composite_mode)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
if (mode) *mode = layer->effective_mode;
|
|
if (blend_space) *blend_space = layer->effective_blend_space;
|
|
if (composite_space) *composite_space = layer->effective_composite_space;
|
|
if (composite_mode) *composite_mode = layer->effective_composite_mode;
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_get_excludes_backdrop (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
|
|
return layer->excludes_backdrop;
|
|
}
|
|
|
|
void
|
|
gimp_layer_set_lock_alpha (GimpLayer *layer,
|
|
gboolean lock_alpha,
|
|
gboolean push_undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (gimp_layer_can_lock_alpha (layer));
|
|
|
|
lock_alpha = lock_alpha ? TRUE : FALSE;
|
|
|
|
if (layer->lock_alpha != lock_alpha)
|
|
{
|
|
if (push_undo && gimp_item_is_attached (GIMP_ITEM (layer)))
|
|
{
|
|
GimpImage *image = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_image_undo_push_layer_lock_alpha (image, NULL, layer);
|
|
}
|
|
|
|
layer->lock_alpha = lock_alpha;
|
|
|
|
g_signal_emit (layer, layer_signals[LOCK_ALPHA_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "lock-alpha");
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_get_lock_alpha (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
|
|
return layer->lock_alpha;
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_can_lock_alpha (GimpLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gimp_layer_is_alpha_locked (GimpLayer *layer,
|
|
GimpLayer **locked_layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), FALSE);
|
|
|
|
return GIMP_LAYER_GET_CLASS (layer)->is_alpha_locked (layer, locked_layer);
|
|
}
|
|
|
|
/* protected functions */
|
|
|
|
void
|
|
gimp_layer_update_effective_mode (GimpLayer *layer)
|
|
{
|
|
GimpLayerMode mode;
|
|
GimpLayerColorSpace blend_space;
|
|
GimpLayerColorSpace composite_space;
|
|
GimpLayerCompositeMode composite_mode;
|
|
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
if (layer->mask && layer->show_mask)
|
|
{
|
|
mode = GIMP_LAYER_MODE_NORMAL;
|
|
blend_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
composite_space = GIMP_LAYER_COLOR_SPACE_AUTO;
|
|
composite_mode = GIMP_LAYER_COMPOSITE_AUTO;
|
|
|
|
/* This makes sure that masks of LEGACY-mode layers are
|
|
* composited in PERCEPTUAL space, and non-LEGACY layers in
|
|
* LINEAR space, or whatever composite space was chosen in the
|
|
* layer attributes dialog
|
|
*/
|
|
composite_space = gimp_layer_get_real_composite_space (layer);
|
|
}
|
|
else
|
|
{
|
|
GIMP_LAYER_GET_CLASS (layer)->get_effective_mode (layer,
|
|
&mode,
|
|
&blend_space,
|
|
&composite_space,
|
|
&composite_mode);
|
|
}
|
|
|
|
if (mode != layer->effective_mode ||
|
|
blend_space != layer->effective_blend_space ||
|
|
composite_space != layer->effective_composite_space ||
|
|
composite_mode != layer->effective_composite_mode)
|
|
{
|
|
layer->effective_mode = mode;
|
|
layer->effective_blend_space = blend_space;
|
|
layer->effective_composite_space = composite_space;
|
|
layer->effective_composite_mode = composite_mode;
|
|
|
|
g_signal_emit (layer, layer_signals[EFFECTIVE_MODE_CHANGED], 0);
|
|
|
|
if (gimp_filter_peek_node (GIMP_FILTER (layer)))
|
|
gimp_layer_update_mode_node (layer);
|
|
|
|
gimp_drawable_update (GIMP_DRAWABLE (layer), 0, 0, -1, -1);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_layer_update_excludes_backdrop (GimpLayer *layer)
|
|
{
|
|
gboolean excludes_backdrop;
|
|
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
excludes_backdrop =
|
|
GIMP_LAYER_GET_CLASS (layer)->get_excludes_backdrop (layer);
|
|
|
|
if (excludes_backdrop != layer->excludes_backdrop)
|
|
{
|
|
layer->excludes_backdrop = excludes_backdrop;
|
|
|
|
g_signal_emit (layer, layer_signals[EXCLUDES_BACKDROP_CHANGED], 0);
|
|
g_object_notify (G_OBJECT (layer), "excludes-backdrop");
|
|
}
|
|
}
|