mirror of
https://gitlab.gnome.org/GNOME/gimp.git
synced 2025-07-03 17:33:25 +00:00

gimp_display_shell_render() writes to a GeglBuffer backed by allocated memory
(shell->profile_data). Unfortunately while converting prevision in
gimp_image_convert_precision(), we change the "precision" property (hence the
source format) first, hence end up trying to write data in a too small buffer.
This crash was hard to find as it was not showing up on my machine (though it
did produce rendering artifacts!), unless I built both GIMP and babl with
`b_sanitize=address`.
Note that an alternate fix was to make sure that the profile_data buffer is big
enough (by calling gimp_display_shell_profile_update() before rendering), but
anyway the image is in an inconsistent state while conversion is in progress:
whereas the `src_format` is the new one, the `src_profile` is still the old one
(and cannot be changed before we finish converting).
Moreover the render happen regularly on progress signals, once after each
converted drawable. So each of these rendering step happens in an inconsistent
state, with the wrong profile set, some of the drawables converted and others
not yet.
We could still render properly if each drawable's buffer used space-aware format
(thus allowing different drawables to use different profiles/spaces), but it
feels over-engineering the problem. It might be much better to ignore rendering
steps while converting the image precision. Moreover it would obviously make a
faster conversion.
See discussions in #9136 for this crash, which didn't have dedicated report
AFAIK.
(cherry picked from commit de25be9210
)
Note: on the `master` branch, even with sanitized code, I don't get the crash.
Yet this change seems relevant enough that I'm adding it.
6196 lines
187 KiB
C
6196 lines
187 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 <string.h>
|
|
#include <time.h>
|
|
|
|
#include <cairo.h>
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <gegl.h>
|
|
#include <gexiv2/gexiv2.h>
|
|
|
|
#include "libgimpcolor/gimpcolor.h"
|
|
#include "libgimpmath/gimpmath.h"
|
|
#include "libgimpbase/gimpbase.h"
|
|
#include "libgimpconfig/gimpconfig.h"
|
|
|
|
#include "core-types.h"
|
|
|
|
#include "config/gimpcoreconfig.h"
|
|
|
|
#include "operations/layer-modes/gimp-layer-modes.h"
|
|
|
|
#include "gegl/gimp-babl.h"
|
|
#include "gegl/gimp-gegl-loops.h"
|
|
|
|
#include "gimp.h"
|
|
#include "gimp-memsize.h"
|
|
#include "gimp-parasites.h"
|
|
#include "gimp-utils.h"
|
|
#include "gimpcontext.h"
|
|
#include "gimpdrawable-floating-selection.h"
|
|
#include "gimpdrawablestack.h"
|
|
#include "gimpgrid.h"
|
|
#include "gimperror.h"
|
|
#include "gimpguide.h"
|
|
#include "gimpidtable.h"
|
|
#include "gimpimage.h"
|
|
#include "gimpimage-color-profile.h"
|
|
#include "gimpimage-colormap.h"
|
|
#include "gimpimage-guides.h"
|
|
#include "gimpimage-item-list.h"
|
|
#include "gimpimage-metadata.h"
|
|
#include "gimpimage-sample-points.h"
|
|
#include "gimpimage-preview.h"
|
|
#include "gimpimage-private.h"
|
|
#include "gimpimage-quick-mask.h"
|
|
#include "gimpimage-symmetry.h"
|
|
#include "gimpimage-undo.h"
|
|
#include "gimpimage-undo-push.h"
|
|
#include "gimpitemlist.h"
|
|
#include "gimpitemtree.h"
|
|
#include "gimplayer.h"
|
|
#include "gimplayer-floating-selection.h"
|
|
#include "gimplayermask.h"
|
|
#include "gimplayerstack.h"
|
|
#include "gimpmarshal.h"
|
|
#include "gimppalette.h"
|
|
#include "gimpparasitelist.h"
|
|
#include "gimppickable.h"
|
|
#include "gimpprojectable.h"
|
|
#include "gimpprojection.h"
|
|
#include "gimpsamplepoint.h"
|
|
#include "gimpselection.h"
|
|
#include "gimpsymmetry.h"
|
|
#include "gimptempbuf.h"
|
|
#include "gimptemplate.h"
|
|
#include "gimpundostack.h"
|
|
|
|
#include "vectors/gimpvectors.h"
|
|
|
|
#include "gimp-log.h"
|
|
#include "gimp-intl.h"
|
|
|
|
|
|
#ifdef DEBUG
|
|
#define TRC(x) g_printerr x
|
|
#else
|
|
#define TRC(x)
|
|
#endif
|
|
|
|
|
|
enum
|
|
{
|
|
MODE_CHANGED,
|
|
PRECISION_CHANGED,
|
|
ALPHA_CHANGED,
|
|
FLOATING_SELECTION_CHANGED,
|
|
SELECTED_CHANNELS_CHANGED,
|
|
SELECTED_VECTORS_CHANGED,
|
|
SELECTED_LAYERS_CHANGED,
|
|
COMPONENT_VISIBILITY_CHANGED,
|
|
COMPONENT_ACTIVE_CHANGED,
|
|
MASK_CHANGED,
|
|
RESOLUTION_CHANGED,
|
|
SIZE_CHANGED_DETAILED,
|
|
UNIT_CHANGED,
|
|
QUICK_MASK_CHANGED,
|
|
SELECTION_INVALIDATE,
|
|
CLEAN,
|
|
DIRTY,
|
|
SAVING,
|
|
SAVED,
|
|
EXPORTED,
|
|
GUIDE_ADDED,
|
|
GUIDE_REMOVED,
|
|
GUIDE_MOVED,
|
|
SAMPLE_POINT_ADDED,
|
|
SAMPLE_POINT_REMOVED,
|
|
SAMPLE_POINT_MOVED,
|
|
PARASITE_ATTACHED,
|
|
PARASITE_DETACHED,
|
|
COLORMAP_CHANGED,
|
|
UNDO_EVENT,
|
|
LAYER_SETS_CHANGED,
|
|
CHANNEL_SETS_CHANGED,
|
|
VECTORS_SETS_CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_GIMP,
|
|
PROP_ID,
|
|
PROP_WIDTH,
|
|
PROP_HEIGHT,
|
|
PROP_BASE_TYPE,
|
|
PROP_PRECISION,
|
|
PROP_METADATA,
|
|
PROP_BUFFER,
|
|
PROP_SYMMETRY,
|
|
PROP_CONVERTING,
|
|
};
|
|
|
|
|
|
/* local function prototypes */
|
|
|
|
static void gimp_color_managed_iface_init (GimpColorManagedInterface *iface);
|
|
static void gimp_projectable_iface_init (GimpProjectableInterface *iface);
|
|
static void gimp_pickable_iface_init (GimpPickableInterface *iface);
|
|
|
|
static void gimp_image_constructed (GObject *object);
|
|
static void gimp_image_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_image_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_image_dispose (GObject *object);
|
|
static void gimp_image_finalize (GObject *object);
|
|
|
|
static void gimp_image_name_changed (GimpObject *object);
|
|
static gint64 gimp_image_get_memsize (GimpObject *object,
|
|
gint64 *gui_size);
|
|
|
|
static gboolean gimp_image_get_size (GimpViewable *viewable,
|
|
gint *width,
|
|
gint *height);
|
|
static void gimp_image_size_changed (GimpViewable *viewable);
|
|
static gchar * gimp_image_get_description (GimpViewable *viewable,
|
|
gchar **tooltip);
|
|
|
|
static void gimp_image_real_mode_changed (GimpImage *image);
|
|
static void gimp_image_real_precision_changed(GimpImage *image);
|
|
static void gimp_image_real_resolution_changed(GimpImage *image);
|
|
static void gimp_image_real_size_changed_detailed
|
|
(GimpImage *image,
|
|
gint previous_origin_x,
|
|
gint previous_origin_y,
|
|
gint previous_width,
|
|
gint previous_height);
|
|
static void gimp_image_real_unit_changed (GimpImage *image);
|
|
static void gimp_image_real_colormap_changed (GimpImage *image,
|
|
gint color_index);
|
|
|
|
static const guint8 *
|
|
gimp_image_color_managed_get_icc_profile (GimpColorManaged *managed,
|
|
gsize *len);
|
|
static GimpColorProfile *
|
|
gimp_image_color_managed_get_color_profile (GimpColorManaged *managed);
|
|
static void
|
|
gimp_image_color_managed_profile_changed (GimpColorManaged *managed);
|
|
|
|
static GimpColorProfile *
|
|
gimp_image_color_managed_get_simulation_profile (GimpColorManaged *managed);
|
|
static void
|
|
gimp_image_color_managed_simulation_profile_changed (GimpColorManaged *managed);
|
|
|
|
static GimpColorRenderingIntent
|
|
gimp_image_color_managed_get_simulation_intent (GimpColorManaged *managed);
|
|
static void
|
|
gimp_image_color_managed_simulation_intent_changed (GimpColorManaged *managed);
|
|
|
|
static gboolean
|
|
gimp_image_color_managed_get_simulation_bpc (GimpColorManaged *managed);
|
|
static void
|
|
gimp_image_color_managed_simulation_bpc_changed (GimpColorManaged *managed);
|
|
|
|
static void gimp_image_projectable_flush (GimpProjectable *projectable,
|
|
gboolean invalidate_preview);
|
|
static GeglRectangle gimp_image_get_bounding_box (GimpProjectable *projectable);
|
|
static GeglNode * gimp_image_get_graph (GimpProjectable *projectable);
|
|
static GimpImage * gimp_image_get_image (GimpProjectable *projectable);
|
|
static const Babl * gimp_image_get_proj_format (GimpProjectable *projectable);
|
|
|
|
static void gimp_image_pickable_flush (GimpPickable *pickable);
|
|
static GeglBuffer * gimp_image_get_buffer (GimpPickable *pickable);
|
|
static gboolean gimp_image_get_pixel_at (GimpPickable *pickable,
|
|
gint x,
|
|
gint y,
|
|
const Babl *format,
|
|
gpointer pixel);
|
|
static gdouble gimp_image_get_opacity_at (GimpPickable *pickable,
|
|
gint x,
|
|
gint y);
|
|
static void gimp_image_get_pixel_average (GimpPickable *pickable,
|
|
const GeglRectangle *rect,
|
|
const Babl *format,
|
|
gpointer pixel);
|
|
static void gimp_image_pixel_to_srgb (GimpPickable *pickable,
|
|
const Babl *format,
|
|
gpointer pixel,
|
|
GimpRGB *color);
|
|
static void gimp_image_srgb_to_pixel (GimpPickable *pickable,
|
|
const GimpRGB *color,
|
|
const Babl *format,
|
|
gpointer pixel);
|
|
|
|
static void gimp_image_projection_buffer_notify
|
|
(GimpProjection *projection,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image);
|
|
static void gimp_image_mask_update (GimpDrawable *drawable,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height,
|
|
GimpImage *image);
|
|
static void gimp_image_layers_changed (GimpContainer *container,
|
|
GimpChannel *channel,
|
|
GimpImage *image);
|
|
static void gimp_image_layer_offset_changed (GimpDrawable *drawable,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image);
|
|
static void gimp_image_layer_bounding_box_changed
|
|
(GimpDrawable *drawable,
|
|
GimpImage *image);
|
|
static void gimp_image_layer_alpha_changed (GimpDrawable *drawable,
|
|
GimpImage *image);
|
|
static void gimp_image_channel_add (GimpContainer *container,
|
|
GimpChannel *channel,
|
|
GimpImage *image);
|
|
static void gimp_image_channel_remove (GimpContainer *container,
|
|
GimpChannel *channel,
|
|
GimpImage *image);
|
|
static void gimp_image_channel_name_changed (GimpChannel *channel,
|
|
GimpImage *image);
|
|
static void gimp_image_channel_color_changed (GimpChannel *channel,
|
|
GimpImage *image);
|
|
|
|
static void gimp_image_selected_layers_notify (GimpItemTree *tree,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image);
|
|
static void gimp_image_selected_channels_notify (GimpItemTree *tree,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image);
|
|
static void gimp_image_selected_vectors_notify (GimpItemTree *tree,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image);
|
|
|
|
static void gimp_image_freeze_bounding_box (GimpImage *image);
|
|
static void gimp_image_thaw_bounding_box (GimpImage *image);
|
|
static void gimp_image_update_bounding_box (GimpImage *image);
|
|
|
|
static gint gimp_image_layer_stack_cmp (GList *layers1,
|
|
GList *layers2);
|
|
static void gimp_image_rec_remove_layer_stack_dups (GimpImage *image,
|
|
GSList *start);
|
|
static void gimp_image_clean_layer_stack (GimpImage *image);
|
|
static void gimp_image_remove_from_layer_stack (GimpImage *image,
|
|
GimpLayer *layer);
|
|
static gint gimp_image_selected_is_descendant (GimpViewable *selected,
|
|
GimpViewable *viewable);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GimpImage, gimp_image, GIMP_TYPE_VIEWABLE,
|
|
G_ADD_PRIVATE (GimpImage)
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_COLOR_MANAGED,
|
|
gimp_color_managed_iface_init)
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_PROJECTABLE,
|
|
gimp_projectable_iface_init)
|
|
G_IMPLEMENT_INTERFACE (GIMP_TYPE_PICKABLE,
|
|
gimp_pickable_iface_init))
|
|
|
|
#define parent_class gimp_image_parent_class
|
|
|
|
static guint gimp_image_signals[LAST_SIGNAL] = { 0 };
|
|
|
|
|
|
static void
|
|
gimp_image_class_init (GimpImageClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GimpObjectClass *gimp_object_class = GIMP_OBJECT_CLASS (klass);
|
|
GimpViewableClass *viewable_class = GIMP_VIEWABLE_CLASS (klass);
|
|
|
|
gimp_image_signals[MODE_CHANGED] =
|
|
g_signal_new ("mode-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, mode_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[PRECISION_CHANGED] =
|
|
g_signal_new ("precision-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, precision_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[ALPHA_CHANGED] =
|
|
g_signal_new ("alpha-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, alpha_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[FLOATING_SELECTION_CHANGED] =
|
|
g_signal_new ("floating-selection-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, floating_selection_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[SELECTED_LAYERS_CHANGED] =
|
|
g_signal_new ("selected-layers-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, selected_layers_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[SELECTED_CHANNELS_CHANGED] =
|
|
g_signal_new ("selected-channels-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, selected_channels_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[SELECTED_VECTORS_CHANGED] =
|
|
g_signal_new ("selected-vectors-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, selected_vectors_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[COMPONENT_VISIBILITY_CHANGED] =
|
|
g_signal_new ("component-visibility-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, component_visibility_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_CHANNEL_TYPE);
|
|
|
|
gimp_image_signals[COMPONENT_ACTIVE_CHANGED] =
|
|
g_signal_new ("component-active-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, component_active_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_CHANNEL_TYPE);
|
|
|
|
gimp_image_signals[MASK_CHANGED] =
|
|
g_signal_new ("mask-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, mask_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[RESOLUTION_CHANGED] =
|
|
g_signal_new ("resolution-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, resolution_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[SIZE_CHANGED_DETAILED] =
|
|
g_signal_new ("size-changed-detailed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, size_changed_detailed),
|
|
NULL, NULL,
|
|
gimp_marshal_VOID__INT_INT_INT_INT,
|
|
G_TYPE_NONE, 4,
|
|
G_TYPE_INT,
|
|
G_TYPE_INT,
|
|
G_TYPE_INT,
|
|
G_TYPE_INT);
|
|
|
|
gimp_image_signals[UNIT_CHANGED] =
|
|
g_signal_new ("unit-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, unit_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[QUICK_MASK_CHANGED] =
|
|
g_signal_new ("quick-mask-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, quick_mask_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[SELECTION_INVALIDATE] =
|
|
g_signal_new ("selection-invalidate",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, selection_invalidate),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[CLEAN] =
|
|
g_signal_new ("clean",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, clean),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_DIRTY_MASK);
|
|
|
|
gimp_image_signals[DIRTY] =
|
|
g_signal_new ("dirty",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, dirty),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_DIRTY_MASK);
|
|
|
|
gimp_image_signals[SAVING] =
|
|
g_signal_new ("saving",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, saving),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[SAVED] =
|
|
g_signal_new ("saved",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, saved),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_FILE);
|
|
|
|
gimp_image_signals[EXPORTED] =
|
|
g_signal_new ("exported",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, exported),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_FILE);
|
|
|
|
gimp_image_signals[GUIDE_ADDED] =
|
|
g_signal_new ("guide-added",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, guide_added),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_GUIDE);
|
|
|
|
gimp_image_signals[GUIDE_REMOVED] =
|
|
g_signal_new ("guide-removed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, guide_removed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_GUIDE);
|
|
|
|
gimp_image_signals[GUIDE_MOVED] =
|
|
g_signal_new ("guide-moved",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, guide_moved),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_GUIDE);
|
|
|
|
gimp_image_signals[SAMPLE_POINT_ADDED] =
|
|
g_signal_new ("sample-point-added",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, sample_point_added),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_SAMPLE_POINT);
|
|
|
|
gimp_image_signals[SAMPLE_POINT_REMOVED] =
|
|
g_signal_new ("sample-point-removed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, sample_point_removed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_SAMPLE_POINT);
|
|
|
|
gimp_image_signals[SAMPLE_POINT_MOVED] =
|
|
g_signal_new ("sample-point-moved",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, sample_point_moved),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
GIMP_TYPE_SAMPLE_POINT);
|
|
|
|
gimp_image_signals[PARASITE_ATTACHED] =
|
|
g_signal_new ("parasite-attached",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, parasite_attached),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_STRING);
|
|
|
|
gimp_image_signals[PARASITE_DETACHED] =
|
|
g_signal_new ("parasite-detached",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, parasite_detached),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_STRING);
|
|
|
|
gimp_image_signals[COLORMAP_CHANGED] =
|
|
g_signal_new ("colormap-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, colormap_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_INT);
|
|
|
|
gimp_image_signals[UNDO_EVENT] =
|
|
g_signal_new ("undo-event",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, undo_event),
|
|
NULL, NULL,
|
|
gimp_marshal_VOID__ENUM_OBJECT,
|
|
G_TYPE_NONE, 2,
|
|
GIMP_TYPE_UNDO_EVENT,
|
|
GIMP_TYPE_UNDO);
|
|
|
|
gimp_image_signals[LAYER_SETS_CHANGED] =
|
|
g_signal_new ("layer-sets-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, layer_sets_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[CHANNEL_SETS_CHANGED] =
|
|
g_signal_new ("channel-sets-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, channel_sets_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
gimp_image_signals[VECTORS_SETS_CHANGED] =
|
|
g_signal_new ("vectors-sets-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpImageClass, vectors_sets_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
object_class->constructed = gimp_image_constructed;
|
|
object_class->set_property = gimp_image_set_property;
|
|
object_class->get_property = gimp_image_get_property;
|
|
object_class->dispose = gimp_image_dispose;
|
|
object_class->finalize = gimp_image_finalize;
|
|
|
|
gimp_object_class->name_changed = gimp_image_name_changed;
|
|
gimp_object_class->get_memsize = gimp_image_get_memsize;
|
|
|
|
viewable_class->default_icon_name = "gimp-image";
|
|
viewable_class->get_size = gimp_image_get_size;
|
|
viewable_class->size_changed = gimp_image_size_changed;
|
|
viewable_class->get_preview_size = gimp_image_get_preview_size;
|
|
viewable_class->get_popup_size = gimp_image_get_popup_size;
|
|
viewable_class->get_new_preview = gimp_image_get_new_preview;
|
|
viewable_class->get_new_pixbuf = gimp_image_get_new_pixbuf;
|
|
viewable_class->get_description = gimp_image_get_description;
|
|
|
|
klass->mode_changed = gimp_image_real_mode_changed;
|
|
klass->precision_changed = gimp_image_real_precision_changed;
|
|
klass->alpha_changed = NULL;
|
|
klass->floating_selection_changed = NULL;
|
|
klass->selected_layers_changed = NULL;
|
|
klass->selected_channels_changed = NULL;
|
|
klass->selected_vectors_changed = NULL;
|
|
klass->component_visibility_changed = NULL;
|
|
klass->component_active_changed = NULL;
|
|
klass->mask_changed = NULL;
|
|
klass->resolution_changed = gimp_image_real_resolution_changed;
|
|
klass->size_changed_detailed = gimp_image_real_size_changed_detailed;
|
|
klass->unit_changed = gimp_image_real_unit_changed;
|
|
klass->quick_mask_changed = NULL;
|
|
klass->selection_invalidate = NULL;
|
|
|
|
klass->clean = NULL;
|
|
klass->dirty = NULL;
|
|
klass->saving = NULL;
|
|
klass->saved = NULL;
|
|
klass->exported = NULL;
|
|
klass->guide_added = NULL;
|
|
klass->guide_removed = NULL;
|
|
klass->guide_moved = NULL;
|
|
klass->sample_point_added = NULL;
|
|
klass->sample_point_removed = NULL;
|
|
klass->sample_point_moved = NULL;
|
|
klass->parasite_attached = NULL;
|
|
klass->parasite_detached = NULL;
|
|
klass->colormap_changed = gimp_image_real_colormap_changed;
|
|
klass->undo_event = NULL;
|
|
|
|
g_object_class_install_property (object_class, PROP_GIMP,
|
|
g_param_spec_object ("gimp", NULL, NULL,
|
|
GIMP_TYPE_GIMP,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
g_object_class_install_property (object_class, PROP_ID,
|
|
g_param_spec_int ("id", NULL, NULL,
|
|
0, G_MAXINT, 0,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (object_class, PROP_WIDTH,
|
|
g_param_spec_int ("width", NULL, NULL,
|
|
1, GIMP_MAX_IMAGE_SIZE, 1,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
g_object_class_install_property (object_class, PROP_HEIGHT,
|
|
g_param_spec_int ("height", NULL, NULL,
|
|
1, GIMP_MAX_IMAGE_SIZE, 1,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
g_object_class_install_property (object_class, PROP_BASE_TYPE,
|
|
g_param_spec_enum ("base-type", NULL, NULL,
|
|
GIMP_TYPE_IMAGE_BASE_TYPE,
|
|
GIMP_RGB,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
g_object_class_install_property (object_class, PROP_PRECISION,
|
|
g_param_spec_enum ("precision", NULL, NULL,
|
|
GIMP_TYPE_PRECISION,
|
|
GIMP_PRECISION_U8_NON_LINEAR,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
g_object_class_install_property (object_class, PROP_METADATA,
|
|
g_param_spec_object ("metadata", NULL, NULL,
|
|
GEXIV2_TYPE_METADATA,
|
|
GIMP_PARAM_READABLE));
|
|
|
|
g_object_class_override_property (object_class, PROP_BUFFER, "buffer");
|
|
|
|
g_object_class_install_property (object_class, PROP_SYMMETRY,
|
|
g_param_spec_gtype ("symmetry",
|
|
NULL, _("Symmetry"),
|
|
GIMP_TYPE_SYMMETRY,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
g_object_class_install_property (object_class, PROP_CONVERTING,
|
|
g_param_spec_boolean ("converting",
|
|
NULL, NULL,
|
|
FALSE,
|
|
GIMP_PARAM_READWRITE));
|
|
}
|
|
|
|
static void
|
|
gimp_color_managed_iface_init (GimpColorManagedInterface *iface)
|
|
{
|
|
iface->get_icc_profile = gimp_image_color_managed_get_icc_profile;
|
|
iface->get_color_profile = gimp_image_color_managed_get_color_profile;
|
|
iface->profile_changed = gimp_image_color_managed_profile_changed;
|
|
iface->get_simulation_profile = gimp_image_color_managed_get_simulation_profile;
|
|
iface->simulation_profile_changed = gimp_image_color_managed_simulation_profile_changed;
|
|
iface->get_simulation_intent = gimp_image_color_managed_get_simulation_intent;
|
|
iface->simulation_intent_changed = gimp_image_color_managed_simulation_intent_changed;
|
|
iface->get_simulation_bpc = gimp_image_color_managed_get_simulation_bpc;
|
|
iface->simulation_bpc_changed = gimp_image_color_managed_simulation_bpc_changed;
|
|
}
|
|
|
|
static void
|
|
gimp_projectable_iface_init (GimpProjectableInterface *iface)
|
|
{
|
|
iface->flush = gimp_image_projectable_flush;
|
|
iface->get_image = gimp_image_get_image;
|
|
iface->get_format = gimp_image_get_proj_format;
|
|
iface->get_bounding_box = gimp_image_get_bounding_box;
|
|
iface->get_graph = gimp_image_get_graph;
|
|
iface->invalidate_preview = (void (*) (GimpProjectable*)) gimp_viewable_invalidate_preview;
|
|
}
|
|
|
|
static void
|
|
gimp_pickable_iface_init (GimpPickableInterface *iface)
|
|
{
|
|
iface->flush = gimp_image_pickable_flush;
|
|
iface->get_image = (GimpImage * (*) (GimpPickable *pickable)) gimp_image_get_image;
|
|
iface->get_format = (const Babl * (*) (GimpPickable *pickable)) gimp_image_get_proj_format;
|
|
iface->get_format_with_alpha = (const Babl * (*) (GimpPickable *pickable)) gimp_image_get_proj_format;
|
|
iface->get_buffer = gimp_image_get_buffer;
|
|
iface->get_pixel_at = gimp_image_get_pixel_at;
|
|
iface->get_opacity_at = gimp_image_get_opacity_at;
|
|
iface->get_pixel_average = gimp_image_get_pixel_average;
|
|
iface->pixel_to_srgb = gimp_image_pixel_to_srgb;
|
|
iface->srgb_to_pixel = gimp_image_srgb_to_pixel;
|
|
}
|
|
|
|
static void
|
|
gimp_image_init (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private = gimp_image_get_instance_private (image);
|
|
gint i;
|
|
|
|
image->priv = private;
|
|
|
|
private->ID = 0;
|
|
|
|
private->load_proc = NULL;
|
|
private->save_proc = NULL;
|
|
|
|
private->width = 0;
|
|
private->height = 0;
|
|
private->xresolution = 1.0;
|
|
private->yresolution = 1.0;
|
|
private->resolution_set = FALSE;
|
|
private->resolution_unit = GIMP_UNIT_INCH;
|
|
private->base_type = GIMP_RGB;
|
|
private->precision = GIMP_PRECISION_U8_NON_LINEAR;
|
|
private->new_layer_mode = -1;
|
|
|
|
private->show_all = 0;
|
|
private->bounding_box.x = 0;
|
|
private->bounding_box.y = 0;
|
|
private->bounding_box.width = 0;
|
|
private->bounding_box.height = 0;
|
|
private->pickable_buffer = NULL;
|
|
|
|
private->palette = NULL;
|
|
|
|
private->metadata = NULL;
|
|
|
|
private->simulation_profile = NULL;
|
|
private->simulation_intent = GIMP_COLOR_RENDERING_INTENT_RELATIVE_COLORIMETRIC;
|
|
private->simulation_bpc = FALSE;
|
|
|
|
private->dirty = 1;
|
|
private->dirty_time = 0;
|
|
private->undo_freeze_count = 0;
|
|
|
|
private->export_dirty = 1;
|
|
|
|
private->instance_count = 0;
|
|
private->disp_count = 0;
|
|
|
|
private->tattoo_state = 0;
|
|
|
|
private->projection = gimp_projection_new (GIMP_PROJECTABLE (image));
|
|
|
|
private->symmetries = NULL;
|
|
private->active_symmetry = NULL;
|
|
|
|
private->guides = NULL;
|
|
private->grid = NULL;
|
|
private->sample_points = NULL;
|
|
|
|
private->layers = gimp_item_tree_new (image,
|
|
GIMP_TYPE_LAYER_STACK,
|
|
GIMP_TYPE_LAYER);
|
|
private->channels = gimp_item_tree_new (image,
|
|
GIMP_TYPE_DRAWABLE_STACK,
|
|
GIMP_TYPE_CHANNEL);
|
|
private->vectors = gimp_item_tree_new (image,
|
|
GIMP_TYPE_ITEM_STACK,
|
|
GIMP_TYPE_VECTORS);
|
|
private->layer_stack = NULL;
|
|
|
|
private->stored_layer_sets = NULL;
|
|
private->stored_channel_sets = NULL;
|
|
private->stored_vectors_sets = NULL;
|
|
|
|
g_signal_connect (private->projection, "notify::buffer",
|
|
G_CALLBACK (gimp_image_projection_buffer_notify),
|
|
image);
|
|
|
|
g_signal_connect (private->layers, "notify::selected-items",
|
|
G_CALLBACK (gimp_image_selected_layers_notify),
|
|
image);
|
|
g_signal_connect (private->channels, "notify::selected-items",
|
|
G_CALLBACK (gimp_image_selected_channels_notify),
|
|
image);
|
|
g_signal_connect (private->vectors, "notify::selected-items",
|
|
G_CALLBACK (gimp_image_selected_vectors_notify),
|
|
image);
|
|
|
|
g_signal_connect_swapped (private->layers->container, "update",
|
|
G_CALLBACK (gimp_image_invalidate),
|
|
image);
|
|
|
|
private->layer_offset_x_handler =
|
|
gimp_container_add_handler (private->layers->container, "notify::offset-x",
|
|
G_CALLBACK (gimp_image_layer_offset_changed),
|
|
image);
|
|
private->layer_offset_y_handler =
|
|
gimp_container_add_handler (private->layers->container, "notify::offset-y",
|
|
G_CALLBACK (gimp_image_layer_offset_changed),
|
|
image);
|
|
private->layer_bounding_box_handler =
|
|
gimp_container_add_handler (private->layers->container, "bounding-box-changed",
|
|
G_CALLBACK (gimp_image_layer_bounding_box_changed),
|
|
image);
|
|
private->layer_alpha_handler =
|
|
gimp_container_add_handler (private->layers->container, "alpha-changed",
|
|
G_CALLBACK (gimp_image_layer_alpha_changed),
|
|
image);
|
|
|
|
g_signal_connect (private->layers->container, "add",
|
|
G_CALLBACK (gimp_image_layers_changed),
|
|
image);
|
|
g_signal_connect (private->layers->container, "remove",
|
|
G_CALLBACK (gimp_image_layers_changed),
|
|
image);
|
|
|
|
g_signal_connect_swapped (private->channels->container, "update",
|
|
G_CALLBACK (gimp_image_invalidate),
|
|
image);
|
|
|
|
private->channel_name_changed_handler =
|
|
gimp_container_add_handler (private->channels->container, "name-changed",
|
|
G_CALLBACK (gimp_image_channel_name_changed),
|
|
image);
|
|
private->channel_color_changed_handler =
|
|
gimp_container_add_handler (private->channels->container, "color-changed",
|
|
G_CALLBACK (gimp_image_channel_color_changed),
|
|
image);
|
|
|
|
g_signal_connect (private->channels->container, "add",
|
|
G_CALLBACK (gimp_image_channel_add),
|
|
image);
|
|
g_signal_connect (private->channels->container, "remove",
|
|
G_CALLBACK (gimp_image_channel_remove),
|
|
image);
|
|
|
|
private->floating_sel = NULL;
|
|
private->selection_mask = NULL;
|
|
|
|
private->parasites = gimp_parasite_list_new ();
|
|
|
|
for (i = 0; i < MAX_CHANNELS; i++)
|
|
{
|
|
private->visible[i] = TRUE;
|
|
private->active[i] = TRUE;
|
|
}
|
|
|
|
private->quick_mask_state = FALSE;
|
|
private->quick_mask_inverted = FALSE;
|
|
gimp_rgba_set (&private->quick_mask_color, 1.0, 0.0, 0.0, 0.5);
|
|
|
|
private->undo_stack = gimp_undo_stack_new (image);
|
|
private->redo_stack = gimp_undo_stack_new (image);
|
|
private->group_count = 0;
|
|
private->pushing_undo_group = GIMP_UNDO_GROUP_NONE;
|
|
|
|
private->flush_accum.alpha_changed = FALSE;
|
|
private->flush_accum.mask_changed = FALSE;
|
|
private->flush_accum.floating_selection_changed = FALSE;
|
|
private->flush_accum.preview_invalidated = FALSE;
|
|
}
|
|
|
|
static void
|
|
gimp_image_constructed (GObject *object)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (object);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
GimpChannel *selection;
|
|
GimpCoreConfig *config;
|
|
GimpTemplate *template;
|
|
|
|
G_OBJECT_CLASS (parent_class)->constructed (object);
|
|
|
|
gimp_assert (GIMP_IS_GIMP (image->gimp));
|
|
|
|
config = image->gimp->config;
|
|
|
|
private->ID = gimp_id_table_insert (image->gimp->image_table, image);
|
|
|
|
template = config->default_image;
|
|
|
|
private->xresolution = gimp_template_get_resolution_x (template);
|
|
private->yresolution = gimp_template_get_resolution_y (template);
|
|
private->resolution_unit = gimp_template_get_resolution_unit (template);
|
|
|
|
private->grid = gimp_config_duplicate (GIMP_CONFIG (config->default_grid));
|
|
|
|
private->quick_mask_color = config->quick_mask_color;
|
|
|
|
gimp_image_update_bounding_box (image);
|
|
|
|
if (private->base_type == GIMP_INDEXED)
|
|
gimp_image_colormap_init (image);
|
|
|
|
selection = gimp_selection_new (image,
|
|
gimp_image_get_width (image),
|
|
gimp_image_get_height (image));
|
|
gimp_image_take_mask (image, selection);
|
|
|
|
g_signal_connect_object (config, "notify::transparency-type",
|
|
G_CALLBACK (gimp_item_stack_invalidate_previews),
|
|
private->layers->container, G_CONNECT_SWAPPED);
|
|
g_signal_connect_object (config, "notify::transparency-size",
|
|
G_CALLBACK (gimp_item_stack_invalidate_previews),
|
|
private->layers->container, G_CONNECT_SWAPPED);
|
|
g_signal_connect_object (config, "notify::transparency-custom-color1",
|
|
G_CALLBACK (gimp_item_stack_invalidate_previews),
|
|
private->layers->container, G_CONNECT_SWAPPED);
|
|
g_signal_connect_object (config, "notify::transparency-custom-color2",
|
|
G_CALLBACK (gimp_item_stack_invalidate_previews),
|
|
private->layers->container, G_CONNECT_SWAPPED);
|
|
g_signal_connect_object (config, "notify::layer-previews",
|
|
G_CALLBACK (gimp_viewable_size_changed),
|
|
image, G_CONNECT_SWAPPED);
|
|
g_signal_connect_object (config, "notify::group-layer-previews",
|
|
G_CALLBACK (gimp_viewable_size_changed),
|
|
image, G_CONNECT_SWAPPED);
|
|
|
|
gimp_container_add (image->gimp->images, GIMP_OBJECT (image));
|
|
}
|
|
|
|
static void
|
|
gimp_image_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (object);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_GIMP:
|
|
image->gimp = g_value_get_object (value);
|
|
break;
|
|
|
|
case PROP_WIDTH:
|
|
private->width = g_value_get_int (value);
|
|
break;
|
|
case PROP_HEIGHT:
|
|
private->height = g_value_get_int (value);
|
|
break;
|
|
|
|
case PROP_BASE_TYPE:
|
|
private->base_type = g_value_get_enum (value);
|
|
_gimp_image_free_color_transforms (image);
|
|
break;
|
|
|
|
case PROP_PRECISION:
|
|
private->precision = g_value_get_enum (value);
|
|
_gimp_image_free_color_transforms (image);
|
|
break;
|
|
|
|
case PROP_SYMMETRY:
|
|
{
|
|
GList *iter;
|
|
GType type = g_value_get_gtype (value);
|
|
|
|
if (private->active_symmetry)
|
|
g_object_set (private->active_symmetry,
|
|
"active", FALSE,
|
|
NULL);
|
|
private->active_symmetry = NULL;
|
|
|
|
for (iter = private->symmetries; iter; iter = g_list_next (iter))
|
|
{
|
|
GimpSymmetry *sym = iter->data;
|
|
|
|
if (type == G_TYPE_FROM_INSTANCE (sym))
|
|
private->active_symmetry = iter->data;
|
|
}
|
|
|
|
if (! private->active_symmetry &&
|
|
g_type_is_a (type, GIMP_TYPE_SYMMETRY))
|
|
{
|
|
GimpSymmetry *sym = gimp_image_symmetry_new (image, type);
|
|
|
|
gimp_image_symmetry_add (image, sym);
|
|
g_object_unref (sym);
|
|
|
|
private->active_symmetry = sym;
|
|
}
|
|
|
|
if (private->active_symmetry)
|
|
g_object_set (private->active_symmetry,
|
|
"active", TRUE,
|
|
NULL);
|
|
}
|
|
break;
|
|
|
|
case PROP_CONVERTING:
|
|
private->converting = g_value_get_boolean (value);
|
|
break;
|
|
|
|
case PROP_ID:
|
|
case PROP_METADATA:
|
|
case PROP_BUFFER:
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (object);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_GIMP:
|
|
g_value_set_object (value, image->gimp);
|
|
break;
|
|
case PROP_ID:
|
|
g_value_set_int (value, private->ID);
|
|
break;
|
|
case PROP_WIDTH:
|
|
g_value_set_int (value, private->width);
|
|
break;
|
|
case PROP_HEIGHT:
|
|
g_value_set_int (value, private->height);
|
|
break;
|
|
case PROP_BASE_TYPE:
|
|
g_value_set_enum (value, private->base_type);
|
|
break;
|
|
case PROP_PRECISION:
|
|
g_value_set_enum (value, private->precision);
|
|
break;
|
|
case PROP_METADATA:
|
|
g_value_set_object (value, gimp_image_get_metadata (image));
|
|
break;
|
|
case PROP_BUFFER:
|
|
g_value_set_object (value, gimp_image_get_buffer (GIMP_PICKABLE (image)));
|
|
break;
|
|
case PROP_SYMMETRY:
|
|
g_value_set_gtype (value,
|
|
private->active_symmetry ?
|
|
G_TYPE_FROM_INSTANCE (private->active_symmetry) :
|
|
G_TYPE_NONE);
|
|
break;
|
|
case PROP_CONVERTING:
|
|
g_value_set_boolean (value, private->converting);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_dispose (GObject *object)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (object);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (private->palette)
|
|
gimp_image_colormap_dispose (image);
|
|
|
|
gimp_image_undo_free (image);
|
|
|
|
g_list_free_full (private->stored_layer_sets, g_object_unref);
|
|
g_list_free_full (private->stored_channel_sets, g_object_unref);
|
|
g_list_free_full (private->stored_vectors_sets, g_object_unref);
|
|
|
|
g_signal_handlers_disconnect_by_func (private->layers->container,
|
|
gimp_image_invalidate,
|
|
image);
|
|
|
|
gimp_container_remove_handler (private->layers->container,
|
|
private->layer_offset_x_handler);
|
|
gimp_container_remove_handler (private->layers->container,
|
|
private->layer_offset_y_handler);
|
|
gimp_container_remove_handler (private->layers->container,
|
|
private->layer_bounding_box_handler);
|
|
gimp_container_remove_handler (private->layers->container,
|
|
private->layer_alpha_handler);
|
|
|
|
g_signal_handlers_disconnect_by_func (private->layers->container,
|
|
gimp_image_layers_changed,
|
|
image);
|
|
|
|
g_signal_handlers_disconnect_by_func (private->channels->container,
|
|
gimp_image_invalidate,
|
|
image);
|
|
|
|
gimp_container_remove_handler (private->channels->container,
|
|
private->channel_name_changed_handler);
|
|
gimp_container_remove_handler (private->channels->container,
|
|
private->channel_color_changed_handler);
|
|
|
|
g_signal_handlers_disconnect_by_func (private->channels->container,
|
|
gimp_image_channel_add,
|
|
image);
|
|
g_signal_handlers_disconnect_by_func (private->channels->container,
|
|
gimp_image_channel_remove,
|
|
image);
|
|
|
|
g_object_run_dispose (G_OBJECT (private->layers));
|
|
g_object_run_dispose (G_OBJECT (private->channels));
|
|
g_object_run_dispose (G_OBJECT (private->vectors));
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gimp_image_finalize (GObject *object)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (object);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
g_clear_object (&private->projection);
|
|
g_clear_object (&private->graph);
|
|
private->visible_mask = NULL;
|
|
|
|
if (private->palette)
|
|
gimp_image_colormap_free (image);
|
|
|
|
_gimp_image_free_color_profile (image);
|
|
|
|
g_clear_object (&private->pickable_buffer);
|
|
g_clear_object (&private->metadata);
|
|
g_clear_object (&private->file);
|
|
g_clear_object (&private->imported_file);
|
|
g_clear_object (&private->exported_file);
|
|
g_clear_object (&private->save_a_copy_file);
|
|
g_clear_object (&private->untitled_file);
|
|
g_clear_object (&private->layers);
|
|
g_clear_object (&private->channels);
|
|
g_clear_object (&private->vectors);
|
|
|
|
if (private->layer_stack)
|
|
{
|
|
g_slist_free_full (private->layer_stack,
|
|
(GDestroyNotify) g_list_free);
|
|
private->layer_stack = NULL;
|
|
}
|
|
|
|
g_clear_object (&private->selection_mask);
|
|
g_clear_object (&private->parasites);
|
|
|
|
if (private->guides)
|
|
{
|
|
g_list_free_full (private->guides, (GDestroyNotify) g_object_unref);
|
|
private->guides = NULL;
|
|
}
|
|
|
|
if (private->symmetries)
|
|
{
|
|
g_list_free_full (private->symmetries, g_object_unref);
|
|
private->symmetries = NULL;
|
|
}
|
|
|
|
g_clear_object (&private->grid);
|
|
|
|
if (private->sample_points)
|
|
{
|
|
g_list_free_full (private->sample_points,
|
|
(GDestroyNotify) g_object_unref);
|
|
private->sample_points = NULL;
|
|
}
|
|
|
|
g_clear_object (&private->undo_stack);
|
|
g_clear_object (&private->redo_stack);
|
|
|
|
if (image->gimp && image->gimp->image_table)
|
|
{
|
|
gimp_id_table_remove (image->gimp->image_table, private->ID);
|
|
image->gimp = NULL;
|
|
}
|
|
|
|
g_clear_pointer (&private->display_name, g_free);
|
|
g_clear_pointer (&private->display_path, g_free);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
|
|
/* Don't just free the list silently. It is supposed to be empty.
|
|
* Instead just assert this fact to warn on improper management of
|
|
* hidden items.
|
|
*/
|
|
if (private->hidden_items != NULL)
|
|
{
|
|
g_warning ("%s: the hidden items list should be empty (%d items remaining).",
|
|
G_STRFUNC, g_list_length (private->hidden_items));
|
|
g_list_free (private->hidden_items);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_name_changed (GimpObject *object)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (object);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
const gchar *name;
|
|
|
|
if (GIMP_OBJECT_CLASS (parent_class)->name_changed)
|
|
GIMP_OBJECT_CLASS (parent_class)->name_changed (object);
|
|
|
|
g_clear_pointer (&private->display_name, g_free);
|
|
g_clear_pointer (&private->display_path, g_free);
|
|
|
|
/* We never want the empty string as a name, so change empty strings
|
|
* to NULL strings (without emitting the "name-changed" signal
|
|
* again)
|
|
*/
|
|
name = gimp_object_get_name (object);
|
|
if (name && strlen (name) == 0)
|
|
{
|
|
gimp_object_name_free (object);
|
|
name = NULL;
|
|
}
|
|
|
|
g_clear_object (&private->file);
|
|
|
|
if (name)
|
|
private->file = g_file_new_for_uri (name);
|
|
}
|
|
|
|
static gint64
|
|
gimp_image_get_memsize (GimpObject *object,
|
|
gint64 *gui_size)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (object);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
gint64 memsize = 0;
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->palette),
|
|
gui_size);
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->projection),
|
|
gui_size);
|
|
|
|
memsize += gimp_g_list_get_memsize (gimp_image_get_guides (image),
|
|
sizeof (GimpGuide));
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->grid), gui_size);
|
|
|
|
memsize += gimp_g_list_get_memsize (gimp_image_get_sample_points (image),
|
|
sizeof (GimpSamplePoint));
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->layers),
|
|
gui_size);
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->channels),
|
|
gui_size);
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->vectors),
|
|
gui_size);
|
|
|
|
memsize += gimp_g_slist_get_memsize (private->layer_stack, 0);
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->selection_mask),
|
|
gui_size);
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->parasites),
|
|
gui_size);
|
|
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->undo_stack),
|
|
gui_size);
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (private->redo_stack),
|
|
gui_size);
|
|
|
|
return memsize + GIMP_OBJECT_CLASS (parent_class)->get_memsize (object,
|
|
gui_size);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_image_get_size (GimpViewable *viewable,
|
|
gint *width,
|
|
gint *height)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (viewable);
|
|
|
|
*width = gimp_image_get_width (image);
|
|
*height = gimp_image_get_height (image);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_image_size_changed (GimpViewable *viewable)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (viewable);
|
|
GList *all_items;
|
|
GList *list;
|
|
|
|
if (GIMP_VIEWABLE_CLASS (parent_class)->size_changed)
|
|
GIMP_VIEWABLE_CLASS (parent_class)->size_changed (viewable);
|
|
|
|
all_items = gimp_image_get_layer_list (image);
|
|
for (list = all_items; list; list = g_list_next (list))
|
|
{
|
|
GimpLayerMask *mask = gimp_layer_get_mask (GIMP_LAYER (list->data));
|
|
|
|
gimp_viewable_size_changed (GIMP_VIEWABLE (list->data));
|
|
|
|
if (mask)
|
|
gimp_viewable_size_changed (GIMP_VIEWABLE (mask));
|
|
}
|
|
g_list_free (all_items);
|
|
|
|
all_items = gimp_image_get_channel_list (image);
|
|
g_list_free_full (all_items, (GDestroyNotify) gimp_viewable_size_changed);
|
|
|
|
all_items = gimp_image_get_vectors_list (image);
|
|
g_list_free_full (all_items, (GDestroyNotify) gimp_viewable_size_changed);
|
|
|
|
gimp_viewable_size_changed (GIMP_VIEWABLE (gimp_image_get_mask (image)));
|
|
|
|
gimp_image_metadata_update_pixel_size (image);
|
|
|
|
g_clear_object (&GIMP_IMAGE_GET_PRIVATE (image)->pickable_buffer);
|
|
|
|
gimp_image_update_bounding_box (image);
|
|
}
|
|
|
|
static gchar *
|
|
gimp_image_get_description (GimpViewable *viewable,
|
|
gchar **tooltip)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (viewable);
|
|
|
|
if (tooltip)
|
|
*tooltip = g_strdup (gimp_image_get_display_path (image));
|
|
|
|
return g_strdup_printf ("%s-%d",
|
|
gimp_image_get_display_name (image),
|
|
gimp_image_get_id (image));
|
|
}
|
|
|
|
static void
|
|
gimp_image_real_mode_changed (GimpImage *image)
|
|
{
|
|
gimp_projectable_structure_changed (GIMP_PROJECTABLE (image));
|
|
}
|
|
|
|
static void
|
|
gimp_image_real_precision_changed (GimpImage *image)
|
|
{
|
|
gimp_image_metadata_update_bits_per_sample (image);
|
|
|
|
gimp_projectable_structure_changed (GIMP_PROJECTABLE (image));
|
|
}
|
|
|
|
static void
|
|
gimp_image_real_resolution_changed (GimpImage *image)
|
|
{
|
|
gimp_image_metadata_update_resolution (image);
|
|
}
|
|
|
|
static void
|
|
gimp_image_real_size_changed_detailed (GimpImage *image,
|
|
gint previous_origin_x,
|
|
gint previous_origin_y,
|
|
gint previous_width,
|
|
gint previous_height)
|
|
{
|
|
/* Whenever GimpImage::size-changed-detailed is emitted, so is
|
|
* GimpViewable::size-changed. Clients choose what signal to listen
|
|
* to depending on how much info they need.
|
|
*/
|
|
gimp_viewable_size_changed (GIMP_VIEWABLE (image));
|
|
}
|
|
|
|
static void
|
|
gimp_image_real_unit_changed (GimpImage *image)
|
|
{
|
|
gimp_image_metadata_update_resolution (image);
|
|
}
|
|
|
|
static void
|
|
gimp_image_real_colormap_changed (GimpImage *image,
|
|
gint color_index)
|
|
{
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
gimp_image_colormap_update_formats (image);
|
|
|
|
if (gimp_image_get_base_type (image) == GIMP_INDEXED)
|
|
{
|
|
/* A colormap alteration affects the whole image */
|
|
gimp_image_invalidate_all (image);
|
|
|
|
gimp_item_stack_invalidate_previews (GIMP_ITEM_STACK (private->layers->container));
|
|
}
|
|
}
|
|
|
|
static const guint8 *
|
|
gimp_image_color_managed_get_icc_profile (GimpColorManaged *managed,
|
|
gsize *len)
|
|
{
|
|
return gimp_image_get_icc_profile (GIMP_IMAGE (managed), len);
|
|
}
|
|
|
|
static GimpColorProfile *
|
|
gimp_image_color_managed_get_color_profile (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
GimpColorProfile *profile;
|
|
|
|
profile = gimp_image_get_color_profile (image);
|
|
|
|
if (! profile)
|
|
profile = gimp_image_get_builtin_color_profile (image);
|
|
|
|
return profile;
|
|
}
|
|
|
|
static void
|
|
gimp_image_color_managed_profile_changed (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
GimpItemStack *layers = GIMP_ITEM_STACK (gimp_image_get_layers (image));
|
|
|
|
gimp_image_metadata_update_colorspace (image);
|
|
|
|
gimp_projectable_structure_changed (GIMP_PROJECTABLE (image));
|
|
gimp_viewable_invalidate_preview (GIMP_VIEWABLE (image));
|
|
gimp_item_stack_profile_changed (layers);
|
|
}
|
|
|
|
static GimpColorProfile *
|
|
gimp_image_color_managed_get_simulation_profile (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
GimpColorProfile *profile;
|
|
|
|
profile = gimp_image_get_simulation_profile (image);
|
|
|
|
return profile;
|
|
}
|
|
|
|
static void
|
|
gimp_image_color_managed_simulation_profile_changed (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
|
|
gimp_projectable_structure_changed (GIMP_PROJECTABLE (image));
|
|
gimp_viewable_invalidate_preview (GIMP_VIEWABLE (image));
|
|
}
|
|
|
|
static GimpColorRenderingIntent
|
|
gimp_image_color_managed_get_simulation_intent (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
|
|
return gimp_image_get_simulation_intent (image);
|
|
}
|
|
|
|
static void
|
|
gimp_image_color_managed_simulation_intent_changed (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
|
|
gimp_projectable_structure_changed (GIMP_PROJECTABLE (image));
|
|
gimp_viewable_invalidate_preview (GIMP_VIEWABLE (image));
|
|
}
|
|
|
|
static gboolean
|
|
gimp_image_color_managed_get_simulation_bpc (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
|
|
return gimp_image_get_simulation_bpc (image);
|
|
}
|
|
|
|
static void
|
|
gimp_image_color_managed_simulation_bpc_changed (GimpColorManaged *managed)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (managed);
|
|
|
|
gimp_projectable_structure_changed (GIMP_PROJECTABLE (image));
|
|
gimp_viewable_invalidate_preview (GIMP_VIEWABLE (image));
|
|
}
|
|
|
|
static void
|
|
gimp_image_projectable_flush (GimpProjectable *projectable,
|
|
gboolean invalidate_preview)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (projectable);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (private->flush_accum.alpha_changed)
|
|
{
|
|
gimp_image_alpha_changed (image);
|
|
private->flush_accum.alpha_changed = FALSE;
|
|
}
|
|
|
|
if (private->flush_accum.mask_changed)
|
|
{
|
|
gimp_image_mask_changed (image);
|
|
private->flush_accum.mask_changed = FALSE;
|
|
}
|
|
|
|
if (private->flush_accum.floating_selection_changed)
|
|
{
|
|
gimp_image_floating_selection_changed (image);
|
|
private->flush_accum.floating_selection_changed = FALSE;
|
|
}
|
|
|
|
if (private->flush_accum.preview_invalidated)
|
|
{
|
|
/* don't invalidate the preview here, the projection does this when
|
|
* it is completely constructed.
|
|
*/
|
|
private->flush_accum.preview_invalidated = FALSE;
|
|
}
|
|
}
|
|
|
|
static GimpImage *
|
|
gimp_image_get_image (GimpProjectable *projectable)
|
|
{
|
|
return GIMP_IMAGE (projectable);
|
|
}
|
|
|
|
static const Babl *
|
|
gimp_image_get_proj_format (GimpProjectable *projectable)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (projectable);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
switch (private->base_type)
|
|
{
|
|
case GIMP_RGB:
|
|
case GIMP_INDEXED:
|
|
return gimp_image_get_format (image, GIMP_RGB,
|
|
gimp_image_get_precision (image), TRUE,
|
|
gimp_image_get_layer_space (image));
|
|
|
|
case GIMP_GRAY:
|
|
return gimp_image_get_format (image, GIMP_GRAY,
|
|
gimp_image_get_precision (image), TRUE,
|
|
gimp_image_get_layer_space (image));
|
|
}
|
|
|
|
g_return_val_if_reached (NULL);
|
|
}
|
|
|
|
static void
|
|
gimp_image_pickable_flush (GimpPickable *pickable)
|
|
{
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (pickable);
|
|
|
|
return gimp_pickable_flush (GIMP_PICKABLE (private->projection));
|
|
}
|
|
|
|
static GeglBuffer *
|
|
gimp_image_get_buffer (GimpPickable *pickable)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (pickable);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (! private->pickable_buffer)
|
|
{
|
|
GeglBuffer *buffer;
|
|
|
|
buffer = gimp_pickable_get_buffer (GIMP_PICKABLE (private->projection));
|
|
|
|
if (! private->show_all)
|
|
{
|
|
private->pickable_buffer = g_object_ref (buffer);
|
|
}
|
|
else
|
|
{
|
|
private->pickable_buffer = gegl_buffer_create_sub_buffer (
|
|
buffer,
|
|
GEGL_RECTANGLE (0, 0,
|
|
gimp_image_get_width (image),
|
|
gimp_image_get_height (image)));
|
|
}
|
|
}
|
|
|
|
return private->pickable_buffer;
|
|
}
|
|
|
|
static gboolean
|
|
gimp_image_get_pixel_at (GimpPickable *pickable,
|
|
gint x,
|
|
gint y,
|
|
const Babl *format,
|
|
gpointer pixel)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (pickable);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (x >= 0 &&
|
|
y >= 0 &&
|
|
x < gimp_image_get_width (image) &&
|
|
y < gimp_image_get_height (image))
|
|
{
|
|
return gimp_pickable_get_pixel_at (GIMP_PICKABLE (private->projection),
|
|
x, y, format, pixel);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gdouble
|
|
gimp_image_get_opacity_at (GimpPickable *pickable,
|
|
gint x,
|
|
gint y)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (pickable);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (x >= 0 &&
|
|
y >= 0 &&
|
|
x < gimp_image_get_width (image) &&
|
|
y < gimp_image_get_height (image))
|
|
{
|
|
return gimp_pickable_get_opacity_at (GIMP_PICKABLE (private->projection),
|
|
x, y);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
gimp_image_get_pixel_average (GimpPickable *pickable,
|
|
const GeglRectangle *rect,
|
|
const Babl *format,
|
|
gpointer pixel)
|
|
{
|
|
GeglBuffer *buffer = gimp_pickable_get_buffer (pickable);
|
|
|
|
return gimp_gegl_average_color (buffer, rect, TRUE, GEGL_ABYSS_NONE, format,
|
|
pixel);
|
|
}
|
|
|
|
static void
|
|
gimp_image_pixel_to_srgb (GimpPickable *pickable,
|
|
const Babl *format,
|
|
gpointer pixel,
|
|
GimpRGB *color)
|
|
{
|
|
gimp_image_color_profile_pixel_to_srgb (GIMP_IMAGE (pickable),
|
|
format, pixel, color);
|
|
}
|
|
|
|
static void
|
|
gimp_image_srgb_to_pixel (GimpPickable *pickable,
|
|
const GimpRGB *color,
|
|
const Babl *format,
|
|
gpointer pixel)
|
|
{
|
|
gimp_image_color_profile_srgb_to_pixel (GIMP_IMAGE (pickable),
|
|
color, format, pixel);
|
|
}
|
|
|
|
static GeglRectangle
|
|
gimp_image_get_bounding_box (GimpProjectable *projectable)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (projectable);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->bounding_box;
|
|
}
|
|
|
|
static GeglNode *
|
|
gimp_image_get_graph (GimpProjectable *projectable)
|
|
{
|
|
GimpImage *image = GIMP_IMAGE (projectable);
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
GeglNode *layers_node;
|
|
GeglNode *channels_node;
|
|
GeglNode *output;
|
|
GimpComponentMask mask;
|
|
|
|
if (private->graph)
|
|
return private->graph;
|
|
|
|
private->graph = gegl_node_new ();
|
|
|
|
layers_node =
|
|
gimp_filter_stack_get_graph (GIMP_FILTER_STACK (private->layers->container));
|
|
|
|
gegl_node_add_child (private->graph, layers_node);
|
|
|
|
mask = ~gimp_image_get_visible_mask (image) & GIMP_COMPONENT_MASK_ALL;
|
|
|
|
private->visible_mask =
|
|
gegl_node_new_child (private->graph,
|
|
"operation", "gimp:mask-components",
|
|
"mask", mask,
|
|
"alpha", 1.0,
|
|
NULL);
|
|
|
|
gegl_node_connect_to (layers_node, "output",
|
|
private->visible_mask, "input");
|
|
|
|
channels_node =
|
|
gimp_filter_stack_get_graph (GIMP_FILTER_STACK (private->channels->container));
|
|
|
|
gegl_node_add_child (private->graph, channels_node);
|
|
|
|
gegl_node_connect_to (private->visible_mask, "output",
|
|
channels_node, "input");
|
|
|
|
output = gegl_node_get_output_proxy (private->graph, "output");
|
|
|
|
gegl_node_connect_to (channels_node, "output",
|
|
output, "input");
|
|
|
|
return private->graph;
|
|
}
|
|
|
|
static void
|
|
gimp_image_projection_buffer_notify (GimpProjection *projection,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image)
|
|
{
|
|
g_clear_object (&GIMP_IMAGE_GET_PRIVATE (image)->pickable_buffer);
|
|
}
|
|
|
|
static void
|
|
gimp_image_mask_update (GimpDrawable *drawable,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height,
|
|
GimpImage *image)
|
|
{
|
|
GIMP_IMAGE_GET_PRIVATE (image)->flush_accum.mask_changed = TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_image_layers_changed (GimpContainer *container,
|
|
GimpChannel *channel,
|
|
GimpImage *image)
|
|
{
|
|
gimp_image_update_bounding_box (image);
|
|
}
|
|
|
|
static void
|
|
gimp_image_layer_offset_changed (GimpDrawable *drawable,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image)
|
|
{
|
|
gimp_image_update_bounding_box (image);
|
|
}
|
|
|
|
static void
|
|
gimp_image_layer_bounding_box_changed (GimpDrawable *drawable,
|
|
GimpImage *image)
|
|
{
|
|
gimp_image_update_bounding_box (image);
|
|
}
|
|
|
|
static void
|
|
gimp_image_layer_alpha_changed (GimpDrawable *drawable,
|
|
GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (gimp_container_get_n_children (private->layers->container) == 1)
|
|
private->flush_accum.alpha_changed = TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_image_channel_add (GimpContainer *container,
|
|
GimpChannel *channel,
|
|
GimpImage *image)
|
|
{
|
|
if (! strcmp (GIMP_IMAGE_QUICK_MASK_NAME,
|
|
gimp_object_get_name (channel)))
|
|
{
|
|
gimp_image_set_quick_mask_state (image, TRUE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_channel_remove (GimpContainer *container,
|
|
GimpChannel *channel,
|
|
GimpImage *image)
|
|
{
|
|
if (! strcmp (GIMP_IMAGE_QUICK_MASK_NAME,
|
|
gimp_object_get_name (channel)))
|
|
{
|
|
gimp_image_set_quick_mask_state (image, FALSE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_channel_name_changed (GimpChannel *channel,
|
|
GimpImage *image)
|
|
{
|
|
if (! strcmp (GIMP_IMAGE_QUICK_MASK_NAME,
|
|
gimp_object_get_name (channel)))
|
|
{
|
|
gimp_image_set_quick_mask_state (image, TRUE);
|
|
}
|
|
else if (gimp_image_get_quick_mask_state (image) &&
|
|
! gimp_image_get_quick_mask (image))
|
|
{
|
|
gimp_image_set_quick_mask_state (image, FALSE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_channel_color_changed (GimpChannel *channel,
|
|
GimpImage *image)
|
|
{
|
|
if (! strcmp (GIMP_IMAGE_QUICK_MASK_NAME,
|
|
gimp_object_get_name (channel)))
|
|
{
|
|
GIMP_IMAGE_GET_PRIVATE (image)->quick_mask_color = channel->color;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_selected_layers_notify (GimpItemTree *tree,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
GList *layers = gimp_image_get_selected_layers (image);
|
|
|
|
if (layers)
|
|
{
|
|
/* Configure the layer stack to reflect this change */
|
|
GSList *prev_layers;
|
|
|
|
while ((prev_layers = g_slist_find_custom (private->layer_stack, layers,
|
|
(GCompareFunc) gimp_image_layer_stack_cmp)))
|
|
{
|
|
g_list_free (prev_layers->data);
|
|
private->layer_stack = g_slist_delete_link (private->layer_stack,
|
|
prev_layers);
|
|
}
|
|
private->layer_stack = g_slist_prepend (private->layer_stack, g_list_copy (layers));
|
|
}
|
|
|
|
g_signal_emit (image, gimp_image_signals[SELECTED_LAYERS_CHANGED], 0);
|
|
|
|
if (layers && gimp_image_get_selected_channels (image))
|
|
gimp_image_set_selected_channels (image, NULL);
|
|
}
|
|
|
|
static void
|
|
gimp_image_selected_channels_notify (GimpItemTree *tree,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image)
|
|
{
|
|
GList *channels = gimp_image_get_selected_channels (image);
|
|
|
|
g_signal_emit (image, gimp_image_signals[SELECTED_CHANNELS_CHANGED], 0);
|
|
|
|
if (channels && gimp_image_get_selected_layers (image))
|
|
gimp_image_set_selected_layers (image, NULL);
|
|
}
|
|
|
|
static void
|
|
gimp_image_selected_vectors_notify (GimpItemTree *tree,
|
|
const GParamSpec *pspec,
|
|
GimpImage *image)
|
|
{
|
|
g_signal_emit (image, gimp_image_signals[SELECTED_VECTORS_CHANGED], 0);
|
|
}
|
|
|
|
static void
|
|
gimp_image_freeze_bounding_box (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->bounding_box_freeze_count++;
|
|
}
|
|
|
|
static void
|
|
gimp_image_thaw_bounding_box (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->bounding_box_freeze_count--;
|
|
|
|
if (private->bounding_box_freeze_count == 0 &&
|
|
private->bounding_box_update_pending)
|
|
{
|
|
private->bounding_box_update_pending = FALSE;
|
|
|
|
gimp_image_update_bounding_box (image);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_image_update_bounding_box (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
GeglRectangle bounding_box;
|
|
|
|
if (private->bounding_box_freeze_count > 0)
|
|
{
|
|
private->bounding_box_update_pending = TRUE;
|
|
|
|
return;
|
|
}
|
|
|
|
bounding_box.x = 0;
|
|
bounding_box.y = 0;
|
|
bounding_box.width = gimp_image_get_width (image);
|
|
bounding_box.height = gimp_image_get_height (image);
|
|
|
|
if (private->show_all)
|
|
{
|
|
GList *iter;
|
|
|
|
for (iter = gimp_image_get_layer_iter (image);
|
|
iter;
|
|
iter = g_list_next (iter))
|
|
{
|
|
GimpLayer *layer = iter->data;
|
|
GeglRectangle layer_bounding_box;
|
|
gint offset_x;
|
|
gint offset_y;
|
|
|
|
gimp_item_get_offset (GIMP_ITEM (layer), &offset_x, &offset_y);
|
|
|
|
layer_bounding_box = gimp_drawable_get_bounding_box (
|
|
GIMP_DRAWABLE (layer));
|
|
|
|
layer_bounding_box.x += offset_x;
|
|
layer_bounding_box.y += offset_y;
|
|
|
|
gegl_rectangle_bounding_box (&bounding_box,
|
|
&bounding_box, &layer_bounding_box);
|
|
}
|
|
}
|
|
|
|
if (! gegl_rectangle_equal (&bounding_box, &private->bounding_box))
|
|
{
|
|
private->bounding_box = bounding_box;
|
|
|
|
gimp_projectable_bounds_changed (GIMP_PROJECTABLE (image), 0, 0);
|
|
}
|
|
}
|
|
|
|
static gint
|
|
gimp_image_layer_stack_cmp (GList *layers1,
|
|
GList *layers2)
|
|
{
|
|
if (g_list_length (layers1) != g_list_length (layers2))
|
|
{
|
|
/* We don't really need to order lists of layers, and only care
|
|
* about identity.
|
|
*/
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
GList *iter;
|
|
|
|
for (iter = layers1; iter; iter = iter->next)
|
|
{
|
|
if (! g_list_find (layers2, iter->data))
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_image_rec_remove_layer_stack_dups:
|
|
* @image:
|
|
*
|
|
* Recursively remove duplicates from the layer stack. You should not
|
|
* call this directly, call gimp_image_clean_layer_stack() instead.
|
|
*/
|
|
static void
|
|
gimp_image_rec_remove_layer_stack_dups (GimpImage *image,
|
|
GSList *start)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GSList *dup_layers;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (start == NULL || start->next == NULL)
|
|
return;
|
|
|
|
while ((dup_layers = g_slist_find_custom (start->next, start->data,
|
|
(GCompareFunc) gimp_image_layer_stack_cmp)))
|
|
{
|
|
g_list_free (dup_layers->data);
|
|
/* We can safely remove the duplicate then search again because we
|
|
* know that @start is never removed as we search after it.
|
|
*/
|
|
private->layer_stack = g_slist_delete_link (private->layer_stack,
|
|
dup_layers);
|
|
}
|
|
|
|
gimp_image_rec_remove_layer_stack_dups (image, start->next);
|
|
}
|
|
|
|
/**
|
|
* gimp_image_clean_layer_stack:
|
|
* @image:
|
|
*
|
|
* Remove any duplicate and empty selections in the layer stack.
|
|
*/
|
|
static void
|
|
gimp_image_clean_layer_stack (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* First remove all empty layer lists. */
|
|
private->layer_stack = g_slist_remove_all (private->layer_stack, NULL);
|
|
/* Then remove all duplicates. */
|
|
gimp_image_rec_remove_layer_stack_dups (image, private->layer_stack);
|
|
}
|
|
|
|
static void
|
|
gimp_image_remove_from_layer_stack (GimpImage *image,
|
|
GimpLayer *layer)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GSList *slist;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* Remove layer itself from the MRU layer stack. */
|
|
for (slist = private->layer_stack; slist; slist = slist->next)
|
|
slist->data = g_list_remove (slist->data, layer);
|
|
|
|
/* Also remove all children of a group layer from the layer_stack */
|
|
if (gimp_viewable_get_children (GIMP_VIEWABLE (layer)))
|
|
{
|
|
GimpContainer *stack = gimp_viewable_get_children (GIMP_VIEWABLE (layer));
|
|
GList *children;
|
|
GList *list;
|
|
|
|
children = gimp_item_stack_get_item_list (GIMP_ITEM_STACK (stack));
|
|
|
|
for (list = children; list; list = g_list_next (list))
|
|
{
|
|
GimpLayer *child = list->data;
|
|
|
|
for (slist = private->layer_stack; slist; slist = slist->next)
|
|
slist->data = g_list_remove (slist->data, child);
|
|
}
|
|
|
|
g_list_free (children);
|
|
}
|
|
|
|
gimp_image_clean_layer_stack (image);
|
|
}
|
|
|
|
static gint
|
|
gimp_image_selected_is_descendant (GimpViewable *selected,
|
|
GimpViewable *viewable)
|
|
{
|
|
/* Used as a GCompareFunc to g_list_find_custom() in order to know if
|
|
* one of the selected items is a descendant to @viewable.
|
|
*/
|
|
if (gimp_viewable_is_ancestor (viewable, selected))
|
|
return 0;
|
|
else
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* public functions */
|
|
|
|
GimpImage *
|
|
gimp_image_new (Gimp *gimp,
|
|
gint width,
|
|
gint height,
|
|
GimpImageBaseType base_type,
|
|
GimpPrecision precision)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_GIMP (gimp), NULL);
|
|
g_return_val_if_fail (gimp_babl_is_valid (base_type, precision), NULL);
|
|
|
|
return g_object_new (GIMP_TYPE_IMAGE,
|
|
"gimp", gimp,
|
|
"width", width,
|
|
"height", height,
|
|
"base-type", base_type,
|
|
"precision", precision,
|
|
NULL);
|
|
}
|
|
|
|
gint64
|
|
gimp_image_estimate_memsize (GimpImage *image,
|
|
GimpComponentType component_type,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GList *drawables;
|
|
GList *list;
|
|
gint current_width;
|
|
gint current_height;
|
|
gint64 current_size;
|
|
gint64 scalable_size = 0;
|
|
gint64 scaled_size = 0;
|
|
gint64 new_size;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
current_width = gimp_image_get_width (image);
|
|
current_height = gimp_image_get_height (image);
|
|
current_size = gimp_object_get_memsize (GIMP_OBJECT (image), NULL);
|
|
|
|
/* the part of the image's memsize that scales linearly with the image */
|
|
drawables = gimp_image_item_list_get_list (image,
|
|
GIMP_ITEM_TYPE_LAYERS |
|
|
GIMP_ITEM_TYPE_CHANNELS,
|
|
GIMP_ITEM_SET_ALL);
|
|
|
|
gimp_image_item_list_filter (drawables);
|
|
|
|
drawables = g_list_prepend (drawables, gimp_image_get_mask (image));
|
|
|
|
for (list = drawables; list; list = g_list_next (list))
|
|
{
|
|
GimpDrawable *drawable = list->data;
|
|
gdouble drawable_width;
|
|
gdouble drawable_height;
|
|
|
|
drawable_width = gimp_item_get_width (GIMP_ITEM (drawable));
|
|
drawable_height = gimp_item_get_height (GIMP_ITEM (drawable));
|
|
|
|
scalable_size += gimp_drawable_estimate_memsize (drawable,
|
|
gimp_drawable_get_component_type (drawable),
|
|
drawable_width,
|
|
drawable_height);
|
|
|
|
scaled_size += gimp_drawable_estimate_memsize (drawable,
|
|
component_type,
|
|
drawable_width * width /
|
|
current_width,
|
|
drawable_height * height /
|
|
current_height);
|
|
}
|
|
|
|
g_list_free (drawables);
|
|
|
|
scalable_size +=
|
|
gimp_projection_estimate_memsize (gimp_image_get_base_type (image),
|
|
gimp_image_get_component_type (image),
|
|
gimp_image_get_width (image),
|
|
gimp_image_get_height (image));
|
|
|
|
scaled_size +=
|
|
gimp_projection_estimate_memsize (gimp_image_get_base_type (image),
|
|
component_type,
|
|
width, height);
|
|
|
|
GIMP_LOG (IMAGE_SCALE,
|
|
"scalable_size = %"G_GINT64_FORMAT" scaled_size = %"G_GINT64_FORMAT,
|
|
scalable_size, scaled_size);
|
|
|
|
new_size = current_size - scalable_size + scaled_size;
|
|
|
|
return new_size;
|
|
}
|
|
|
|
GimpImageBaseType
|
|
gimp_image_get_base_type (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), -1);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->base_type;
|
|
}
|
|
|
|
GimpComponentType
|
|
gimp_image_get_component_type (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), -1);
|
|
|
|
return gimp_babl_component_type (GIMP_IMAGE_GET_PRIVATE (image)->precision);
|
|
}
|
|
|
|
GimpPrecision
|
|
gimp_image_get_precision (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), -1);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->precision;
|
|
}
|
|
|
|
const Babl *
|
|
gimp_image_get_format (GimpImage *image,
|
|
GimpImageBaseType base_type,
|
|
GimpPrecision precision,
|
|
gboolean with_alpha,
|
|
const Babl *space)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
switch (base_type)
|
|
{
|
|
case GIMP_RGB:
|
|
case GIMP_GRAY:
|
|
return gimp_babl_format (base_type, precision, with_alpha, space);
|
|
|
|
case GIMP_INDEXED:
|
|
if (precision == GIMP_PRECISION_U8_NON_LINEAR)
|
|
{
|
|
if (with_alpha)
|
|
return gimp_image_colormap_get_rgba_format (image);
|
|
else
|
|
return gimp_image_colormap_get_rgb_format (image);
|
|
}
|
|
}
|
|
|
|
g_return_val_if_reached (NULL);
|
|
}
|
|
|
|
const Babl *
|
|
gimp_image_get_layer_space (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->layer_space;
|
|
}
|
|
|
|
const Babl *
|
|
gimp_image_get_layer_format (GimpImage *image,
|
|
gboolean with_alpha)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return gimp_image_get_format (image,
|
|
gimp_image_get_base_type (image),
|
|
gimp_image_get_precision (image),
|
|
with_alpha,
|
|
gimp_image_get_layer_space (image));
|
|
}
|
|
|
|
const Babl *
|
|
gimp_image_get_channel_format (GimpImage *image)
|
|
{
|
|
GimpPrecision precision;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
precision = gimp_image_get_precision (image);
|
|
|
|
if (precision == GIMP_PRECISION_U8_NON_LINEAR)
|
|
return gimp_image_get_format (image, GIMP_GRAY,
|
|
gimp_image_get_precision (image),
|
|
FALSE, NULL);
|
|
|
|
return gimp_babl_mask_format (precision);
|
|
}
|
|
|
|
const Babl *
|
|
gimp_image_get_mask_format (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return gimp_babl_mask_format (gimp_image_get_precision (image));
|
|
}
|
|
|
|
GimpLayerMode
|
|
gimp_image_get_default_new_layer_mode (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), GIMP_LAYER_MODE_NORMAL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (private->new_layer_mode == -1)
|
|
{
|
|
GList *layers = gimp_image_get_layer_list (image);
|
|
|
|
if (layers)
|
|
{
|
|
GList *list;
|
|
|
|
for (list = layers; list; list = g_list_next (list))
|
|
{
|
|
GimpLayer *layer = list->data;
|
|
GimpLayerMode mode = gimp_layer_get_mode (layer);
|
|
|
|
if (! gimp_layer_mode_is_legacy (mode))
|
|
{
|
|
/* any non-legacy layer switches the mode to non-legacy
|
|
*/
|
|
private->new_layer_mode = GIMP_LAYER_MODE_NORMAL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* only if all layers are legacy, the mode is also legacy
|
|
*/
|
|
if (! list)
|
|
private->new_layer_mode = GIMP_LAYER_MODE_NORMAL_LEGACY;
|
|
|
|
g_list_free (layers);
|
|
}
|
|
else
|
|
{
|
|
/* empty images are never considered legacy
|
|
*/
|
|
private->new_layer_mode = GIMP_LAYER_MODE_NORMAL;
|
|
}
|
|
}
|
|
|
|
return private->new_layer_mode;
|
|
}
|
|
|
|
void
|
|
gimp_image_unset_default_new_layer_mode (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->new_layer_mode = -1;
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_id (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), -1);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->ID;
|
|
}
|
|
|
|
GimpImage *
|
|
gimp_image_get_by_id (Gimp *gimp,
|
|
gint image_id)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_GIMP (gimp), NULL);
|
|
|
|
if (gimp->image_table == NULL)
|
|
return NULL;
|
|
|
|
return (GimpImage *) gimp_id_table_lookup (gimp->image_table, image_id);
|
|
}
|
|
|
|
void
|
|
gimp_image_set_file (GimpImage *image,
|
|
GFile *file)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (file == NULL || G_IS_FILE (file));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (private->file != file)
|
|
{
|
|
gimp_object_take_name (GIMP_OBJECT (image),
|
|
file ? g_file_get_uri (file) : NULL);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_image_get_untitled_file:
|
|
*
|
|
* Returns: A #GFile saying "Untitled" for newly created images.
|
|
**/
|
|
GFile *
|
|
gimp_image_get_untitled_file (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (! private->untitled_file)
|
|
private->untitled_file = g_file_new_for_uri (_("Untitled"));
|
|
|
|
return private->untitled_file;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_get_file_or_untitled:
|
|
* @image: A #GimpImage.
|
|
*
|
|
* Get the file of the XCF image, or the "Untitled" file if there is no file.
|
|
*
|
|
* Returns: A #GFile.
|
|
**/
|
|
GFile *
|
|
gimp_image_get_file_or_untitled (GimpImage *image)
|
|
{
|
|
GFile *file;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
file = gimp_image_get_file (image);
|
|
|
|
if (! file)
|
|
file = gimp_image_get_untitled_file (image);
|
|
|
|
return file;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_get_file:
|
|
* @image: A #GimpImage.
|
|
*
|
|
* Get the file of the XCF image, or %NULL if there is no file.
|
|
*
|
|
* Returns: (nullable): The file, or %NULL.
|
|
**/
|
|
GFile *
|
|
gimp_image_get_file (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->file;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_get_imported_file:
|
|
* @image: A #GimpImage.
|
|
*
|
|
* Returns: (nullable): The file of the imported image, or %NULL if the image
|
|
* has been saved as XCF after it was imported.
|
|
**/
|
|
GFile *
|
|
gimp_image_get_imported_file (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->imported_file;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_get_exported_file:
|
|
* @image: A #GimpImage.
|
|
*
|
|
* Returns: (nullable): The file of the image last exported from this XCF file,
|
|
* or %NULL if the image has never been exported.
|
|
**/
|
|
GFile *
|
|
gimp_image_get_exported_file (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->exported_file;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_get_save_a_copy_file:
|
|
* @image: A #GimpImage.
|
|
*
|
|
* Returns: The URI of the last copy that was saved of this XCF file.
|
|
**/
|
|
GFile *
|
|
gimp_image_get_save_a_copy_file (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->save_a_copy_file;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_get_any_file:
|
|
* @image: A #GimpImage.
|
|
*
|
|
* Returns: The XCF file, the imported file, or the exported file, in
|
|
* that order of precedence.
|
|
**/
|
|
GFile *
|
|
gimp_image_get_any_file (GimpImage *image)
|
|
{
|
|
GFile *file;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
file = gimp_image_get_file (image);
|
|
if (! file)
|
|
{
|
|
file = gimp_image_get_imported_file (image);
|
|
if (! file)
|
|
{
|
|
file = gimp_image_get_exported_file (image);
|
|
}
|
|
}
|
|
|
|
return file;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_set_imported_uri:
|
|
* @image: A #GimpImage.
|
|
* @file:
|
|
*
|
|
* Sets the URI this file was imported from.
|
|
**/
|
|
void
|
|
gimp_image_set_imported_file (GimpImage *image,
|
|
GFile *file)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (file == NULL || G_IS_FILE (file));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (g_set_object (&private->imported_file, file))
|
|
{
|
|
gimp_object_name_changed (GIMP_OBJECT (image));
|
|
}
|
|
|
|
if (! private->resolution_set && file != NULL)
|
|
{
|
|
/* Unlike new files (which follow technological progress and will
|
|
* use higher default resolution, or explicitly chosen templates),
|
|
* imported files have a more backward-compatible value.
|
|
*
|
|
* 72 PPI is traditionally the default value when none other had
|
|
* been explicitly set (for instance it is the default when no
|
|
* resolution metadata was set in Exif version 2.32, and below,
|
|
* standard). This historical value will only ever apply to loaded
|
|
* images. New images will continue having more modern or
|
|
* templated defaults.
|
|
*/
|
|
private->xresolution = 72.0;
|
|
private->yresolution = 72.0;
|
|
private->resolution_unit = GIMP_UNIT_INCH;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_image_set_exported_file:
|
|
* @image: A #GimpImage.
|
|
* @file:
|
|
*
|
|
* Sets the file this image was last exported to. Note that saving as
|
|
* XCF is not "exporting".
|
|
**/
|
|
void
|
|
gimp_image_set_exported_file (GimpImage *image,
|
|
GFile *file)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (file == NULL || G_IS_FILE (file));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (g_set_object (&private->exported_file, file))
|
|
{
|
|
gimp_object_name_changed (GIMP_OBJECT (image));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_image_set_save_a_copy_file:
|
|
* @image: A #GimpImage.
|
|
* @uri:
|
|
*
|
|
* Set the URI to the last copy this XCF file was saved to through the
|
|
* "save a copy" action.
|
|
**/
|
|
void
|
|
gimp_image_set_save_a_copy_file (GimpImage *image,
|
|
GFile *file)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (file == NULL || G_IS_FILE (file));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
g_set_object (&private->save_a_copy_file, file);
|
|
}
|
|
|
|
static gchar *
|
|
gimp_image_format_display_uri (GimpImage *image,
|
|
gboolean basename)
|
|
{
|
|
const gchar *uri_format = NULL;
|
|
const gchar *export_status = NULL;
|
|
GFile *file = NULL;
|
|
GFile *source = NULL;
|
|
GFile *dest = NULL;
|
|
GFile *display_file = NULL;
|
|
gboolean is_imported;
|
|
gboolean is_exported;
|
|
gchar *display_uri = NULL;
|
|
gchar *format_string;
|
|
gchar *tmp;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
file = gimp_image_get_file (image);
|
|
source = gimp_image_get_imported_file (image);
|
|
dest = gimp_image_get_exported_file (image);
|
|
|
|
is_imported = (source != NULL);
|
|
is_exported = (dest != NULL);
|
|
|
|
if (file)
|
|
{
|
|
display_file = g_object_ref (file);
|
|
uri_format = "%s";
|
|
}
|
|
else
|
|
{
|
|
if (is_imported)
|
|
display_file = source;
|
|
|
|
/* Calculate filename suffix */
|
|
if (! gimp_image_is_export_dirty (image))
|
|
{
|
|
if (is_exported)
|
|
{
|
|
display_file = dest;
|
|
export_status = _(" (exported)");
|
|
}
|
|
else if (is_imported)
|
|
{
|
|
export_status = _(" (overwritten)");
|
|
}
|
|
else
|
|
{
|
|
g_warning ("Unexpected code path, Save+export implementation is buggy!");
|
|
}
|
|
}
|
|
else if (is_imported)
|
|
{
|
|
export_status = _(" (imported)");
|
|
}
|
|
|
|
if (display_file)
|
|
display_file = gimp_file_with_new_extension (display_file, NULL);
|
|
|
|
uri_format = "[%s]";
|
|
}
|
|
|
|
if (! display_file)
|
|
display_file = g_object_ref (gimp_image_get_untitled_file (image));
|
|
|
|
if (basename)
|
|
display_uri = g_path_get_basename (gimp_file_get_utf8_name (display_file));
|
|
else
|
|
display_uri = g_strdup (gimp_file_get_utf8_name (display_file));
|
|
|
|
g_object_unref (display_file);
|
|
|
|
format_string = g_strconcat (uri_format, export_status, NULL);
|
|
|
|
tmp = g_strdup_printf (format_string, display_uri);
|
|
g_free (display_uri);
|
|
display_uri = tmp;
|
|
|
|
g_free (format_string);
|
|
|
|
return display_uri;
|
|
}
|
|
|
|
const gchar *
|
|
gimp_image_get_display_name (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (! private->display_name)
|
|
private->display_name = gimp_image_format_display_uri (image, TRUE);
|
|
|
|
return private->display_name;
|
|
}
|
|
|
|
const gchar *
|
|
gimp_image_get_display_path (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (! private->display_path)
|
|
private->display_path = gimp_image_format_display_uri (image, FALSE);
|
|
|
|
return private->display_path;
|
|
}
|
|
|
|
void
|
|
gimp_image_set_load_proc (GimpImage *image,
|
|
GimpPlugInProcedure *proc)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->load_proc = proc;
|
|
}
|
|
|
|
GimpPlugInProcedure *
|
|
gimp_image_get_load_proc (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->load_proc;
|
|
}
|
|
|
|
void
|
|
gimp_image_set_save_proc (GimpImage *image,
|
|
GimpPlugInProcedure *proc)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->save_proc = proc;
|
|
}
|
|
|
|
GimpPlugInProcedure *
|
|
gimp_image_get_save_proc (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->save_proc;
|
|
}
|
|
|
|
void
|
|
gimp_image_set_export_proc (GimpImage *image,
|
|
GimpPlugInProcedure *proc)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->export_proc = proc;
|
|
}
|
|
|
|
GimpPlugInProcedure *
|
|
gimp_image_get_export_proc (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->export_proc;
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_xcf_version (GimpImage *image,
|
|
gboolean zlib_compression,
|
|
gint *gimp_version,
|
|
const gchar **version_string,
|
|
gchar **version_reason)
|
|
{
|
|
GList *items;
|
|
GList *list;
|
|
GList *reasons = NULL;
|
|
gint version = 0; /* default to oldest */
|
|
const gchar *enum_desc;
|
|
|
|
#define ADD_REASON(_reason) \
|
|
if (version_reason) { \
|
|
gchar *tmp = _reason; \
|
|
if (g_list_find_custom (reasons, tmp, (GCompareFunc) strcmp)) \
|
|
g_free (tmp); \
|
|
else \
|
|
reasons = g_list_prepend (reasons, tmp); }
|
|
|
|
/* need version 1 for colormaps */
|
|
if (gimp_image_get_colormap_palette (image))
|
|
version = 1;
|
|
|
|
items = gimp_image_get_layer_list (image);
|
|
for (list = items; list; list = g_list_next (list))
|
|
{
|
|
GimpLayer *layer = GIMP_LAYER (list->data);
|
|
|
|
switch (gimp_layer_get_mode (layer))
|
|
{
|
|
/* Modes that exist since ancient times */
|
|
case GIMP_LAYER_MODE_NORMAL_LEGACY:
|
|
case GIMP_LAYER_MODE_DISSOLVE:
|
|
case GIMP_LAYER_MODE_BEHIND_LEGACY:
|
|
case GIMP_LAYER_MODE_MULTIPLY_LEGACY:
|
|
case GIMP_LAYER_MODE_SCREEN_LEGACY:
|
|
case GIMP_LAYER_MODE_OVERLAY_LEGACY:
|
|
case GIMP_LAYER_MODE_DIFFERENCE_LEGACY:
|
|
case GIMP_LAYER_MODE_ADDITION_LEGACY:
|
|
case GIMP_LAYER_MODE_SUBTRACT_LEGACY:
|
|
case GIMP_LAYER_MODE_DARKEN_ONLY_LEGACY:
|
|
case GIMP_LAYER_MODE_LIGHTEN_ONLY_LEGACY:
|
|
case GIMP_LAYER_MODE_HSV_HUE_LEGACY:
|
|
case GIMP_LAYER_MODE_HSV_SATURATION_LEGACY:
|
|
case GIMP_LAYER_MODE_HSL_COLOR_LEGACY:
|
|
case GIMP_LAYER_MODE_HSV_VALUE_LEGACY:
|
|
case GIMP_LAYER_MODE_DIVIDE_LEGACY:
|
|
case GIMP_LAYER_MODE_DODGE_LEGACY:
|
|
case GIMP_LAYER_MODE_BURN_LEGACY:
|
|
case GIMP_LAYER_MODE_HARDLIGHT_LEGACY:
|
|
break;
|
|
|
|
/* Since 2.6 */
|
|
case GIMP_LAYER_MODE_SOFTLIGHT_LEGACY:
|
|
case GIMP_LAYER_MODE_GRAIN_EXTRACT_LEGACY:
|
|
case GIMP_LAYER_MODE_GRAIN_MERGE_LEGACY:
|
|
case GIMP_LAYER_MODE_COLOR_ERASE_LEGACY:
|
|
gimp_enum_get_value (GIMP_TYPE_LAYER_MODE,
|
|
gimp_layer_get_mode (layer),
|
|
NULL, NULL, &enum_desc, NULL);
|
|
ADD_REASON (g_strdup_printf (_("Layer mode '%s' was added in %s"),
|
|
enum_desc, "GIMP 2.6"));
|
|
version = MAX (2, version);
|
|
break;
|
|
|
|
/* Since 2.10 */
|
|
case GIMP_LAYER_MODE_OVERLAY:
|
|
case GIMP_LAYER_MODE_LCH_HUE:
|
|
case GIMP_LAYER_MODE_LCH_CHROMA:
|
|
case GIMP_LAYER_MODE_LCH_COLOR:
|
|
case GIMP_LAYER_MODE_LCH_LIGHTNESS:
|
|
gimp_enum_get_value (GIMP_TYPE_LAYER_MODE,
|
|
gimp_layer_get_mode (layer),
|
|
NULL, NULL, &enum_desc, NULL);
|
|
ADD_REASON (g_strdup_printf (_("Layer mode '%s' was added in %s"),
|
|
enum_desc, "GIMP 2.10"));
|
|
version = MAX (9, version);
|
|
break;
|
|
|
|
/* Since 2.10 */
|
|
case GIMP_LAYER_MODE_NORMAL:
|
|
case GIMP_LAYER_MODE_BEHIND:
|
|
case GIMP_LAYER_MODE_MULTIPLY:
|
|
case GIMP_LAYER_MODE_SCREEN:
|
|
case GIMP_LAYER_MODE_DIFFERENCE:
|
|
case GIMP_LAYER_MODE_ADDITION:
|
|
case GIMP_LAYER_MODE_SUBTRACT:
|
|
case GIMP_LAYER_MODE_DARKEN_ONLY:
|
|
case GIMP_LAYER_MODE_LIGHTEN_ONLY:
|
|
case GIMP_LAYER_MODE_HSV_HUE:
|
|
case GIMP_LAYER_MODE_HSV_SATURATION:
|
|
case GIMP_LAYER_MODE_HSL_COLOR:
|
|
case GIMP_LAYER_MODE_HSV_VALUE:
|
|
case GIMP_LAYER_MODE_DIVIDE:
|
|
case GIMP_LAYER_MODE_DODGE:
|
|
case GIMP_LAYER_MODE_BURN:
|
|
case GIMP_LAYER_MODE_HARDLIGHT:
|
|
case GIMP_LAYER_MODE_SOFTLIGHT:
|
|
case GIMP_LAYER_MODE_GRAIN_EXTRACT:
|
|
case GIMP_LAYER_MODE_GRAIN_MERGE:
|
|
case GIMP_LAYER_MODE_VIVID_LIGHT:
|
|
case GIMP_LAYER_MODE_PIN_LIGHT:
|
|
case GIMP_LAYER_MODE_LINEAR_LIGHT:
|
|
case GIMP_LAYER_MODE_HARD_MIX:
|
|
case GIMP_LAYER_MODE_EXCLUSION:
|
|
case GIMP_LAYER_MODE_LINEAR_BURN:
|
|
case GIMP_LAYER_MODE_LUMA_DARKEN_ONLY:
|
|
case GIMP_LAYER_MODE_LUMA_LIGHTEN_ONLY:
|
|
case GIMP_LAYER_MODE_LUMINANCE:
|
|
case GIMP_LAYER_MODE_COLOR_ERASE:
|
|
case GIMP_LAYER_MODE_ERASE:
|
|
case GIMP_LAYER_MODE_MERGE:
|
|
case GIMP_LAYER_MODE_SPLIT:
|
|
case GIMP_LAYER_MODE_PASS_THROUGH:
|
|
gimp_enum_get_value (GIMP_TYPE_LAYER_MODE,
|
|
gimp_layer_get_mode (layer),
|
|
NULL, NULL, &enum_desc, NULL);
|
|
ADD_REASON (g_strdup_printf (_("Layer mode '%s' was added in %s"),
|
|
enum_desc, "GIMP 2.10"));
|
|
version = MAX (10, version);
|
|
break;
|
|
|
|
/* Just here instead of default so we get compiler warnings */
|
|
case GIMP_LAYER_MODE_REPLACE:
|
|
case GIMP_LAYER_MODE_ANTI_ERASE:
|
|
case GIMP_LAYER_MODE_SEPARATOR:
|
|
break;
|
|
}
|
|
|
|
/* need version 3 for layer trees */
|
|
if (gimp_viewable_get_children (GIMP_VIEWABLE (layer)))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Layer groups were added in %s"),
|
|
"GIMP 2.8"));
|
|
version = MAX (3, version);
|
|
|
|
/* need version 13 for group layers with masks */
|
|
if (gimp_layer_get_mask (layer))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Masks on layer groups were "
|
|
"added in %s"), "GIMP 2.10"));
|
|
version = MAX (13, version);
|
|
}
|
|
|
|
if (gimp_item_get_lock_position (GIMP_ITEM (layer)))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Position locks on layer groups were added in %s"),
|
|
"GIMP 3.0"));
|
|
version = MAX (17, version);
|
|
}
|
|
|
|
if (gimp_layer_get_lock_alpha (layer))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Alpha channel locks on layer groups were added in %s"),
|
|
"GIMP 3.0"));
|
|
version = MAX (17, version);
|
|
}
|
|
}
|
|
|
|
if (gimp_item_get_lock_visibility (GIMP_ITEM (layer)))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Visibility locks were added in %s"),
|
|
"GIMP 3.0"));
|
|
version = MAX (17, version);
|
|
}
|
|
}
|
|
g_list_free (items);
|
|
|
|
items = gimp_image_get_channel_list (image);
|
|
for (list = items; list; list = g_list_next (list))
|
|
{
|
|
GimpChannel *channel = GIMP_CHANNEL (list->data);
|
|
|
|
if (gimp_item_get_lock_visibility (GIMP_ITEM (channel)))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Visibility locks were added in %s"),
|
|
"GIMP 3.0"));
|
|
version = MAX (17, version);
|
|
}
|
|
}
|
|
g_list_free (items);
|
|
|
|
if (g_list_length (gimp_image_get_selected_vectors (image)) > 1)
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Multiple path selection was "
|
|
"added in %s"), "GIMP 3.0.0"));
|
|
version = MAX (18, version);
|
|
}
|
|
|
|
items = gimp_image_get_vectors_list (image);
|
|
for (list = items; list; list = g_list_next (list))
|
|
{
|
|
GimpVectors *vectors = GIMP_VECTORS (list->data);
|
|
|
|
if (gimp_item_get_color_tag (GIMP_ITEM (vectors)) != GIMP_COLOR_TAG_NONE)
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Storing color tags in path was "
|
|
"added in %s"), "GIMP 3.0.0"));
|
|
version = MAX (18, version);
|
|
}
|
|
if (gimp_item_get_lock_content (GIMP_ITEM (list->data)) ||
|
|
gimp_item_get_lock_position (GIMP_ITEM (list->data)))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Storing locks in path was "
|
|
"added in %s"), "GIMP 3.0.0"));
|
|
version = MAX (18, version);
|
|
}
|
|
}
|
|
g_list_free (items);
|
|
|
|
/* version 6 for new metadata has been dropped since they are
|
|
* saved through parasites, which is compatible with older versions.
|
|
*/
|
|
|
|
/* need version 7 for != 8-bit gamma images */
|
|
if (gimp_image_get_precision (image) != GIMP_PRECISION_U8_NON_LINEAR)
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("High bit-depth images were added "
|
|
"in %s"), "GIMP 2.10"));
|
|
version = MAX (7, version);
|
|
}
|
|
|
|
/* need version 12 for > 8-bit images for proper endian swapping */
|
|
if (gimp_image_get_component_type (image) > GIMP_COMPONENT_TYPE_U8)
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Encoding of high bit-depth images was "
|
|
"fixed in %s"), "GIMP 2.10"));
|
|
version = MAX (12, version);
|
|
}
|
|
|
|
/* need version 8 for zlib compression */
|
|
if (zlib_compression)
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Internal zlib compression was "
|
|
"added in %s"), "GIMP 2.10"));
|
|
version = MAX (8, version);
|
|
}
|
|
|
|
/* if version is 10 (lots of new layer modes), go to version 11 with
|
|
* 64 bit offsets right away
|
|
*/
|
|
if (version == 10)
|
|
version = 11;
|
|
|
|
/* use the image's in-memory size as an upper bound to estimate the
|
|
* need for 64 bit file offsets inside the XCF, this is a *very*
|
|
* conservative estimate and should never fail
|
|
*/
|
|
if (gimp_object_get_memsize (GIMP_OBJECT (image), NULL) >= ((gint64) 1 << 32))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Support for image files larger than "
|
|
"4GB was added in %s"), "GIMP 2.10"));
|
|
version = MAX (11, version);
|
|
}
|
|
|
|
if (g_list_length (gimp_image_get_selected_layers (image)) > 1)
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Multiple layer selection was "
|
|
"added in %s"), "GIMP 3.0.0"));
|
|
version = MAX (14, version);
|
|
}
|
|
|
|
if ((list = gimp_image_get_guides (image)))
|
|
{
|
|
for (; list; list = g_list_next (list))
|
|
{
|
|
gint32 position = gimp_guide_get_position (list->data);
|
|
|
|
if (position < 0 ||
|
|
(gimp_guide_get_orientation (list->data) == GIMP_ORIENTATION_HORIZONTAL &&
|
|
position > gimp_image_get_height (image)) ||
|
|
(gimp_guide_get_orientation (list->data) == GIMP_ORIENTATION_VERTICAL &&
|
|
position > gimp_image_get_width (image)))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Off-canvas guides "
|
|
"added in %s"), "GIMP 3.0.0"));
|
|
version = MAX (15, version);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (gimp_image_get_stored_item_sets (image, GIMP_TYPE_LAYER) ||
|
|
gimp_image_get_stored_item_sets (image, GIMP_TYPE_CHANNEL))
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Item set and pattern search in item's name were "
|
|
"added in %s"), "GIMP 3.0.0"));
|
|
version = MAX (16, version);
|
|
}
|
|
if (g_list_length (gimp_image_get_selected_channels (image)) > 1)
|
|
{
|
|
ADD_REASON (g_strdup_printf (_("Multiple channel selection was "
|
|
"added in %s"), "GIMP 3.0.0"));
|
|
version = MAX (16, version);
|
|
}
|
|
|
|
|
|
#undef ADD_REASON
|
|
|
|
switch (version)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
case 2:
|
|
if (gimp_version) *gimp_version = 206;
|
|
if (version_string) *version_string = "GIMP 2.6";
|
|
break;
|
|
|
|
case 3:
|
|
if (gimp_version) *gimp_version = 208;
|
|
if (version_string) *version_string = "GIMP 2.8";
|
|
break;
|
|
|
|
case 4:
|
|
case 5:
|
|
case 6:
|
|
case 7:
|
|
case 8:
|
|
case 9:
|
|
case 10:
|
|
case 11:
|
|
case 12:
|
|
case 13:
|
|
if (gimp_version) *gimp_version = 210;
|
|
if (version_string) *version_string = "GIMP 2.10";
|
|
break;
|
|
case 14:
|
|
case 15:
|
|
case 16:
|
|
case 17:
|
|
case 18:
|
|
if (gimp_version) *gimp_version = 300;
|
|
if (version_string) *version_string = "GIMP 3.0";
|
|
break;
|
|
}
|
|
|
|
if (version_reason && reasons)
|
|
{
|
|
GString *reason = g_string_new (NULL);
|
|
|
|
reasons = g_list_sort (reasons, (GCompareFunc) strcmp);
|
|
|
|
for (list = reasons; list; list = g_list_next (list))
|
|
{
|
|
g_string_append (reason, list->data);
|
|
if (g_list_next (list))
|
|
g_string_append_c (reason, '\n');
|
|
}
|
|
|
|
*version_reason = g_string_free (reason, FALSE);
|
|
}
|
|
if (reasons)
|
|
g_list_free_full (reasons, g_free);
|
|
|
|
return version;
|
|
}
|
|
|
|
void
|
|
gimp_image_set_xcf_compression (GimpImage *image,
|
|
gboolean compression)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->xcf_compression = compression;
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_get_xcf_compression (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->xcf_compression;
|
|
}
|
|
|
|
void
|
|
gimp_image_set_resolution (GimpImage *image,
|
|
gdouble xresolution,
|
|
gdouble yresolution)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* don't allow to set the resolution out of bounds */
|
|
if (xresolution < GIMP_MIN_RESOLUTION || xresolution > GIMP_MAX_RESOLUTION ||
|
|
yresolution < GIMP_MIN_RESOLUTION || yresolution > GIMP_MAX_RESOLUTION)
|
|
return;
|
|
|
|
private->resolution_set = TRUE;
|
|
|
|
if ((ABS (private->xresolution - xresolution) >= 1e-5) ||
|
|
(ABS (private->yresolution - yresolution) >= 1e-5))
|
|
{
|
|
gimp_image_undo_push_image_resolution (image,
|
|
C_("undo-type", "Change Image Resolution"));
|
|
|
|
private->xresolution = xresolution;
|
|
private->yresolution = yresolution;
|
|
|
|
gimp_image_resolution_changed (image);
|
|
gimp_image_size_changed_detailed (image,
|
|
0,
|
|
0,
|
|
gimp_image_get_width (image),
|
|
gimp_image_get_height (image));
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_image_get_resolution (GimpImage *image,
|
|
gdouble *xresolution,
|
|
gdouble *yresolution)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (xresolution != NULL && yresolution != NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
*xresolution = private->xresolution;
|
|
*yresolution = private->yresolution;
|
|
}
|
|
|
|
void
|
|
gimp_image_resolution_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[RESOLUTION_CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
gimp_image_set_unit (GimpImage *image,
|
|
GimpUnit unit)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (unit > GIMP_UNIT_PIXEL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (private->resolution_unit != unit)
|
|
{
|
|
gimp_image_undo_push_image_resolution (image,
|
|
C_("undo-type", "Change Image Unit"));
|
|
|
|
private->resolution_unit = unit;
|
|
gimp_image_unit_changed (image);
|
|
}
|
|
}
|
|
|
|
GimpUnit
|
|
gimp_image_get_unit (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), GIMP_UNIT_INCH);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->resolution_unit;
|
|
}
|
|
|
|
void
|
|
gimp_image_unit_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[UNIT_CHANGED], 0);
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_width (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->width;
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_height (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->height;
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_has_alpha (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GimpLayer *layer;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), TRUE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
layer = GIMP_LAYER (gimp_container_get_first_child (private->layers->container));
|
|
|
|
return ((gimp_image_get_n_layers (image) > 1) ||
|
|
(layer && gimp_drawable_has_alpha (GIMP_DRAWABLE (layer))));
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_is_empty (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), TRUE);
|
|
|
|
return gimp_container_is_empty (GIMP_IMAGE_GET_PRIVATE (image)->layers->container);
|
|
}
|
|
|
|
void
|
|
gimp_image_set_floating_selection (GimpImage *image,
|
|
GimpLayer *floating_sel)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (floating_sel == NULL || GIMP_IS_LAYER (floating_sel));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (private->floating_sel != floating_sel)
|
|
{
|
|
private->floating_sel = floating_sel;
|
|
|
|
private->flush_accum.floating_selection_changed = TRUE;
|
|
}
|
|
}
|
|
|
|
GimpLayer *
|
|
gimp_image_get_floating_selection (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->floating_sel;
|
|
}
|
|
|
|
void
|
|
gimp_image_floating_selection_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[FLOATING_SELECTION_CHANGED], 0);
|
|
}
|
|
|
|
GimpChannel *
|
|
gimp_image_get_mask (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->selection_mask;
|
|
}
|
|
|
|
void
|
|
gimp_image_mask_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[MASK_CHANGED], 0);
|
|
}
|
|
|
|
/**
|
|
* gimp_item_mask_intersect:
|
|
* @image: the #GimpImage
|
|
* @items: a list of #GimpItem
|
|
* @x: (out) (optional): return location for x
|
|
* @y: (out) (optional): return location for y
|
|
* @width: (out) (optional): return location for the width
|
|
* @height: (out) (optional): return location for the height
|
|
*
|
|
* Intersect the area of the @items and its image's selection mask.
|
|
* The computed area is the bounding box of the selection intersection
|
|
* within the image. These values are only returned if the function
|
|
* returns %TRUE.
|
|
*
|
|
* Note that even though the items bounding box may not be empty, it is
|
|
* possible to get a return value of %FALSE. Imagine disjoint items, one
|
|
* on the left, one on the right, and a selection in the middle not
|
|
* intersecting with any of the items.
|
|
*
|
|
* Returns: %TRUE if the selection intersects with any of the @items.
|
|
*/
|
|
gboolean
|
|
gimp_image_mask_intersect (GimpImage *image,
|
|
GList *items,
|
|
gint *x,
|
|
gint *y,
|
|
gint *width,
|
|
gint *height)
|
|
{
|
|
GimpChannel *selection;
|
|
GList *iter;
|
|
gint sel_x, sel_y, sel_width, sel_height;
|
|
gint x1 = G_MAXINT;
|
|
gint y1 = G_MAXINT;
|
|
gint x2 = G_MININT;
|
|
gint y2 = G_MININT;
|
|
gboolean intersect = FALSE;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
for (iter = items; iter; iter = iter->next)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_ITEM (iter->data), FALSE);
|
|
g_return_val_if_fail (gimp_item_is_attached (iter->data), FALSE);
|
|
g_return_val_if_fail (gimp_item_get_image (iter->data) == image, FALSE);
|
|
}
|
|
|
|
selection = gimp_image_get_mask (image);
|
|
if (selection)
|
|
gimp_item_bounds (GIMP_ITEM (selection),
|
|
&sel_x, &sel_y, &sel_width, &sel_height);
|
|
|
|
for (iter = items; iter; iter = iter->next)
|
|
{
|
|
GimpItem *item = iter->data;
|
|
gboolean item_intersect;
|
|
gint tmp_x, tmp_y;
|
|
gint tmp_width, tmp_height;
|
|
|
|
gimp_item_get_offset (item, &tmp_x, &tmp_y);
|
|
|
|
/* check for is_empty() before intersecting so we ignore the
|
|
* selection if it is suspended (like when stroking)
|
|
*/
|
|
if (GIMP_ITEM (selection) != item &&
|
|
! gimp_channel_is_empty (selection))
|
|
{
|
|
item_intersect = gimp_rectangle_intersect (sel_x, sel_y, sel_width, sel_height,
|
|
tmp_x, tmp_y,
|
|
gimp_item_get_width (item),
|
|
gimp_item_get_height (item),
|
|
&tmp_x, &tmp_y,
|
|
&tmp_width, &tmp_height);
|
|
}
|
|
else
|
|
{
|
|
tmp_width = gimp_item_get_width (item);
|
|
tmp_height = gimp_item_get_height (item);
|
|
|
|
item_intersect = TRUE;
|
|
}
|
|
|
|
if (item_intersect)
|
|
{
|
|
x1 = MIN (x1, tmp_x);
|
|
y1 = MIN (y1, tmp_y);
|
|
x2 = MAX (x2, tmp_x + tmp_width);
|
|
y2 = MAX (y2, tmp_y + tmp_height);
|
|
|
|
intersect = TRUE;
|
|
}
|
|
}
|
|
|
|
if (intersect)
|
|
{
|
|
if (x) *x = x1;
|
|
if (y) *y = y1;
|
|
if (width) *width = x2 - x1;
|
|
if (height) *height = y2 - y1;
|
|
}
|
|
|
|
return intersect;
|
|
}
|
|
|
|
void
|
|
gimp_image_take_mask (GimpImage *image,
|
|
GimpChannel *mask)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_SELECTION (mask));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (private->selection_mask)
|
|
g_object_unref (private->selection_mask);
|
|
|
|
private->selection_mask = g_object_ref_sink (mask);
|
|
|
|
g_signal_connect (private->selection_mask, "update",
|
|
G_CALLBACK (gimp_image_mask_update),
|
|
image);
|
|
}
|
|
|
|
|
|
/* image components */
|
|
|
|
const Babl *
|
|
gimp_image_get_component_format (GimpImage *image,
|
|
GimpChannelType channel)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
switch (channel)
|
|
{
|
|
case GIMP_CHANNEL_RED:
|
|
return gimp_babl_component_format (GIMP_RGB,
|
|
gimp_image_get_precision (image),
|
|
RED);
|
|
|
|
case GIMP_CHANNEL_GREEN:
|
|
return gimp_babl_component_format (GIMP_RGB,
|
|
gimp_image_get_precision (image),
|
|
GREEN);
|
|
|
|
case GIMP_CHANNEL_BLUE:
|
|
return gimp_babl_component_format (GIMP_RGB,
|
|
gimp_image_get_precision (image),
|
|
BLUE);
|
|
|
|
case GIMP_CHANNEL_ALPHA:
|
|
return gimp_babl_component_format (GIMP_RGB,
|
|
gimp_image_get_precision (image),
|
|
ALPHA);
|
|
|
|
case GIMP_CHANNEL_GRAY:
|
|
return gimp_babl_component_format (GIMP_GRAY,
|
|
gimp_image_get_precision (image),
|
|
GRAY);
|
|
|
|
case GIMP_CHANNEL_INDEXED:
|
|
return babl_format ("Y u8"); /* will extract grayscale, the best
|
|
* we can do here */
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_component_index (GimpImage *image,
|
|
GimpChannelType channel)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), -1);
|
|
|
|
switch (channel)
|
|
{
|
|
case GIMP_CHANNEL_RED: return RED;
|
|
case GIMP_CHANNEL_GREEN: return GREEN;
|
|
case GIMP_CHANNEL_BLUE: return BLUE;
|
|
case GIMP_CHANNEL_GRAY: return GRAY;
|
|
case GIMP_CHANNEL_INDEXED: return INDEXED;
|
|
case GIMP_CHANNEL_ALPHA:
|
|
switch (gimp_image_get_base_type (image))
|
|
{
|
|
case GIMP_RGB: return ALPHA;
|
|
case GIMP_GRAY: return ALPHA_G;
|
|
case GIMP_INDEXED: return ALPHA_I;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
void
|
|
gimp_image_set_component_active (GimpImage *image,
|
|
GimpChannelType channel,
|
|
gboolean active)
|
|
{
|
|
GimpImagePrivate *private;
|
|
gint index = -1;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
index = gimp_image_get_component_index (image, channel);
|
|
|
|
if (index != -1 && active != private->active[index])
|
|
{
|
|
private->active[index] = active ? TRUE : FALSE;
|
|
|
|
/* If there is an active channel and we mess with the components,
|
|
* the active channel gets unset...
|
|
*/
|
|
gimp_image_unset_selected_channels (image);
|
|
|
|
g_signal_emit (image,
|
|
gimp_image_signals[COMPONENT_ACTIVE_CHANGED], 0,
|
|
channel);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_get_component_active (GimpImage *image,
|
|
GimpChannelType channel)
|
|
{
|
|
gint index = -1;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
index = gimp_image_get_component_index (image, channel);
|
|
|
|
if (index != -1)
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->active[index];
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
gimp_image_get_active_array (GimpImage *image,
|
|
gboolean *components)
|
|
{
|
|
GimpImagePrivate *private;
|
|
gint i;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (components != NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
for (i = 0; i < MAX_CHANNELS; i++)
|
|
components[i] = private->active[i];
|
|
}
|
|
|
|
GimpComponentMask
|
|
gimp_image_get_active_mask (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GimpComponentMask mask = 0;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
switch (gimp_image_get_base_type (image))
|
|
{
|
|
case GIMP_RGB:
|
|
mask |= (private->active[RED]) ? GIMP_COMPONENT_MASK_RED : 0;
|
|
mask |= (private->active[GREEN]) ? GIMP_COMPONENT_MASK_GREEN : 0;
|
|
mask |= (private->active[BLUE]) ? GIMP_COMPONENT_MASK_BLUE : 0;
|
|
mask |= (private->active[ALPHA]) ? GIMP_COMPONENT_MASK_ALPHA : 0;
|
|
break;
|
|
|
|
case GIMP_GRAY:
|
|
case GIMP_INDEXED:
|
|
mask |= (private->active[GRAY]) ? GIMP_COMPONENT_MASK_RED : 0;
|
|
mask |= (private->active[GRAY]) ? GIMP_COMPONENT_MASK_GREEN : 0;
|
|
mask |= (private->active[GRAY]) ? GIMP_COMPONENT_MASK_BLUE : 0;
|
|
mask |= (private->active[ALPHA_G]) ? GIMP_COMPONENT_MASK_ALPHA : 0;
|
|
break;
|
|
}
|
|
|
|
return mask;
|
|
}
|
|
|
|
void
|
|
gimp_image_set_component_visible (GimpImage *image,
|
|
GimpChannelType channel,
|
|
gboolean visible)
|
|
{
|
|
GimpImagePrivate *private;
|
|
gint index = -1;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
index = gimp_image_get_component_index (image, channel);
|
|
|
|
if (index != -1 && visible != private->visible[index])
|
|
{
|
|
private->visible[index] = visible ? TRUE : FALSE;
|
|
|
|
if (private->visible_mask)
|
|
{
|
|
GimpComponentMask mask;
|
|
|
|
mask = ~gimp_image_get_visible_mask (image) & GIMP_COMPONENT_MASK_ALL;
|
|
|
|
gegl_node_set (private->visible_mask,
|
|
"mask", mask,
|
|
NULL);
|
|
}
|
|
|
|
g_signal_emit (image,
|
|
gimp_image_signals[COMPONENT_VISIBILITY_CHANGED], 0,
|
|
channel);
|
|
|
|
gimp_image_invalidate_all (image);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_get_component_visible (GimpImage *image,
|
|
GimpChannelType channel)
|
|
{
|
|
gint index = -1;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
index = gimp_image_get_component_index (image, channel);
|
|
|
|
if (index != -1)
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->visible[index];
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
gimp_image_get_visible_array (GimpImage *image,
|
|
gboolean *components)
|
|
{
|
|
GimpImagePrivate *private;
|
|
gint i;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (components != NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
for (i = 0; i < MAX_CHANNELS; i++)
|
|
components[i] = private->visible[i];
|
|
}
|
|
|
|
GimpComponentMask
|
|
gimp_image_get_visible_mask (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GimpComponentMask mask = 0;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
switch (gimp_image_get_base_type (image))
|
|
{
|
|
case GIMP_RGB:
|
|
mask |= (private->visible[RED]) ? GIMP_COMPONENT_MASK_RED : 0;
|
|
mask |= (private->visible[GREEN]) ? GIMP_COMPONENT_MASK_GREEN : 0;
|
|
mask |= (private->visible[BLUE]) ? GIMP_COMPONENT_MASK_BLUE : 0;
|
|
mask |= (private->visible[ALPHA]) ? GIMP_COMPONENT_MASK_ALPHA : 0;
|
|
break;
|
|
|
|
case GIMP_GRAY:
|
|
case GIMP_INDEXED:
|
|
mask |= (private->visible[GRAY]) ? GIMP_COMPONENT_MASK_RED : 0;
|
|
mask |= (private->visible[GRAY]) ? GIMP_COMPONENT_MASK_GREEN : 0;
|
|
mask |= (private->visible[GRAY]) ? GIMP_COMPONENT_MASK_BLUE : 0;
|
|
mask |= (private->visible[ALPHA]) ? GIMP_COMPONENT_MASK_ALPHA : 0;
|
|
break;
|
|
}
|
|
|
|
return mask;
|
|
}
|
|
|
|
|
|
/* emitting image signals */
|
|
|
|
void
|
|
gimp_image_mode_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[MODE_CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
gimp_image_precision_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[PRECISION_CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
gimp_image_alpha_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[ALPHA_CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
gimp_image_invalidate (GimpImage *image,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
gimp_projectable_invalidate (GIMP_PROJECTABLE (image),
|
|
x, y, width, height);
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->flush_accum.preview_invalidated = TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_image_invalidate_all (GimpImage *image)
|
|
{
|
|
const GeglRectangle *bounding_box;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
bounding_box = &GIMP_IMAGE_GET_PRIVATE (image)->bounding_box;
|
|
|
|
gimp_image_invalidate (image,
|
|
bounding_box->x, bounding_box->y,
|
|
bounding_box->width, bounding_box->height);
|
|
}
|
|
|
|
void
|
|
gimp_image_guide_added (GimpImage *image,
|
|
GimpGuide *guide)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_GUIDE (guide));
|
|
|
|
g_signal_emit (image, gimp_image_signals[GUIDE_ADDED], 0,
|
|
guide);
|
|
}
|
|
|
|
void
|
|
gimp_image_guide_removed (GimpImage *image,
|
|
GimpGuide *guide)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_GUIDE (guide));
|
|
|
|
g_signal_emit (image, gimp_image_signals[GUIDE_REMOVED], 0,
|
|
guide);
|
|
}
|
|
|
|
void
|
|
gimp_image_guide_moved (GimpImage *image,
|
|
GimpGuide *guide)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_GUIDE (guide));
|
|
|
|
g_signal_emit (image, gimp_image_signals[GUIDE_MOVED], 0,
|
|
guide);
|
|
}
|
|
|
|
void
|
|
gimp_image_sample_point_added (GimpImage *image,
|
|
GimpSamplePoint *sample_point)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_SAMPLE_POINT (sample_point));
|
|
|
|
g_signal_emit (image, gimp_image_signals[SAMPLE_POINT_ADDED], 0,
|
|
sample_point);
|
|
}
|
|
|
|
void
|
|
gimp_image_sample_point_removed (GimpImage *image,
|
|
GimpSamplePoint *sample_point)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_SAMPLE_POINT (sample_point));
|
|
|
|
g_signal_emit (image, gimp_image_signals[SAMPLE_POINT_REMOVED], 0,
|
|
sample_point);
|
|
}
|
|
|
|
void
|
|
gimp_image_sample_point_moved (GimpImage *image,
|
|
GimpSamplePoint *sample_point)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_SAMPLE_POINT (sample_point));
|
|
|
|
g_signal_emit (image, gimp_image_signals[SAMPLE_POINT_MOVED], 0,
|
|
sample_point);
|
|
}
|
|
|
|
/**
|
|
* gimp_image_size_changed_detailed:
|
|
* @image:
|
|
* @previous_origin_x:
|
|
* @previous_origin_y:
|
|
*
|
|
* Emits the size-changed-detailed signal that is typically used to adjust the
|
|
* position of the image in the display shell on various operations,
|
|
* e.g. crop.
|
|
*
|
|
* This function makes sure that GimpViewable::size-changed is also emitted.
|
|
**/
|
|
void
|
|
gimp_image_size_changed_detailed (GimpImage *image,
|
|
gint previous_origin_x,
|
|
gint previous_origin_y,
|
|
gint previous_width,
|
|
gint previous_height)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[SIZE_CHANGED_DETAILED], 0,
|
|
previous_origin_x,
|
|
previous_origin_y,
|
|
previous_width,
|
|
previous_height);
|
|
}
|
|
|
|
void
|
|
gimp_image_colormap_changed (GimpImage *image,
|
|
gint color_index)
|
|
{
|
|
GimpPalette *palette;
|
|
gint n_colors;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
palette = GIMP_IMAGE_GET_PRIVATE (image)->palette;
|
|
n_colors = palette ? gimp_palette_get_n_colors (palette) : 0;
|
|
g_return_if_fail (color_index >= -1 && color_index < n_colors);
|
|
|
|
g_signal_emit (image, gimp_image_signals[COLORMAP_CHANGED], 0,
|
|
color_index);
|
|
}
|
|
|
|
void
|
|
gimp_image_selection_invalidate (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[SELECTION_INVALIDATE], 0);
|
|
}
|
|
|
|
void
|
|
gimp_image_quick_mask_changed (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[QUICK_MASK_CHANGED], 0);
|
|
}
|
|
|
|
void
|
|
gimp_image_undo_event (GimpImage *image,
|
|
GimpUndoEvent event,
|
|
GimpUndo *undo)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (((event == GIMP_UNDO_EVENT_UNDO_FREE ||
|
|
event == GIMP_UNDO_EVENT_UNDO_FREEZE ||
|
|
event == GIMP_UNDO_EVENT_UNDO_THAW) && undo == NULL) ||
|
|
GIMP_IS_UNDO (undo));
|
|
|
|
g_signal_emit (image, gimp_image_signals[UNDO_EVENT], 0, event, undo);
|
|
}
|
|
|
|
|
|
/* dirty counters */
|
|
|
|
/* NOTE about the image->dirty counter:
|
|
* If 0, then the image is clean (ie, copy on disk is the same as the one
|
|
* in memory).
|
|
* If positive, then that's the number of dirtying operations done
|
|
* on the image since the last save.
|
|
* If negative, then user has hit undo and gone back in time prior
|
|
* to the saved copy. Hitting redo will eventually come back to
|
|
* the saved copy.
|
|
*
|
|
* The image is dirty (ie, needs saving) if counter is non-zero.
|
|
*
|
|
* If the counter is around 100000, this is due to undo-ing back
|
|
* before a saved version, then changing the image (thus destroying
|
|
* the redo stack). Once this has happened, it's impossible to get
|
|
* the image back to the state on disk, since the redo info has been
|
|
* freed. See gimpimage-undo.c for the gory details.
|
|
*/
|
|
|
|
/*
|
|
* NEVER CALL gimp_image_dirty() directly!
|
|
*
|
|
* If your code has just dirtied the image, push an undo instead.
|
|
* Failing that, push the trivial undo which tells the user the
|
|
* command is not undoable: undo_push_cantundo() (But really, it would
|
|
* be best to push a proper undo). If you just dirty the image
|
|
* without pushing an undo then the dirty count is increased, but
|
|
* popping that many undo actions won't lead to a clean image.
|
|
*/
|
|
|
|
gint
|
|
gimp_image_dirty (GimpImage *image,
|
|
GimpDirtyMask dirty_mask)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->dirty++;
|
|
private->export_dirty++;
|
|
|
|
if (! private->dirty_time)
|
|
private->dirty_time = time (NULL);
|
|
|
|
g_signal_emit (image, gimp_image_signals[DIRTY], 0, dirty_mask);
|
|
|
|
TRC (("dirty %d -> %d\n", private->dirty - 1, private->dirty));
|
|
|
|
return private->dirty;
|
|
}
|
|
|
|
gint
|
|
gimp_image_clean (GimpImage *image,
|
|
GimpDirtyMask dirty_mask)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->dirty--;
|
|
private->export_dirty--;
|
|
|
|
g_signal_emit (image, gimp_image_signals[CLEAN], 0, dirty_mask);
|
|
|
|
TRC (("clean %d -> %d\n", private->dirty + 1, private->dirty));
|
|
|
|
return private->dirty;
|
|
}
|
|
|
|
void
|
|
gimp_image_clean_all (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->dirty = 0;
|
|
private->dirty_time = 0;
|
|
|
|
g_signal_emit (image, gimp_image_signals[CLEAN], 0, GIMP_DIRTY_ALL);
|
|
|
|
gimp_object_name_changed (GIMP_OBJECT (image));
|
|
}
|
|
|
|
void
|
|
gimp_image_export_clean_all (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->export_dirty = 0;
|
|
|
|
g_signal_emit (image, gimp_image_signals[CLEAN], 0, GIMP_DIRTY_ALL);
|
|
|
|
gimp_object_name_changed (GIMP_OBJECT (image));
|
|
}
|
|
|
|
/**
|
|
* gimp_image_is_dirty:
|
|
* @image:
|
|
*
|
|
* Returns: True if the image is dirty, false otherwise.
|
|
**/
|
|
gint
|
|
gimp_image_is_dirty (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->dirty != 0;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_is_export_dirty:
|
|
* @image:
|
|
*
|
|
* Returns: True if the image export is dirty, false otherwise.
|
|
**/
|
|
gboolean
|
|
gimp_image_is_export_dirty (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->export_dirty != 0;
|
|
}
|
|
|
|
gint64
|
|
gimp_image_get_dirty_time (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->dirty_time;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_saving:
|
|
* @image:
|
|
*
|
|
* Emits the "saving" signal, indicating that @image is about to be saved,
|
|
* or exported.
|
|
*/
|
|
void
|
|
gimp_image_saving (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_signal_emit (image, gimp_image_signals[SAVING], 0);
|
|
}
|
|
|
|
/**
|
|
* gimp_image_saved:
|
|
* @image:
|
|
* @file:
|
|
*
|
|
* Emits the "saved" signal, indicating that @image was saved to the
|
|
* location specified by @file.
|
|
*/
|
|
void
|
|
gimp_image_saved (GimpImage *image,
|
|
GFile *file)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (G_IS_FILE (file));
|
|
|
|
g_signal_emit (image, gimp_image_signals[SAVED], 0, file);
|
|
}
|
|
|
|
/**
|
|
* gimp_image_exported:
|
|
* @image:
|
|
* @file:
|
|
*
|
|
* Emits the "exported" signal, indicating that @image was exported to the
|
|
* location specified by @file.
|
|
*/
|
|
void
|
|
gimp_image_exported (GimpImage *image,
|
|
GFile *file)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (G_IS_FILE (file));
|
|
|
|
g_signal_emit (image, gimp_image_signals[EXPORTED], 0, file);
|
|
}
|
|
|
|
|
|
/* flush this image's displays */
|
|
|
|
void
|
|
gimp_image_flush (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
gimp_projectable_flush (GIMP_PROJECTABLE (image),
|
|
GIMP_IMAGE_GET_PRIVATE (image)->flush_accum.preview_invalidated);
|
|
}
|
|
|
|
|
|
/* display / instance counters */
|
|
|
|
gint
|
|
gimp_image_get_display_count (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->disp_count;
|
|
}
|
|
|
|
void
|
|
gimp_image_inc_display_count (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->disp_count++;
|
|
}
|
|
|
|
void
|
|
gimp_image_dec_display_count (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->disp_count--;
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_instance_count (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->instance_count;
|
|
}
|
|
|
|
void
|
|
gimp_image_inc_instance_count (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->instance_count++;
|
|
}
|
|
|
|
void
|
|
gimp_image_inc_show_all_count (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->show_all++;
|
|
|
|
if (GIMP_IMAGE_GET_PRIVATE (image)->show_all == 1)
|
|
{
|
|
g_clear_object (&GIMP_IMAGE_GET_PRIVATE (image)->pickable_buffer);
|
|
|
|
gimp_image_update_bounding_box (image);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_image_dec_show_all_count (GimpImage *image)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
GIMP_IMAGE_GET_PRIVATE (image)->show_all--;
|
|
|
|
if (GIMP_IMAGE_GET_PRIVATE (image)->show_all == 0)
|
|
{
|
|
g_clear_object (&GIMP_IMAGE_GET_PRIVATE (image)->pickable_buffer);
|
|
|
|
gimp_image_update_bounding_box (image);
|
|
}
|
|
}
|
|
|
|
|
|
/* parasites */
|
|
|
|
const GimpParasite *
|
|
gimp_image_parasite_find (GimpImage *image,
|
|
const gchar *name)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return gimp_parasite_list_find (GIMP_IMAGE_GET_PRIVATE (image)->parasites,
|
|
name);
|
|
}
|
|
|
|
static void
|
|
list_func (gchar *key,
|
|
GimpParasite *p,
|
|
gchar ***cur)
|
|
{
|
|
*(*cur)++ = (gchar *) g_strdup (key);
|
|
}
|
|
|
|
gchar **
|
|
gimp_image_parasite_list (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
gint count;
|
|
gchar **list;
|
|
gchar **cur;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
count = gimp_parasite_list_length (private->parasites);
|
|
cur = list = g_new0 (gchar *, count + 1);
|
|
|
|
gimp_parasite_list_foreach (private->parasites, (GHFunc) list_func, &cur);
|
|
|
|
return list;
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_parasite_validate (GimpImage *image,
|
|
const GimpParasite *parasite,
|
|
GError **error)
|
|
{
|
|
const gchar *name;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (parasite != NULL, FALSE);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
|
|
name = gimp_parasite_get_name (parasite);
|
|
|
|
if (strcmp (name, GIMP_ICC_PROFILE_PARASITE_NAME) == 0 ||
|
|
strcmp (name, GIMP_SIMULATION_ICC_PROFILE_PARASITE_NAME) == 0)
|
|
{
|
|
return gimp_image_validate_icc_parasite (image, parasite,
|
|
name, NULL, error);
|
|
}
|
|
else if (strcmp (name, "gimp-comment") == 0)
|
|
{
|
|
const gchar *data;
|
|
guint32 length;
|
|
gboolean valid = FALSE;
|
|
|
|
data = gimp_parasite_get_data (parasite, &length);
|
|
if (length > 0)
|
|
{
|
|
if (data[length - 1] == '\0')
|
|
valid = g_utf8_validate (data, -1, NULL);
|
|
else
|
|
valid = g_utf8_validate (data, length, NULL);
|
|
}
|
|
|
|
if (! valid)
|
|
{
|
|
g_set_error (error, GIMP_ERROR, GIMP_FAILED,
|
|
_("'gimp-comment' parasite validation failed: "
|
|
"comment contains invalid UTF-8"));
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_image_parasite_attach (GimpImage *image,
|
|
const GimpParasite *parasite,
|
|
gboolean push_undo)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GimpParasite copy;
|
|
const gchar *name;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (parasite != NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
name = gimp_parasite_get_name (parasite);
|
|
|
|
/* this is so ugly and is only for the PDB */
|
|
if (strcmp (name, GIMP_ICC_PROFILE_PARASITE_NAME) == 0 ||
|
|
strcmp (name, GIMP_SIMULATION_ICC_PROFILE_PARASITE_NAME) == 0)
|
|
{
|
|
GimpColorProfile *profile;
|
|
GimpColorProfile *builtin;
|
|
const guint8 *parasite_data;
|
|
guint32 parasite_length;
|
|
|
|
parasite_data = gimp_parasite_get_data (parasite, ¶site_length);
|
|
profile =
|
|
gimp_color_profile_new_from_icc_profile (parasite_data, parasite_length,
|
|
NULL);
|
|
builtin = gimp_image_get_builtin_color_profile (image);
|
|
|
|
if (gimp_color_profile_is_equal (profile, builtin))
|
|
{
|
|
/* setting the builtin profile is equal to removing the profile */
|
|
gimp_image_parasite_detach (image, GIMP_ICC_PROFILE_PARASITE_NAME,
|
|
push_undo);
|
|
g_object_unref (profile);
|
|
return;
|
|
}
|
|
|
|
g_object_unref (profile);
|
|
}
|
|
|
|
/* make a temporary copy of the GimpParasite struct because
|
|
* gimp_parasite_shift_parent() changes it
|
|
*/
|
|
copy = *parasite;
|
|
|
|
/* only set the dirty bit manually if we can be saved and the new
|
|
* parasite differs from the current one and we aren't undoable
|
|
*/
|
|
if (push_undo && gimp_parasite_is_undoable (©))
|
|
gimp_image_undo_push_image_parasite (image,
|
|
C_("undo-type", "Attach Parasite to Image"),
|
|
©);
|
|
|
|
/* We used to push a cantundo on the stack here. This made the undo stack
|
|
* unusable (NULL on the stack) and prevented people from undoing after a
|
|
* save (since most save plug-ins attach an undoable comment parasite).
|
|
* Now we simply attach the parasite without pushing an undo. That way
|
|
* it's undoable but does not block the undo system. --Sven
|
|
*/
|
|
gimp_parasite_list_add (private->parasites, ©);
|
|
|
|
if (push_undo && gimp_parasite_has_flag (©, GIMP_PARASITE_ATTACH_PARENT))
|
|
{
|
|
gimp_parasite_shift_parent (©);
|
|
gimp_parasite_attach (image->gimp, ©);
|
|
}
|
|
|
|
if (strcmp (name, GIMP_ICC_PROFILE_PARASITE_NAME) == 0)
|
|
_gimp_image_update_color_profile (image, parasite);
|
|
|
|
if (strcmp (name, GIMP_SIMULATION_ICC_PROFILE_PARASITE_NAME) == 0)
|
|
_gimp_image_update_simulation_profile (image, parasite);
|
|
|
|
g_signal_emit (image, gimp_image_signals[PARASITE_ATTACHED], 0,
|
|
name);
|
|
}
|
|
|
|
void
|
|
gimp_image_parasite_detach (GimpImage *image,
|
|
const gchar *name,
|
|
gboolean push_undo)
|
|
{
|
|
GimpImagePrivate *private;
|
|
const GimpParasite *parasite;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (name != NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (! (parasite = gimp_parasite_list_find (private->parasites, name)))
|
|
return;
|
|
|
|
if (push_undo && gimp_parasite_is_undoable (parasite))
|
|
gimp_image_undo_push_image_parasite_remove (image,
|
|
C_("undo-type", "Remove Parasite from Image"),
|
|
name);
|
|
|
|
gimp_parasite_list_remove (private->parasites, name);
|
|
|
|
if (strcmp (name, GIMP_ICC_PROFILE_PARASITE_NAME) == 0)
|
|
_gimp_image_update_color_profile (image, NULL);
|
|
|
|
if (strcmp (name, GIMP_SIMULATION_ICC_PROFILE_PARASITE_NAME) == 0)
|
|
_gimp_image_update_simulation_profile (image, NULL);
|
|
|
|
g_signal_emit (image, gimp_image_signals[PARASITE_DETACHED], 0,
|
|
name);
|
|
}
|
|
|
|
|
|
/* tattoos */
|
|
|
|
GimpTattoo
|
|
gimp_image_get_new_tattoo (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->tattoo_state++;
|
|
|
|
if (G_UNLIKELY (private->tattoo_state == 0))
|
|
g_warning ("%s: Tattoo state corrupted (integer overflow).", G_STRFUNC);
|
|
|
|
return private->tattoo_state;
|
|
}
|
|
|
|
GimpTattoo
|
|
gimp_image_get_tattoo_state (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->tattoo_state;
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_set_tattoo_state (GimpImage *image,
|
|
GimpTattoo val)
|
|
{
|
|
GList *all_items;
|
|
GList *list;
|
|
gboolean retval = TRUE;
|
|
GimpTattoo maxval = 0;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
/* Check that the layer tattoos don't overlap with channel or vector ones */
|
|
all_items = gimp_image_get_layer_list (image);
|
|
|
|
for (list = all_items; list; list = g_list_next (list))
|
|
{
|
|
GimpTattoo ltattoo;
|
|
|
|
ltattoo = gimp_item_get_tattoo (GIMP_ITEM (list->data));
|
|
if (ltattoo > maxval)
|
|
maxval = ltattoo;
|
|
|
|
if (gimp_image_get_channel_by_tattoo (image, ltattoo))
|
|
retval = FALSE; /* Oopps duplicated tattoo in channel */
|
|
|
|
if (gimp_image_get_vectors_by_tattoo (image, ltattoo))
|
|
retval = FALSE; /* Oopps duplicated tattoo in vectors */
|
|
}
|
|
|
|
g_list_free (all_items);
|
|
|
|
/* Now check that the channel and vectors tattoos don't overlap */
|
|
all_items = gimp_image_get_channel_list (image);
|
|
|
|
for (list = all_items; list; list = g_list_next (list))
|
|
{
|
|
GimpTattoo ctattoo;
|
|
|
|
ctattoo = gimp_item_get_tattoo (GIMP_ITEM (list->data));
|
|
if (ctattoo > maxval)
|
|
maxval = ctattoo;
|
|
|
|
if (gimp_image_get_vectors_by_tattoo (image, ctattoo))
|
|
retval = FALSE; /* Oopps duplicated tattoo in vectors */
|
|
}
|
|
|
|
g_list_free (all_items);
|
|
|
|
/* Find the max tattoo value in the vectors */
|
|
all_items = gimp_image_get_vectors_list (image);
|
|
|
|
for (list = all_items; list; list = g_list_next (list))
|
|
{
|
|
GimpTattoo vtattoo;
|
|
|
|
vtattoo = gimp_item_get_tattoo (GIMP_ITEM (list->data));
|
|
if (vtattoo > maxval)
|
|
maxval = vtattoo;
|
|
}
|
|
|
|
g_list_free (all_items);
|
|
|
|
if (val < maxval)
|
|
retval = FALSE;
|
|
|
|
/* Must check if the state is valid */
|
|
if (retval == TRUE)
|
|
GIMP_IMAGE_GET_PRIVATE (image)->tattoo_state = val;
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
/* projection */
|
|
|
|
GimpProjection *
|
|
gimp_image_get_projection (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->projection;
|
|
}
|
|
|
|
|
|
/* layers / channels / vectors */
|
|
|
|
GimpItemTree *
|
|
gimp_image_get_layer_tree (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->layers;
|
|
}
|
|
|
|
GimpItemTree *
|
|
gimp_image_get_channel_tree (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->channels;
|
|
}
|
|
|
|
GimpItemTree *
|
|
gimp_image_get_vectors_tree (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->vectors;
|
|
}
|
|
|
|
GimpContainer *
|
|
gimp_image_get_layers (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->layers->container;
|
|
}
|
|
|
|
GimpContainer *
|
|
gimp_image_get_channels (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->channels->container;
|
|
}
|
|
|
|
GimpContainer *
|
|
gimp_image_get_vectors (GimpImage *image)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
return GIMP_IMAGE_GET_PRIVATE (image)->vectors->container;
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_n_layers (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_layers (image));
|
|
|
|
return gimp_item_stack_get_n_items (stack);
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_n_channels (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_channels (image));
|
|
|
|
return gimp_item_stack_get_n_items (stack);
|
|
}
|
|
|
|
gint
|
|
gimp_image_get_n_vectors (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), 0);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_vectors (image));
|
|
|
|
return gimp_item_stack_get_n_items (stack);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_layer_iter (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_layers (image));
|
|
|
|
return gimp_item_stack_get_item_iter (stack);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_channel_iter (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_channels (image));
|
|
|
|
return gimp_item_stack_get_item_iter (stack);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_vectors_iter (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_vectors (image));
|
|
|
|
return gimp_item_stack_get_item_iter (stack);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_layer_list (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_layers (image));
|
|
|
|
return gimp_item_stack_get_item_list (stack);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_channel_list (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_channels (image));
|
|
|
|
return gimp_item_stack_get_item_list (stack);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_vectors_list (GimpImage *image)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_vectors (image));
|
|
|
|
return gimp_item_stack_get_item_list (stack);
|
|
}
|
|
|
|
|
|
/* active drawable, layer, channel, vectors */
|
|
|
|
void
|
|
gimp_image_unset_selected_channels (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *channels;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
channels = gimp_image_get_selected_channels (image);
|
|
|
|
if (channels)
|
|
{
|
|
gimp_image_set_selected_channels (image, NULL);
|
|
|
|
if (private->layer_stack)
|
|
gimp_image_set_selected_layers (image, private->layer_stack->data);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_image_is_selected_drawable:
|
|
* @image:
|
|
* @drawable:
|
|
*
|
|
* Checks if @drawable belong to the list of currently selected
|
|
* drawables. It doesn't mean this is the only selected drawable (if
|
|
* this is what you want to check, use
|
|
* gimp_image_equal_selected_drawables() with a list containing only
|
|
* this drawable).
|
|
*
|
|
* Returns: %TRUE is @drawable is one of the selected drawables.
|
|
*/
|
|
gboolean
|
|
gimp_image_is_selected_drawable (GimpImage *image,
|
|
GimpDrawable *drawable)
|
|
{
|
|
GList *selected_drawables;
|
|
gboolean found;
|
|
|
|
selected_drawables = gimp_image_get_selected_drawables (image);
|
|
found = (g_list_find (selected_drawables, drawable) != NULL);
|
|
g_list_free (selected_drawables);
|
|
|
|
return found;
|
|
}
|
|
|
|
/**
|
|
* gimp_image_equal_selected_drawables:
|
|
* @image:
|
|
* @drawables:
|
|
*
|
|
* Compare the list of @drawables with the selected drawables in @image
|
|
* (i.e. the result of gimp_image_equal_selected_drawables()).
|
|
* The order of the @drawables does not matter, only if the size and
|
|
* contents of the list is the same.
|
|
*/
|
|
gboolean
|
|
gimp_image_equal_selected_drawables (GimpImage *image,
|
|
GList *drawables)
|
|
{
|
|
GList *selected_drawables;
|
|
GList *iter;
|
|
gboolean equal = FALSE;
|
|
|
|
selected_drawables = gimp_image_get_selected_drawables (image);
|
|
|
|
if (g_list_length (drawables) == g_list_length (selected_drawables))
|
|
{
|
|
equal = TRUE;
|
|
|
|
for (iter = drawables; iter; iter = iter->next)
|
|
if (! g_list_find (selected_drawables, iter->data))
|
|
{
|
|
equal = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
g_list_free (selected_drawables);
|
|
|
|
return equal;
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_selected_drawables (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *selected_channels;
|
|
GList *selected_layers;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
selected_channels = gimp_item_tree_get_selected_items (private->channels);
|
|
selected_layers = gimp_item_tree_get_selected_items (private->layers);
|
|
|
|
/* If there is an active channel (a saved selection, etc.),
|
|
* we ignore the active layer
|
|
*/
|
|
if (selected_channels)
|
|
{
|
|
return g_list_copy (selected_channels);
|
|
}
|
|
else if (selected_layers)
|
|
{
|
|
selected_layers = g_list_copy (selected_layers);
|
|
if (g_list_length (selected_layers) == 1)
|
|
{
|
|
/* As a special case, if only one layer is selected and mask
|
|
* edit is in progress, we return the mask as selected
|
|
* drawable instead of the layer.
|
|
*/
|
|
GimpLayer *layer = GIMP_LAYER (selected_layers->data);
|
|
GimpLayerMask *mask = gimp_layer_get_mask (layer);
|
|
|
|
if (mask && gimp_layer_get_edit_mask (layer))
|
|
selected_layers->data = mask;
|
|
}
|
|
return selected_layers;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_selected_layers (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
return gimp_item_tree_get_selected_items (private->layers);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_selected_channels (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
return gimp_item_tree_get_selected_items (private->channels);
|
|
}
|
|
|
|
GList *
|
|
gimp_image_get_selected_vectors (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
return gimp_item_tree_get_selected_items (private->vectors);
|
|
}
|
|
|
|
void
|
|
gimp_image_set_selected_layers (GimpImage *image,
|
|
GList *layers)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GimpLayer *floating_sel;
|
|
GList *selected_layers;
|
|
GList *layers2;
|
|
GList *iter;
|
|
gboolean selection_changed = TRUE;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
layers2 = g_list_copy (layers);
|
|
for (iter = layers; iter; iter = iter->next)
|
|
{
|
|
g_return_if_fail (GIMP_IS_LAYER (iter->data));
|
|
g_return_if_fail (gimp_item_get_image (GIMP_ITEM (iter->data)) == image);
|
|
|
|
/* Silently remove non-attached layers from selection. Do not
|
|
* error out on it as it may happen for instance when selection
|
|
* changes while in the process of removing a layer group.
|
|
*/
|
|
if (! gimp_item_is_attached (GIMP_ITEM (iter->data)))
|
|
layers2 = g_list_remove (layers2, iter->data);
|
|
}
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
floating_sel = gimp_image_get_floating_selection (image);
|
|
|
|
/* Make sure the floating_sel always is the active layer */
|
|
if (floating_sel && (g_list_length (layers2) != 1 || layers2->data != floating_sel))
|
|
return;
|
|
|
|
selected_layers = gimp_image_get_selected_layers (image);
|
|
|
|
if (g_list_length (layers2) == g_list_length (selected_layers))
|
|
{
|
|
selection_changed = FALSE;
|
|
for (iter = layers2; iter; iter = iter->next)
|
|
{
|
|
if (g_list_find (selected_layers, iter->data) == NULL)
|
|
{
|
|
selection_changed = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (selection_changed)
|
|
{
|
|
/* Don't cache selection info for the previous active layer */
|
|
if (selected_layers)
|
|
gimp_drawable_invalidate_boundary (GIMP_DRAWABLE (selected_layers->data));
|
|
|
|
gimp_item_tree_set_selected_items (private->layers, layers2);
|
|
|
|
/* We cannot edit masks with multiple selected layers. */
|
|
if (g_list_length (layers2) > 1)
|
|
{
|
|
for (iter = layers2; iter; iter = iter->next)
|
|
{
|
|
if (gimp_layer_get_mask (iter->data))
|
|
gimp_layer_set_edit_mask (iter->data, FALSE);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
g_list_free (layers2);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_image_set_selected_channels (GimpImage *image,
|
|
GList *channels)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *iter;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
for (iter = channels; iter; iter = iter->next)
|
|
{
|
|
g_return_if_fail (GIMP_IS_CHANNEL (iter->data));
|
|
g_return_if_fail (gimp_item_is_attached (GIMP_ITEM (iter->data)) &&
|
|
gimp_item_get_image (GIMP_ITEM (iter->data)) == image);
|
|
}
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* Not if there is a floating selection */
|
|
if (g_list_length (channels) > 0 && gimp_image_get_floating_selection (image))
|
|
return;
|
|
|
|
gimp_item_tree_set_selected_items (private->channels, g_list_copy (channels));
|
|
}
|
|
|
|
void
|
|
gimp_image_set_selected_vectors (GimpImage *image,
|
|
GList *vectors)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *iter;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
for (iter = vectors; iter; iter = iter->next)
|
|
{
|
|
g_return_if_fail (GIMP_IS_VECTORS (iter->data));
|
|
g_return_if_fail (gimp_item_is_attached (GIMP_ITEM (iter->data)) &&
|
|
gimp_item_get_image (GIMP_ITEM (iter->data)) == image);
|
|
}
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
gimp_item_tree_set_selected_items (private->vectors, g_list_copy (vectors));
|
|
}
|
|
|
|
|
|
/* layer, channel, vectors by tattoo */
|
|
|
|
GimpLayer *
|
|
gimp_image_get_layer_by_tattoo (GimpImage *image,
|
|
GimpTattoo tattoo)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_layers (image));
|
|
|
|
return GIMP_LAYER (gimp_item_stack_get_item_by_tattoo (stack, tattoo));
|
|
}
|
|
|
|
GimpChannel *
|
|
gimp_image_get_channel_by_tattoo (GimpImage *image,
|
|
GimpTattoo tattoo)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_channels (image));
|
|
|
|
return GIMP_CHANNEL (gimp_item_stack_get_item_by_tattoo (stack, tattoo));
|
|
}
|
|
|
|
GimpVectors *
|
|
gimp_image_get_vectors_by_tattoo (GimpImage *image,
|
|
GimpTattoo tattoo)
|
|
{
|
|
GimpItemStack *stack;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
stack = GIMP_ITEM_STACK (gimp_image_get_vectors (image));
|
|
|
|
return GIMP_VECTORS (gimp_item_stack_get_item_by_tattoo (stack, tattoo));
|
|
}
|
|
|
|
|
|
/* layer, channel, vectors by name */
|
|
|
|
GimpLayer *
|
|
gimp_image_get_layer_by_name (GimpImage *image,
|
|
const gchar *name)
|
|
{
|
|
GimpItemTree *tree;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
tree = gimp_image_get_layer_tree (image);
|
|
|
|
return GIMP_LAYER (gimp_item_tree_get_item_by_name (tree, name));
|
|
}
|
|
|
|
GimpChannel *
|
|
gimp_image_get_channel_by_name (GimpImage *image,
|
|
const gchar *name)
|
|
{
|
|
GimpItemTree *tree;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
tree = gimp_image_get_channel_tree (image);
|
|
|
|
return GIMP_CHANNEL (gimp_item_tree_get_item_by_name (tree, name));
|
|
}
|
|
|
|
GimpVectors *
|
|
gimp_image_get_vectors_by_name (GimpImage *image,
|
|
const gchar *name)
|
|
{
|
|
GimpItemTree *tree;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
tree = gimp_image_get_vectors_tree (image);
|
|
|
|
return GIMP_VECTORS (gimp_item_tree_get_item_by_name (tree, name));
|
|
}
|
|
|
|
|
|
/* items */
|
|
|
|
gboolean
|
|
gimp_image_reorder_item (GimpImage *image,
|
|
GimpItem *item,
|
|
GimpItem *new_parent,
|
|
gint new_index,
|
|
gboolean push_undo,
|
|
const gchar *undo_desc)
|
|
{
|
|
GimpItemTree *tree;
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_ITEM (item), FALSE);
|
|
g_return_val_if_fail (gimp_item_get_image (item) == image, FALSE);
|
|
|
|
tree = gimp_item_get_tree (item);
|
|
|
|
g_return_val_if_fail (tree != NULL, FALSE);
|
|
|
|
if (push_undo)
|
|
{
|
|
if (! undo_desc)
|
|
undo_desc = GIMP_ITEM_GET_CLASS (item)->reorder_desc;
|
|
|
|
gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_IMAGE_ITEM_REORDER,
|
|
undo_desc);
|
|
}
|
|
|
|
gimp_image_freeze_bounding_box (image);
|
|
|
|
gimp_item_start_move (item, push_undo);
|
|
|
|
/* item and new_parent are type-checked in GimpItemTree
|
|
*/
|
|
result = gimp_item_tree_reorder_item (tree, item,
|
|
new_parent, new_index,
|
|
push_undo, undo_desc);
|
|
|
|
gimp_item_end_move (item, push_undo);
|
|
|
|
gimp_image_thaw_bounding_box (image);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_end (image);
|
|
|
|
return result;
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_raise_item (GimpImage *image,
|
|
GimpItem *item,
|
|
GError **error)
|
|
{
|
|
gint index;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_ITEM (item), FALSE);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
|
|
index = gimp_item_get_index (item);
|
|
|
|
g_return_val_if_fail (index != -1, FALSE);
|
|
|
|
if (index == 0)
|
|
{
|
|
g_set_error_literal (error, GIMP_ERROR, GIMP_FAILED,
|
|
GIMP_ITEM_GET_CLASS (item)->raise_failed);
|
|
return FALSE;
|
|
}
|
|
|
|
return gimp_image_reorder_item (image, item,
|
|
gimp_item_get_parent (item), index - 1,
|
|
TRUE, GIMP_ITEM_GET_CLASS (item)->raise_desc);
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_raise_item_to_top (GimpImage *image,
|
|
GimpItem *item)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_ITEM (item), FALSE);
|
|
|
|
return gimp_image_reorder_item (image, item,
|
|
gimp_item_get_parent (item), 0,
|
|
TRUE, GIMP_ITEM_GET_CLASS (item)->raise_to_top_desc);
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_lower_item (GimpImage *image,
|
|
GimpItem *item,
|
|
GError **error)
|
|
{
|
|
GimpContainer *container;
|
|
gint index;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_ITEM (item), FALSE);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
|
|
container = gimp_item_get_container (item);
|
|
|
|
g_return_val_if_fail (container != NULL, FALSE);
|
|
|
|
index = gimp_item_get_index (item);
|
|
|
|
if (index == gimp_container_get_n_children (container) - 1)
|
|
{
|
|
g_set_error_literal (error, GIMP_ERROR, GIMP_FAILED,
|
|
GIMP_ITEM_GET_CLASS (item)->lower_failed);
|
|
return FALSE;
|
|
}
|
|
|
|
return gimp_image_reorder_item (image, item,
|
|
gimp_item_get_parent (item), index + 1,
|
|
TRUE, GIMP_ITEM_GET_CLASS (item)->lower_desc);
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_lower_item_to_bottom (GimpImage *image,
|
|
GimpItem *item)
|
|
{
|
|
GimpContainer *container;
|
|
gint length;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_ITEM (item), FALSE);
|
|
|
|
container = gimp_item_get_container (item);
|
|
|
|
g_return_val_if_fail (container != NULL, FALSE);
|
|
|
|
length = gimp_container_get_n_children (container);
|
|
|
|
return gimp_image_reorder_item (image, item,
|
|
gimp_item_get_parent (item), length - 1,
|
|
TRUE, GIMP_ITEM_GET_CLASS (item)->lower_to_bottom_desc);
|
|
}
|
|
|
|
|
|
/* layers */
|
|
|
|
gboolean
|
|
gimp_image_add_layer (GimpImage *image,
|
|
GimpLayer *layer,
|
|
GimpLayer *parent,
|
|
gint position,
|
|
gboolean push_undo)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *selected_layers;
|
|
gboolean old_has_alpha;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* item and parent are type-checked in GimpItemTree
|
|
*/
|
|
if (! gimp_item_tree_get_insert_pos (private->layers,
|
|
(GimpItem *) layer,
|
|
(GimpItem **) &parent,
|
|
&position))
|
|
return FALSE;
|
|
|
|
gimp_image_unset_default_new_layer_mode (image);
|
|
|
|
/* If there is a floating selection (and this isn't it!),
|
|
* make sure the insert position is greater than 0
|
|
*/
|
|
if (parent == NULL && position == 0 &&
|
|
gimp_image_get_floating_selection (image))
|
|
position = 1;
|
|
|
|
old_has_alpha = gimp_image_has_alpha (image);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_layer_add (image, C_("undo-type", "Add Layer"),
|
|
layer,
|
|
gimp_image_get_selected_layers (image));
|
|
|
|
gimp_item_tree_add_item (private->layers, GIMP_ITEM (layer),
|
|
GIMP_ITEM (parent), position);
|
|
|
|
selected_layers = g_list_prepend (NULL, layer);
|
|
gimp_image_set_selected_layers (image, selected_layers);
|
|
g_list_free (selected_layers);
|
|
|
|
/* If the layer is a floating selection, attach it to the drawable */
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
gimp_drawable_attach_floating_sel (gimp_layer_get_floating_sel_drawable (layer),
|
|
layer);
|
|
|
|
if (old_has_alpha != gimp_image_has_alpha (image))
|
|
private->flush_accum.alpha_changed = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_image_remove_layer (GimpImage *image,
|
|
GimpLayer *layer,
|
|
gboolean push_undo,
|
|
GList *new_selected)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *selected_layers;
|
|
gboolean old_has_alpha;
|
|
const gchar *undo_desc;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_LAYER (layer));
|
|
g_return_if_fail (gimp_item_is_attached (GIMP_ITEM (layer)));
|
|
g_return_if_fail (gimp_item_get_image (GIMP_ITEM (layer)) == image);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
gimp_image_unset_default_new_layer_mode (image);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_IMAGE_ITEM_REMOVE,
|
|
C_("undo-type", "Remove Layer"));
|
|
|
|
gimp_item_start_move (GIMP_ITEM (layer), push_undo);
|
|
|
|
if (gimp_drawable_get_floating_sel (GIMP_DRAWABLE (layer)))
|
|
{
|
|
if (! push_undo)
|
|
{
|
|
g_warning ("%s() was called from an undo function while the layer "
|
|
"had a floating selection. Please report this at "
|
|
"https://www.gimp.org/bugs/", G_STRFUNC);
|
|
return;
|
|
}
|
|
|
|
gimp_image_remove_layer (image,
|
|
gimp_drawable_get_floating_sel (GIMP_DRAWABLE (layer)),
|
|
TRUE, NULL);
|
|
}
|
|
|
|
selected_layers = g_list_copy (gimp_image_get_selected_layers (image));
|
|
|
|
old_has_alpha = gimp_image_has_alpha (image);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
{
|
|
undo_desc = C_("undo-type", "Remove Floating Selection");
|
|
|
|
gimp_drawable_detach_floating_sel (gimp_layer_get_floating_sel_drawable (layer));
|
|
}
|
|
else
|
|
{
|
|
undo_desc = C_("undo-type", "Remove Layer");
|
|
}
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_layer_remove (image, undo_desc, layer,
|
|
gimp_layer_get_parent (layer),
|
|
gimp_item_get_index (GIMP_ITEM (layer)),
|
|
selected_layers);
|
|
|
|
g_object_ref (layer);
|
|
|
|
/* Make sure we're not caching any old selection info */
|
|
if (g_list_find (selected_layers, layer))
|
|
gimp_drawable_invalidate_boundary (GIMP_DRAWABLE (layer));
|
|
|
|
/* Remove layer and its children from the MRU layer stack. */
|
|
gimp_image_remove_from_layer_stack (image, layer);
|
|
|
|
new_selected = gimp_item_tree_remove_item (private->layers,
|
|
GIMP_ITEM (layer),
|
|
new_selected);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
{
|
|
/* If this was the floating selection, activate the underlying drawable
|
|
*/
|
|
floating_sel_activate_drawable (layer);
|
|
}
|
|
else if (selected_layers &&
|
|
(g_list_find (selected_layers, layer) ||
|
|
g_list_find_custom (selected_layers, layer,
|
|
(GCompareFunc) gimp_image_selected_is_descendant)))
|
|
{
|
|
gimp_image_set_selected_layers (image, new_selected);
|
|
}
|
|
|
|
gimp_item_end_move (GIMP_ITEM (layer), push_undo);
|
|
|
|
g_object_unref (layer);
|
|
g_list_free (selected_layers);
|
|
if (new_selected)
|
|
g_list_free (new_selected);
|
|
|
|
if (old_has_alpha != gimp_image_has_alpha (image))
|
|
private->flush_accum.alpha_changed = TRUE;
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_end (image);
|
|
}
|
|
|
|
void
|
|
gimp_image_add_layers (GimpImage *image,
|
|
GList *layers,
|
|
GimpLayer *parent,
|
|
gint position,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height,
|
|
const gchar *undo_desc)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *list;
|
|
gint layers_x = G_MAXINT;
|
|
gint layers_y = G_MAXINT;
|
|
gint layers_width = 0;
|
|
gint layers_height = 0;
|
|
gint offset_x;
|
|
gint offset_y;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (layers != NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* item and parent are type-checked in GimpItemTree
|
|
*/
|
|
if (! gimp_item_tree_get_insert_pos (private->layers,
|
|
(GimpItem *) layers->data,
|
|
(GimpItem **) &parent,
|
|
&position))
|
|
return;
|
|
|
|
for (list = layers; list; list = g_list_next (list))
|
|
{
|
|
GimpItem *item = GIMP_ITEM (list->data);
|
|
gint off_x, off_y;
|
|
|
|
gimp_item_get_offset (item, &off_x, &off_y);
|
|
|
|
layers_x = MIN (layers_x, off_x);
|
|
layers_y = MIN (layers_y, off_y);
|
|
|
|
layers_width = MAX (layers_width,
|
|
off_x + gimp_item_get_width (item) - layers_x);
|
|
layers_height = MAX (layers_height,
|
|
off_y + gimp_item_get_height (item) - layers_y);
|
|
}
|
|
|
|
offset_x = x + (width - layers_width) / 2 - layers_x;
|
|
offset_y = y + (height - layers_height) / 2 - layers_y;
|
|
|
|
gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_LAYER_ADD, undo_desc);
|
|
|
|
for (list = layers; list; list = g_list_next (list))
|
|
{
|
|
GimpItem *new_item = GIMP_ITEM (list->data);
|
|
|
|
gimp_item_translate (new_item, offset_x, offset_y, FALSE);
|
|
|
|
gimp_image_add_layer (image, GIMP_LAYER (new_item),
|
|
parent, position, TRUE);
|
|
position++;
|
|
}
|
|
|
|
if (layers)
|
|
gimp_image_set_selected_layers (image, layers);
|
|
|
|
gimp_image_undo_group_end (image);
|
|
}
|
|
|
|
/*
|
|
* gimp_image_store_item_set:
|
|
* @image:
|
|
* @set: (transfer full): a set of items which @images takes ownership
|
|
* of.
|
|
*
|
|
* Store a new set of @layers.
|
|
* If a set with the same name and type existed, this call will silently
|
|
* replace it with the new set of layers.
|
|
*/
|
|
void
|
|
gimp_image_store_item_set (GimpImage *image,
|
|
GimpItemList *set)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList **stored_sets;
|
|
GList *iter;
|
|
guint signal;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_ITEM_LIST (set));
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (gimp_item_list_get_item_type (set) == GIMP_TYPE_LAYER)
|
|
{
|
|
stored_sets = &private->stored_layer_sets;
|
|
signal = gimp_image_signals[LAYER_SETS_CHANGED];
|
|
}
|
|
else if (gimp_item_list_get_item_type (set) == GIMP_TYPE_CHANNEL)
|
|
{
|
|
stored_sets = &private->stored_channel_sets;
|
|
signal = gimp_image_signals[CHANNEL_SETS_CHANGED];
|
|
}
|
|
else if (gimp_item_list_get_item_type (set) == GIMP_TYPE_VECTORS)
|
|
{
|
|
stored_sets = &private->stored_vectors_sets;
|
|
signal = gimp_image_signals[VECTORS_SETS_CHANGED];
|
|
}
|
|
else
|
|
{
|
|
g_return_if_reached ();
|
|
}
|
|
|
|
for (iter = *stored_sets; iter; iter = iter->next)
|
|
{
|
|
gboolean is_pattern;
|
|
gboolean is_pattern2;
|
|
GimpSelectMethod pattern_syntax;
|
|
GimpSelectMethod pattern_syntax2;
|
|
|
|
is_pattern = gimp_item_list_is_pattern (iter->data, &pattern_syntax);
|
|
is_pattern2 = gimp_item_list_is_pattern (set, &pattern_syntax2);
|
|
|
|
/* Remove a previous item set of same type and name. */
|
|
if (is_pattern == is_pattern2 && (! is_pattern || pattern_syntax == pattern_syntax2) &&
|
|
g_strcmp0 (gimp_object_get_name (iter->data), gimp_object_get_name (set)) == 0)
|
|
break;
|
|
}
|
|
if (iter)
|
|
{
|
|
g_object_unref (iter->data);
|
|
*stored_sets = g_list_delete_link (*stored_sets, iter);
|
|
}
|
|
|
|
*stored_sets = g_list_prepend (*stored_sets, set);
|
|
g_signal_emit (image, signal, 0);
|
|
}
|
|
|
|
/*
|
|
* @gimp_image_unlink_item_set:
|
|
* @image:
|
|
* @link_name:
|
|
*
|
|
* Remove the set of layers named @link_name.
|
|
*
|
|
* Returns: %TRUE if the set was removed, %FALSE if no sets with this
|
|
* name existed.
|
|
*/
|
|
gboolean
|
|
gimp_image_unlink_item_set (GimpImage *image,
|
|
GimpItemList *set)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *found;
|
|
GList **stored_sets;
|
|
guint signal;
|
|
gboolean success;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (gimp_item_list_get_item_type (set) == GIMP_TYPE_LAYER)
|
|
{
|
|
stored_sets = &private->stored_layer_sets;
|
|
signal = gimp_image_signals[LAYER_SETS_CHANGED];
|
|
}
|
|
else if (gimp_item_list_get_item_type (set) == GIMP_TYPE_CHANNEL)
|
|
{
|
|
stored_sets = &private->stored_channel_sets;
|
|
signal = gimp_image_signals[CHANNEL_SETS_CHANGED];
|
|
}
|
|
else if (gimp_item_list_get_item_type (set) == GIMP_TYPE_VECTORS)
|
|
{
|
|
stored_sets = &private->stored_vectors_sets;
|
|
signal = gimp_image_signals[VECTORS_SETS_CHANGED];
|
|
}
|
|
else
|
|
{
|
|
g_return_val_if_reached (FALSE);
|
|
}
|
|
|
|
found = g_list_find (*stored_sets, set);
|
|
success = (found != NULL);
|
|
if (success)
|
|
{
|
|
*stored_sets = g_list_delete_link (*stored_sets, found);
|
|
g_object_unref (set);
|
|
g_signal_emit (image, signal, 0);
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
/*
|
|
* @gimp_image_get_stored_item_sets:
|
|
* @image:
|
|
* @item_type:
|
|
*
|
|
* Returns: (transfer none): the list of all the layer sets (which you
|
|
* should not modify). Order of items is relevant.
|
|
*/
|
|
GList *
|
|
gimp_image_get_stored_item_sets (GimpImage *image,
|
|
GType item_type)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
return private->stored_layer_sets;
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
return private->stored_channel_sets;
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
return private->stored_vectors_sets;
|
|
|
|
g_return_val_if_reached (FALSE);
|
|
}
|
|
|
|
/*
|
|
* @gimp_image_select_item_set:
|
|
* @image:
|
|
* @set:
|
|
*
|
|
* Replace currently selected layers in @image with the layers belonging
|
|
* to @set.
|
|
*/
|
|
void
|
|
gimp_image_select_item_set (GimpImage *image,
|
|
GimpItemList *set)
|
|
{
|
|
GList *items;
|
|
GError *error = NULL;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_ITEM_LIST (set));
|
|
|
|
items = gimp_item_list_get_items (set, &error);
|
|
|
|
if (! error)
|
|
{
|
|
GType item_type = gimp_item_list_get_item_type (set);
|
|
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
gimp_image_set_selected_layers (image, items);
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
gimp_image_set_selected_channels (image, items);
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
gimp_image_set_selected_vectors (image, items);
|
|
else
|
|
g_return_if_reached ();
|
|
}
|
|
|
|
g_list_free (items);
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
/*
|
|
* @gimp_image_add_item_set:
|
|
* @image:
|
|
* @set:
|
|
*
|
|
* Add the layers belonging to @set to the items currently selected in
|
|
* @image.
|
|
*/
|
|
void
|
|
gimp_image_add_item_set (GimpImage *image,
|
|
GimpItemList *set)
|
|
{
|
|
GList *items;
|
|
GError *error = NULL;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_ITEM_LIST (set));
|
|
|
|
items = gimp_item_list_get_items (set, &error);
|
|
|
|
if (! error)
|
|
{
|
|
GList *selected;
|
|
GList *iter;
|
|
GType item_type = gimp_item_list_get_item_type (set);
|
|
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
selected = gimp_image_get_selected_layers (image);
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
selected = gimp_image_get_selected_channels (image);
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
selected = gimp_image_get_selected_vectors (image);
|
|
else
|
|
g_return_if_reached ();
|
|
|
|
selected = g_list_copy (selected);
|
|
for (iter = items; iter; iter = iter->next)
|
|
{
|
|
if (! g_list_find (selected, iter->data))
|
|
selected = g_list_prepend (selected, iter->data);
|
|
}
|
|
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
gimp_image_set_selected_layers (image, selected);
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
gimp_image_set_selected_channels (image, selected);
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
gimp_image_set_selected_vectors (image, items);
|
|
|
|
g_list_free (selected);
|
|
}
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
/*
|
|
* @gimp_image_remove_item_set:
|
|
* @image:
|
|
* @set:
|
|
*
|
|
* Remove the layers belonging to the set named @link_name (which must
|
|
* exist) from the layers currently selected in @image.
|
|
*
|
|
* Returns: %TRUE if the selection change is done (even if it turned out
|
|
* selected layers stay the same), %FALSE if no sets with this
|
|
* name existed.
|
|
*/
|
|
void
|
|
gimp_image_remove_item_set (GimpImage *image,
|
|
GimpItemList *set)
|
|
{
|
|
GList *items;
|
|
GError *error = NULL;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_ITEM_LIST (set));
|
|
|
|
items = gimp_item_list_get_items (set, &error);
|
|
|
|
if (! error)
|
|
{
|
|
GList *selected;
|
|
GList *iter;
|
|
GType item_type = gimp_item_list_get_item_type (set);
|
|
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
selected = gimp_image_get_selected_layers (image);
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
selected = gimp_image_get_selected_channels (image);
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
selected = gimp_image_get_selected_vectors (image);
|
|
else
|
|
g_return_if_reached ();
|
|
|
|
selected = g_list_copy (selected);
|
|
for (iter = items; iter; iter = iter->next)
|
|
{
|
|
GList *remove;
|
|
|
|
if ((remove = g_list_find (selected, iter->data)))
|
|
selected = g_list_delete_link (selected, remove);
|
|
}
|
|
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
gimp_image_set_selected_layers (image, selected);
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
gimp_image_set_selected_channels (image, selected);
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
gimp_image_set_selected_vectors (image, items);
|
|
|
|
g_list_free (selected);
|
|
}
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
/*
|
|
* @gimp_image_intersect_item_set:
|
|
* @image:
|
|
* @set:
|
|
*
|
|
* Remove any layers from the layers currently selected in @image if
|
|
* they don't also belong to the set named @link_name (which must
|
|
* exist).
|
|
*
|
|
* Returns: %TRUE if the selection change is done (even if it turned out
|
|
* selected layers stay the same), %FALSE if no sets with this
|
|
* name existed.
|
|
*/
|
|
void
|
|
gimp_image_intersect_item_set (GimpImage *image,
|
|
GimpItemList *set)
|
|
{
|
|
GList *items;
|
|
GError *error = NULL;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_ITEM_LIST (set));
|
|
|
|
items = gimp_item_list_get_items (set, &error);
|
|
|
|
if (! error)
|
|
{
|
|
GList *selected;
|
|
GList *remove = NULL;
|
|
GList *iter;
|
|
GType item_type = gimp_item_list_get_item_type (set);
|
|
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
selected = gimp_image_get_selected_layers (image);
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
selected = gimp_image_get_selected_channels (image);
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
selected = gimp_image_get_selected_vectors (image);
|
|
else
|
|
g_return_if_reached ();
|
|
|
|
selected = g_list_copy (selected);
|
|
|
|
/* Remove items in selected but not in items. */
|
|
for (iter = selected; iter; iter = iter->next)
|
|
{
|
|
if (! g_list_find (items, iter->data))
|
|
remove = g_list_prepend (remove, iter);
|
|
}
|
|
for (iter = remove; iter; iter = iter->next)
|
|
selected = g_list_delete_link (selected, iter->data);
|
|
g_list_free (remove);
|
|
|
|
/* Finally select the intersection. */
|
|
if (item_type == GIMP_TYPE_LAYER)
|
|
gimp_image_set_selected_layers (image, selected);
|
|
else if (item_type == GIMP_TYPE_CHANNEL)
|
|
gimp_image_set_selected_channels (image, selected);
|
|
else if (item_type == GIMP_TYPE_VECTORS)
|
|
gimp_image_set_selected_vectors (image, items);
|
|
|
|
g_list_free (selected);
|
|
}
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
|
|
/* channels */
|
|
|
|
gboolean
|
|
gimp_image_add_channel (GimpImage *image,
|
|
GimpChannel *channel,
|
|
GimpChannel *parent,
|
|
gint position,
|
|
gboolean push_undo)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *channels;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* item and parent are type-checked in GimpItemTree
|
|
*/
|
|
if (! gimp_item_tree_get_insert_pos (private->channels,
|
|
(GimpItem *) channel,
|
|
(GimpItem **) &parent,
|
|
&position))
|
|
return FALSE;
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_channel_add (image, C_("undo-type", "Add Channel"),
|
|
channel,
|
|
gimp_image_get_selected_channels (image));
|
|
|
|
gimp_item_tree_add_item (private->channels, GIMP_ITEM (channel),
|
|
GIMP_ITEM (parent), position);
|
|
|
|
channels = g_list_prepend (NULL, channel);
|
|
gimp_image_set_selected_channels (image, channels);
|
|
g_list_free (channels);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_image_remove_channel (GimpImage *image,
|
|
GimpChannel *channel,
|
|
gboolean push_undo,
|
|
GList *new_selected)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *selected_channels;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_CHANNEL (channel));
|
|
g_return_if_fail (gimp_item_is_attached (GIMP_ITEM (channel)));
|
|
g_return_if_fail (gimp_item_get_image (GIMP_ITEM (channel)) == image);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_IMAGE_ITEM_REMOVE,
|
|
C_("undo-type", "Remove Channel"));
|
|
|
|
gimp_item_start_move (GIMP_ITEM (channel), push_undo);
|
|
|
|
if (gimp_drawable_get_floating_sel (GIMP_DRAWABLE (channel)))
|
|
{
|
|
if (! push_undo)
|
|
{
|
|
g_warning ("%s() was called from an undo function while the channel "
|
|
"had a floating selection. Please report this at "
|
|
"https://www.gimp.org/bugs/", G_STRFUNC);
|
|
return;
|
|
}
|
|
|
|
gimp_image_remove_layer (image,
|
|
gimp_drawable_get_floating_sel (GIMP_DRAWABLE (channel)),
|
|
TRUE, NULL);
|
|
}
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
selected_channels = gimp_image_get_selected_channels (image);
|
|
selected_channels = g_list_copy (selected_channels);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_channel_remove (image, C_("undo-type", "Remove Channel"), channel,
|
|
gimp_channel_get_parent (channel),
|
|
gimp_item_get_index (GIMP_ITEM (channel)),
|
|
selected_channels);
|
|
|
|
g_object_ref (channel);
|
|
|
|
new_selected = gimp_item_tree_remove_item (private->channels,
|
|
GIMP_ITEM (channel),
|
|
new_selected);
|
|
|
|
if (selected_channels &&
|
|
(g_list_find (selected_channels, channel) ||
|
|
g_list_find_custom (selected_channels, channel,
|
|
(GCompareFunc) gimp_image_selected_is_descendant)))
|
|
{
|
|
if (new_selected)
|
|
gimp_image_set_selected_channels (image, new_selected);
|
|
else
|
|
gimp_image_unset_selected_channels (image);
|
|
}
|
|
|
|
g_list_free (selected_channels);
|
|
|
|
gimp_item_end_move (GIMP_ITEM (channel), push_undo);
|
|
|
|
g_object_unref (channel);
|
|
if (new_selected)
|
|
g_list_free (new_selected);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_end (image);
|
|
}
|
|
|
|
|
|
/* vectors */
|
|
|
|
gboolean
|
|
gimp_image_add_vectors (GimpImage *image,
|
|
GimpVectors *vectors,
|
|
GimpVectors *parent,
|
|
gint position,
|
|
gboolean push_undo)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *list = NULL;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
/* item and parent are type-checked in GimpItemTree
|
|
*/
|
|
if (! gimp_item_tree_get_insert_pos (private->vectors,
|
|
(GimpItem *) vectors,
|
|
(GimpItem **) &parent,
|
|
&position))
|
|
return FALSE;
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_vectors_add (image, C_("undo-type", "Add Path"),
|
|
vectors,
|
|
gimp_image_get_selected_vectors (image));
|
|
|
|
gimp_item_tree_add_item (private->vectors, GIMP_ITEM (vectors),
|
|
GIMP_ITEM (parent), position);
|
|
|
|
if (vectors != NULL)
|
|
list = g_list_prepend (NULL, vectors);
|
|
|
|
gimp_image_set_selected_vectors (image, list);
|
|
|
|
g_list_free (list);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_image_remove_vectors (GimpImage *image,
|
|
GimpVectors *vectors,
|
|
gboolean push_undo,
|
|
GList *new_selected)
|
|
{
|
|
GimpImagePrivate *private;
|
|
GList *selected_vectors;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_VECTORS (vectors));
|
|
g_return_if_fail (gimp_item_is_attached (GIMP_ITEM (vectors)));
|
|
g_return_if_fail (gimp_item_get_image (GIMP_ITEM (vectors)) == image);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_IMAGE_ITEM_REMOVE,
|
|
C_("undo-type", "Remove Path"));
|
|
|
|
gimp_item_start_move (GIMP_ITEM (vectors), push_undo);
|
|
|
|
selected_vectors = gimp_image_get_selected_vectors (image);
|
|
selected_vectors = g_list_copy (selected_vectors);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_push_vectors_remove (image, C_("undo-type", "Remove Path"), vectors,
|
|
gimp_vectors_get_parent (vectors),
|
|
gimp_item_get_index (GIMP_ITEM (vectors)),
|
|
selected_vectors);
|
|
|
|
g_object_ref (vectors);
|
|
|
|
new_selected = gimp_item_tree_remove_item (private->vectors,
|
|
GIMP_ITEM (vectors),
|
|
new_selected);
|
|
|
|
if (selected_vectors &&
|
|
(g_list_find (selected_vectors, vectors) ||
|
|
g_list_find_custom (selected_vectors, vectors,
|
|
(GCompareFunc) gimp_image_selected_is_descendant)))
|
|
{
|
|
gimp_image_set_selected_vectors (image, new_selected);
|
|
}
|
|
|
|
g_list_free (selected_vectors);
|
|
|
|
gimp_item_end_move (GIMP_ITEM (vectors), push_undo);
|
|
|
|
g_object_unref (vectors);
|
|
if (new_selected)
|
|
g_list_free (new_selected);
|
|
|
|
if (push_undo)
|
|
gimp_image_undo_group_end (image);
|
|
}
|
|
|
|
/* hidden items */
|
|
|
|
/* Sometimes you want to create a channel or other types of drawables to
|
|
* work on them without adding them to the public trees and to be
|
|
* visible in the GUI. This is when you create hidden items. No undo
|
|
* steps will ever be created either for any processing on these items.
|
|
*
|
|
* Note that you are expected to manage the hidden items properly. In
|
|
* particular, once you are done with them, remove them with
|
|
* gimp_image_remove_hidden_item() and free them.
|
|
* @image is not assuming ownership of @item.
|
|
*/
|
|
gboolean
|
|
gimp_image_add_hidden_item (GimpImage *image,
|
|
GimpItem *item)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_ITEM (item), FALSE);
|
|
g_return_val_if_fail (! gimp_item_is_attached (item), FALSE);
|
|
g_return_val_if_fail (gimp_item_get_image (item) == image, FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
private->hidden_items = g_list_prepend (private->hidden_items, item);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_image_remove_hidden_item (GimpImage *image,
|
|
GimpItem *item)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
g_return_if_fail (GIMP_IS_ITEM (item));
|
|
g_return_if_fail (gimp_item_get_image (item) == image);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
g_return_if_fail (g_list_find (private->hidden_items, item) != NULL);
|
|
|
|
private->hidden_items = g_list_remove (private->hidden_items, item);
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_is_hidden_item (GimpImage *image,
|
|
GimpItem *item)
|
|
{
|
|
GimpImagePrivate *private;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_ITEM (item), FALSE);
|
|
g_return_val_if_fail (gimp_item_get_image (item) == image, FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
return (g_list_find (private->hidden_items, item) != NULL);
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_coords_in_active_pickable (GimpImage *image,
|
|
const GimpCoords *coords,
|
|
gboolean show_all,
|
|
gboolean sample_merged,
|
|
gboolean selected_only)
|
|
{
|
|
gint x, y;
|
|
gboolean in_pickable = FALSE;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
x = floor (coords->x);
|
|
y = floor (coords->y);
|
|
|
|
if (sample_merged)
|
|
{
|
|
if (show_all || (x >= 0 && x < gimp_image_get_width (image) &&
|
|
y >= 0 && y < gimp_image_get_height (image)))
|
|
{
|
|
in_pickable = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
GList *drawables = gimp_image_get_selected_drawables (image);
|
|
GList *iter;
|
|
|
|
for (iter = drawables; iter; iter = iter->next)
|
|
{
|
|
GimpItem *item = iter->data;
|
|
gint off_x, off_y;
|
|
gint d_x, d_y;
|
|
|
|
gimp_item_get_offset (item, &off_x, &off_y);
|
|
|
|
d_x = x - off_x;
|
|
d_y = y - off_y;
|
|
|
|
if (d_x >= 0 && d_x < gimp_item_get_width (item) &&
|
|
d_y >= 0 && d_y < gimp_item_get_height (item))
|
|
{
|
|
in_pickable = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
g_list_free (drawables);
|
|
}
|
|
|
|
if (in_pickable && selected_only)
|
|
{
|
|
GimpChannel *selection = gimp_image_get_mask (image);
|
|
|
|
if (! gimp_channel_is_empty (selection) &&
|
|
! gimp_pickable_get_opacity_at (GIMP_PICKABLE (selection),
|
|
x, y))
|
|
{
|
|
in_pickable = FALSE;
|
|
}
|
|
}
|
|
|
|
return in_pickable;
|
|
}
|
|
|
|
void
|
|
gimp_image_invalidate_previews (GimpImage *image)
|
|
{
|
|
GimpItemStack *layers;
|
|
GimpItemStack *channels;
|
|
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
layers = GIMP_ITEM_STACK (gimp_image_get_layers (image));
|
|
channels = GIMP_ITEM_STACK (gimp_image_get_channels (image));
|
|
|
|
gimp_item_stack_invalidate_previews (layers);
|
|
gimp_item_stack_invalidate_previews (channels);
|
|
}
|
|
|
|
/* Sets the image into a "converting" state, which is there to warn other code
|
|
* (such as shell render code) that the image properties might be in an
|
|
* inconsistent state. For instance when converting to another precision with
|
|
* gimp_image_convert_precision(), the babl format may be updated first, and the
|
|
* profile later, after all drawables are converted. Rendering the image
|
|
* in-between would at best render broken previews (at worst, crash, e.g.
|
|
* because we depend on allocated data which might have become too small).
|
|
*/
|
|
void
|
|
gimp_image_set_converting (GimpImage *image,
|
|
gboolean converting)
|
|
{
|
|
g_return_if_fail (GIMP_IS_IMAGE (image));
|
|
|
|
g_object_set (image,
|
|
"converting", converting,
|
|
NULL);
|
|
}
|
|
|
|
gboolean
|
|
gimp_image_get_converting (GimpImage *image)
|
|
{
|
|
GimpImagePrivate *private;
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), FALSE);
|
|
|
|
private = GIMP_IMAGE_GET_PRIVATE (image);
|
|
|
|
return private->converting;
|
|
}
|