mirror of
https://gitlab.gnome.org/GNOME/gimp.git
synced 2025-10-06 01:12:40 +02:00
After much thought, tests and discussions with Aryeom, we decided adding back an active item concept additionally to the selected items ones is a bad idea as it makes only usage unecessarily complex. We will just have selected layers. Some kind of operations will work when more than one item (layers, channels, vectors) are selected while others will require exacty one item. In particular, let's replace instances of gimp_image_(s|g)et_active_*() by corresponding gimp_image_(s|g)et_selected_*(). Also replace single item in various undo classes by GList of items. Also "active-*-changed" GimpImage signals are no more, fully replaced by "selected-*s-changed". This is still work-in-progress.
230 lines
8.0 KiB
C
230 lines
8.0 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 <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <gegl.h>
|
|
|
|
#include "libgimpbase/gimpbase.h"
|
|
|
|
#include "core-types.h"
|
|
|
|
#include "gimpimage.h"
|
|
#include "gimpchannel.h"
|
|
#include "gimpchannelundo.h"
|
|
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_PREV_PARENT,
|
|
PROP_PREV_POSITION,
|
|
PROP_PREV_CHANNELS
|
|
};
|
|
|
|
|
|
static void gimp_channel_undo_constructed (GObject *object);
|
|
static void gimp_channel_undo_finalize (GObject *object);
|
|
static void gimp_channel_undo_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_channel_undo_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
|
|
static gint64 gimp_channel_undo_get_memsize (GimpObject *object,
|
|
gint64 *gui_size);
|
|
|
|
static void gimp_channel_undo_pop (GimpUndo *undo,
|
|
GimpUndoMode undo_mode,
|
|
GimpUndoAccumulator *accum);
|
|
|
|
|
|
G_DEFINE_TYPE (GimpChannelUndo, gimp_channel_undo, GIMP_TYPE_ITEM_UNDO)
|
|
|
|
#define parent_class gimp_channel_undo_parent_class
|
|
|
|
|
|
static void
|
|
gimp_channel_undo_class_init (GimpChannelUndoClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GimpObjectClass *gimp_object_class = GIMP_OBJECT_CLASS (klass);
|
|
GimpUndoClass *undo_class = GIMP_UNDO_CLASS (klass);
|
|
|
|
object_class->constructed = gimp_channel_undo_constructed;
|
|
object_class->finalize = gimp_channel_undo_finalize;
|
|
object_class->set_property = gimp_channel_undo_set_property;
|
|
object_class->get_property = gimp_channel_undo_get_property;
|
|
|
|
gimp_object_class->get_memsize = gimp_channel_undo_get_memsize;
|
|
|
|
undo_class->pop = gimp_channel_undo_pop;
|
|
|
|
g_object_class_install_property (object_class, PROP_PREV_PARENT,
|
|
g_param_spec_object ("prev-parent",
|
|
NULL, NULL,
|
|
GIMP_TYPE_CHANNEL,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
g_object_class_install_property (object_class, PROP_PREV_POSITION,
|
|
g_param_spec_int ("prev-position",
|
|
NULL, NULL,
|
|
0, G_MAXINT, 0,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
g_object_class_install_property (object_class, PROP_PREV_CHANNELS,
|
|
g_param_spec_pointer ("prev-channels",
|
|
NULL, NULL,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
}
|
|
|
|
static void
|
|
gimp_channel_undo_init (GimpChannelUndo *undo)
|
|
{
|
|
undo->prev_channels = NULL;
|
|
}
|
|
|
|
static void
|
|
gimp_channel_undo_constructed (GObject *object)
|
|
{
|
|
G_OBJECT_CLASS (parent_class)->constructed (object);
|
|
|
|
gimp_assert (GIMP_IS_CHANNEL (GIMP_ITEM_UNDO (object)->item));
|
|
}
|
|
|
|
static void
|
|
gimp_channel_undo_finalize (GObject *object)
|
|
{
|
|
GimpChannelUndo *channel_undo = GIMP_CHANNEL_UNDO (object);
|
|
|
|
g_clear_pointer (&channel_undo->prev_channels, g_list_free);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gimp_channel_undo_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpChannelUndo *channel_undo = GIMP_CHANNEL_UNDO (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_PREV_PARENT:
|
|
channel_undo->prev_parent = g_value_get_object (value);
|
|
break;
|
|
case PROP_PREV_POSITION:
|
|
channel_undo->prev_position = g_value_get_int (value);
|
|
break;
|
|
case PROP_PREV_CHANNELS:
|
|
channel_undo->prev_channels = g_list_copy (g_value_get_pointer (value));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_channel_undo_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpChannelUndo *channel_undo = GIMP_CHANNEL_UNDO (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_PREV_PARENT:
|
|
g_value_set_object (value, channel_undo->prev_parent);
|
|
break;
|
|
case PROP_PREV_POSITION:
|
|
g_value_set_int (value, channel_undo->prev_position);
|
|
break;
|
|
case PROP_PREV_CHANNELS:
|
|
g_value_set_pointer (value, channel_undo->prev_channels);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static gint64
|
|
gimp_channel_undo_get_memsize (GimpObject *object,
|
|
gint64 *gui_size)
|
|
{
|
|
GimpItemUndo *item_undo = GIMP_ITEM_UNDO (object);
|
|
gint64 memsize = 0;
|
|
|
|
if (! gimp_item_is_attached (item_undo->item))
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (item_undo->item),
|
|
gui_size);
|
|
|
|
return memsize + GIMP_OBJECT_CLASS (parent_class)->get_memsize (object,
|
|
gui_size);
|
|
}
|
|
|
|
static void
|
|
gimp_channel_undo_pop (GimpUndo *undo,
|
|
GimpUndoMode undo_mode,
|
|
GimpUndoAccumulator *accum)
|
|
{
|
|
GimpChannelUndo *channel_undo = GIMP_CHANNEL_UNDO (undo);
|
|
GimpChannel *channel = GIMP_CHANNEL (GIMP_ITEM_UNDO (undo)->item);
|
|
|
|
GIMP_UNDO_CLASS (parent_class)->pop (undo, undo_mode, accum);
|
|
|
|
if ((undo_mode == GIMP_UNDO_MODE_UNDO &&
|
|
undo->undo_type == GIMP_UNDO_CHANNEL_ADD) ||
|
|
(undo_mode == GIMP_UNDO_MODE_REDO &&
|
|
undo->undo_type == GIMP_UNDO_CHANNEL_REMOVE))
|
|
{
|
|
/* remove channel */
|
|
|
|
/* record the current parent and position */
|
|
channel_undo->prev_parent = gimp_channel_get_parent (channel);
|
|
channel_undo->prev_position = gimp_item_get_index (GIMP_ITEM (channel));
|
|
|
|
gimp_image_remove_channel (undo->image, channel, FALSE,
|
|
channel_undo->prev_channels);
|
|
}
|
|
else
|
|
{
|
|
/* restore channel */
|
|
|
|
/* record the active channel */
|
|
g_clear_pointer (&channel_undo->prev_channels, g_list_free);
|
|
channel_undo->prev_channels = g_list_copy (gimp_image_get_selected_channels (undo->image));
|
|
|
|
gimp_image_add_channel (undo->image, channel,
|
|
channel_undo->prev_parent,
|
|
channel_undo->prev_position, FALSE);
|
|
}
|
|
}
|