/* PIKA - Photo and Image Kooker Application * a rebranding of The GNU Image Manipulation Program (created with heckimp) * A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio * * Original copyright, applying to most contents (license remains unchanged): * Copyright (C) 1995 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 . */ #include "config.h" #include #include #include "libpikaconfig/pikaconfig.h" #include "actions-types.h" #include "config/pikadialogconfig.h" #include "core/pika.h" #include "core/pikacontext.h" #include "core/pikaimage.h" #include "core/pikaimage-undo.h" #include "core/pikaitem.h" #include "core/pikaitemundo.h" #include "dialogs/dialogs.h" #include "dialogs/fill-dialog.h" #include "dialogs/stroke-dialog.h" #include "actions.h" #include "items-commands.h" #include "pika-intl.h" /* local function prototypes */ static void items_fill_callback (GtkWidget *dialog, GList *items, GList *drawables, PikaContext *context, PikaFillOptions *options, gpointer user_data); static void items_stroke_callback (GtkWidget *dialog, GList *items, GList *drawables, PikaContext *context, PikaStrokeOptions *options, gpointer user_data); /* public functions */ void items_visible_cmd_callback (PikaAction *action, GVariant *value, PikaImage *image, GList *items) { PikaUndo *undo; gboolean push_undo = TRUE; GList *start = NULL; GList *iter; gboolean visible = g_variant_get_boolean (value); gint n_items = 0; for (iter = items; iter; iter = iter->next) { if (visible && pika_item_get_visible (iter->data)) { /* If any of the items are already visible, we don't * toggle the selection visibility. */ return; } } for (iter = items; iter; iter = iter->next) if (visible != pika_item_get_visible (iter->data)) { if (start == NULL) start = iter; n_items++; } if (n_items == 0) { return; } else if (n_items == 1) { undo = pika_image_undo_can_compress (image, PIKA_TYPE_ITEM_UNDO, PIKA_UNDO_ITEM_VISIBILITY); if (undo && PIKA_ITEM_UNDO (undo)->item == PIKA_ITEM (start->data)) push_undo = FALSE; } else { /* TODO: undo groups cannot be compressed so far. */ pika_image_undo_group_start (image, PIKA_UNDO_GROUP_ITEM_VISIBILITY, "Item visibility"); } for (iter = start; iter; iter = iter->next) if (visible != pika_item_get_visible (iter->data)) pika_item_set_visible (iter->data, visible, push_undo); if (n_items != 1) pika_image_undo_group_end (image); pika_image_flush (image); } void items_lock_content_cmd_callback (PikaAction *action, GVariant *value, PikaImage *image, GList *items) { GList *locked_items = NULL; GList *iter; gchar *undo_label; gboolean locked = g_variant_get_boolean (value); for (iter = items; iter; iter = iter->next) if (pika_item_can_lock_content (iter->data)) { if (! locked && ! pika_item_get_lock_content (iter->data)) { /* When unlocking, we expect all selected items to be locked. */ g_list_free (locked_items); return; } else if (locked != pika_item_get_lock_content (iter->data)) { locked_items = g_list_prepend (locked_items, iter->data); } } if (! locked_items) return; if (locked) undo_label = _("Lock content"); else undo_label = _("Unlock content"); pika_image_undo_group_start (image, PIKA_UNDO_GROUP_ITEM_LOCK_CONTENTS, undo_label); for (iter = locked_items; iter; iter = iter->next) pika_item_set_lock_content (iter->data, locked, TRUE); pika_image_flush (image); pika_image_undo_group_end (image); g_list_free (locked_items); } void items_lock_position_cmd_callback (PikaAction *action, GVariant *value, PikaImage *image, GList *items) { GList *iter; GList *locked_items = NULL; gchar *undo_label; gboolean locked = g_variant_get_boolean (value); for (iter = items; iter; iter = iter->next) if (pika_item_can_lock_position (iter->data)) { if (! locked && ! pika_item_get_lock_position (iter->data)) { /* When unlocking, we expect all selected items to be locked. */ g_list_free (locked_items); return; } else if (locked != pika_item_get_lock_position (iter->data)) { locked_items = g_list_prepend (locked_items, iter->data); } } if (! locked_items) return; if (locked) undo_label = _("Lock position"); else undo_label = _("Unlock position"); pika_image_undo_group_start (image, PIKA_UNDO_GROUP_ITEM_LOCK_POSITION, undo_label); for (iter = locked_items; iter; iter = iter->next) pika_item_set_lock_position (iter->data, locked, TRUE); pika_image_flush (image); pika_image_undo_group_end (image); g_list_free (locked_items); } void items_color_tag_cmd_callback (PikaAction *action, PikaImage *image, GList *items, PikaColorTag color_tag) { PikaUndo *undo; gboolean push_undo = TRUE; GList *iter; if (g_list_length (items) == 1) { undo = pika_image_undo_can_compress (image, PIKA_TYPE_ITEM_UNDO, PIKA_UNDO_ITEM_COLOR_TAG); if (undo && PIKA_ITEM_UNDO (undo)->item == PIKA_ITEM (items->data)) push_undo = FALSE; } else { /* TODO: undo groups cannot be compressed so far. */ pika_image_undo_group_start (image, PIKA_UNDO_GROUP_ITEM_PROPERTIES, "Item color tag"); } for (iter = items; iter; iter = iter->next) if (color_tag != pika_item_get_color_tag (iter->data)) pika_item_set_color_tag (iter->data, color_tag, push_undo); if (g_list_length (items) != 1) pika_image_undo_group_end (image); pika_image_flush (image); } void items_fill_cmd_callback (PikaAction *action, PikaImage *image, GList *items, const gchar *dialog_title, const gchar *dialog_icon_name, const gchar *dialog_help_id, gpointer data) { GList *drawables; GtkWidget *dialog; GtkWidget *widget; return_if_no_widget (widget, data); drawables = pika_image_get_selected_drawables (image); if (! drawables) { pika_message_literal (image->pika, G_OBJECT (widget), PIKA_MESSAGE_WARNING, _("There are no selected items to fill.")); return; } dialog = fill_dialog_new (items, drawables, action_data_get_context (data), dialog_title, dialog_icon_name, dialog_help_id, widget, PIKA_DIALOG_CONFIG (image->pika->config)->fill_options, items_fill_callback, NULL); gtk_window_present (GTK_WINDOW (dialog)); g_list_free (drawables); } void items_fill_last_vals_cmd_callback (PikaAction *action, PikaImage *image, GList *items, gpointer data) { GList *drawables; PikaDialogConfig *config; GtkWidget *widget; GError *error = NULL; return_if_no_widget (widget, data); drawables = pika_image_get_selected_drawables (image); if (! drawables) { pika_message_literal (image->pika, G_OBJECT (widget), PIKA_MESSAGE_WARNING, _("There are no selected layers or channels to fill.")); return; } config = PIKA_DIALOG_CONFIG (image->pika->config); pika_image_undo_group_start (image, PIKA_UNDO_GROUP_DRAWABLE_MOD, "Fill"); for (GList *iter = items; iter; iter = iter->next) if (! pika_item_fill (iter->data, drawables, config->fill_options, TRUE, NULL, &error)) { pika_message_literal (image->pika, G_OBJECT (widget), PIKA_MESSAGE_WARNING, error->message); g_clear_error (&error); break; } pika_image_undo_group_end (image); pika_image_flush (image); g_list_free (drawables); } void items_stroke_cmd_callback (PikaAction *action, PikaImage *image, GList *items, const gchar *dialog_title, const gchar *dialog_icon_name, const gchar *dialog_help_id, gpointer data) { GList *drawables; GtkWidget *dialog; GtkWidget *widget; return_if_no_widget (widget, data); drawables = pika_image_get_selected_drawables (image); if (! drawables) { pika_message_literal (image->pika, G_OBJECT (widget), PIKA_MESSAGE_WARNING, _("There are no selected layers or channels to stroke to.")); return; } dialog = stroke_dialog_new (items, drawables, action_data_get_context (data), dialog_title, dialog_icon_name, dialog_help_id, widget, PIKA_DIALOG_CONFIG (image->pika->config)->stroke_options, items_stroke_callback, NULL); gtk_window_present (GTK_WINDOW (dialog)); g_list_free (drawables); } void items_stroke_last_vals_cmd_callback (PikaAction *action, PikaImage *image, GList *items, gpointer data) { GList *drawables; PikaDialogConfig *config; GtkWidget *widget; GError *error = NULL; return_if_no_widget (widget, data); drawables = pika_image_get_selected_drawables (image); if (! drawables) { pika_message_literal (image->pika, G_OBJECT (widget), PIKA_MESSAGE_WARNING, _("There are no selected layers or channels to stroke to.")); return; } config = PIKA_DIALOG_CONFIG (image->pika->config); pika_image_undo_group_start (image, PIKA_UNDO_GROUP_DRAWABLE_MOD, "Stroke"); for (GList *iter = items; iter; iter = iter->next) if (! pika_item_stroke (iter->data, drawables, action_data_get_context (data), config->stroke_options, NULL, TRUE, NULL, &error)) { pika_message_literal (image->pika, G_OBJECT (widget), PIKA_MESSAGE_WARNING, error->message); g_clear_error (&error); break; } pika_image_undo_group_end (image); pika_image_flush (image); g_list_free (drawables); } /* private functions */ static void items_fill_callback (GtkWidget *dialog, GList *items, GList *drawables, PikaContext *context, PikaFillOptions *options, gpointer user_data) { PikaDialogConfig *config = PIKA_DIALOG_CONFIG (context->pika->config); PikaImage *image = pika_item_get_image (items->data); GError *error = NULL; pika_config_sync (G_OBJECT (options), G_OBJECT (config->fill_options), 0); pika_image_undo_group_start (image, PIKA_UNDO_GROUP_DRAWABLE_MOD, "Fill"); for (GList *iter = items; iter; iter = iter->next) if (! pika_item_fill (iter->data, drawables, options, TRUE, NULL, &error)) { pika_message_literal (context->pika, G_OBJECT (dialog), PIKA_MESSAGE_WARNING, error ? error->message : "NULL"); g_clear_error (&error); break; } pika_image_undo_group_end (image); pika_image_flush (image); gtk_widget_destroy (dialog); } static void items_stroke_callback (GtkWidget *dialog, GList *items, GList *drawables, PikaContext *context, PikaStrokeOptions *options, gpointer data) { PikaDialogConfig *config = PIKA_DIALOG_CONFIG (context->pika->config); PikaImage *image = pika_item_get_image (items->data); GError *error = NULL; pika_config_sync (G_OBJECT (options), G_OBJECT (config->stroke_options), 0); pika_image_undo_group_start (image, PIKA_UNDO_GROUP_DRAWABLE_MOD, "Stroke"); for (GList *iter = items; iter; iter = iter->next) if (! pika_item_stroke (iter->data, drawables, context, options, NULL, TRUE, NULL, &error)) { pika_message_literal (context->pika, G_OBJECT (dialog), PIKA_MESSAGE_WARNING, error ? error->message : "NULL"); g_clear_error (&error); break; } pika_image_undo_group_end (image); pika_image_flush (image); gtk_widget_destroy (dialog); }