/* 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 * * pikasavedialog.c * Copyright (C) 2015 Jehan * * 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 "libpikabase/pikabase.h" #include "libpikawidgets/pikawidgets.h" #include "widgets-types.h" #include "core/pika.h" #include "core/pika-utils.h" #include "core/pikaimage.h" #include "core/pikaimage-metadata.h" #include "file/pika-file.h" #include "pikahelp-ids.h" #include "pikasavedialog.h" #include "pika-intl.h" typedef struct _PikaSaveDialogState PikaSaveDialogState; struct _PikaSaveDialogState { gchar *filter_name; gboolean compression; }; static void pika_save_dialog_constructed (GObject *object); static void pika_save_dialog_save_state (PikaFileDialog *dialog, const gchar *state_name); static void pika_save_dialog_load_state (PikaFileDialog *dialog, const gchar *state_name); static void pika_save_dialog_add_extra_widgets (PikaSaveDialog *dialog); static void pika_save_dialog_compression_toggled (GtkToggleButton *button, PikaSaveDialog *dialog); static PikaSaveDialogState * pika_save_dialog_get_state (PikaSaveDialog *dialog); static void pika_save_dialog_set_state (PikaSaveDialog *dialog, PikaSaveDialogState *state); static void pika_save_dialog_state_destroy (PikaSaveDialogState *state); G_DEFINE_TYPE (PikaSaveDialog, pika_save_dialog, PIKA_TYPE_FILE_DIALOG) #define parent_class pika_save_dialog_parent_class static void pika_save_dialog_class_init (PikaSaveDialogClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); PikaFileDialogClass *fd_class = PIKA_FILE_DIALOG_CLASS (klass); object_class->constructed = pika_save_dialog_constructed; fd_class->save_state = pika_save_dialog_save_state; fd_class->load_state = pika_save_dialog_load_state; } static void pika_save_dialog_init (PikaSaveDialog *dialog) { } static void pika_save_dialog_constructed (GObject *object) { PikaSaveDialog *dialog = PIKA_SAVE_DIALOG (object); /* PikaFileDialog's constructed() is doing a few initialization * common to all file dialogs. */ G_OBJECT_CLASS (parent_class)->constructed (object); pika_save_dialog_add_extra_widgets (dialog); } static void pika_save_dialog_save_state (PikaFileDialog *dialog, const gchar *state_name) { g_object_set_data_full (G_OBJECT (dialog->pika), state_name, pika_save_dialog_get_state (PIKA_SAVE_DIALOG (dialog)), (GDestroyNotify) pika_save_dialog_state_destroy); } static void pika_save_dialog_load_state (PikaFileDialog *dialog, const gchar *state_name) { PikaSaveDialogState *state; state = g_object_get_data (G_OBJECT (dialog->pika), state_name); if (state) pika_save_dialog_set_state (PIKA_SAVE_DIALOG (dialog), state); } /* public functions */ GtkWidget * pika_save_dialog_new (Pika *pika) { g_return_val_if_fail (PIKA_IS_PIKA (pika), NULL); return g_object_new (PIKA_TYPE_SAVE_DIALOG, "pika", pika, "title", _("Save Image"), "role", "pika-file-save", "help-id", PIKA_HELP_FILE_SAVE, "ok-button-label", _("_Save"), "automatic-label", _("By Extension"), "automatic-help-id", PIKA_HELP_FILE_SAVE_BY_EXTENSION, "action", GTK_FILE_CHOOSER_ACTION_SAVE, "file-procs", PIKA_FILE_PROCEDURE_GROUP_SAVE, "file-procs-all-images", PIKA_FILE_PROCEDURE_GROUP_EXPORT, "file-filter-label", _("All XCF images"), NULL); } void pika_save_dialog_set_image (PikaSaveDialog *dialog, PikaImage *image, gboolean save_a_copy, gboolean close_after_saving, PikaObject *display) { PikaFileDialog *file_dialog; GtkWidget *compression_toggle; GFile *dir_file = NULL; GFile *name_file = NULL; GFile *ext_file = NULL; gchar *basename; const gchar *version_string; gint rle_version; gint zlib_version; g_return_if_fail (PIKA_IS_SAVE_DIALOG (dialog)); g_return_if_fail (PIKA_IS_IMAGE (image)); file_dialog = PIKA_FILE_DIALOG (dialog); file_dialog->image = image; dialog->save_a_copy = save_a_copy; dialog->close_after_saving = close_after_saving; dialog->display_to_close = display; pika_file_dialog_set_file_proc (file_dialog, NULL); /* * Priority of default paths for Save: * * 1. Last Save a copy-path (applies only to Save a copy) * 2. Last Save path * 3. Path of source XCF * 4. Path of Import source * 5. Last Save path of any PIKA document * 6. The default path (usually the OS 'Documents' path) */ if (save_a_copy) dir_file = pika_image_get_save_a_copy_file (image); if (! dir_file) dir_file = pika_image_get_file (image); if (! dir_file) dir_file = g_object_get_data (G_OBJECT (image), "pika-image-source-file"); if (! dir_file) dir_file = pika_image_get_imported_file (image); if (! dir_file) dir_file = g_object_get_data (G_OBJECT (file_dialog->pika), PIKA_FILE_SAVE_LAST_FILE_KEY); if (! dir_file) dir_file = pika_file_dialog_get_default_folder (file_dialog); /* Priority of default basenames for Save: * * 1. Last Save a copy-name (applies only to Save a copy) * 2. Last Save name * 3. Last Export name * 3. The source image path * 3. 'Untitled' */ if (save_a_copy) name_file = pika_image_get_save_a_copy_file (image); if (! name_file) name_file = pika_image_get_file (image); if (! name_file) name_file = pika_image_get_exported_file (image); if (! name_file) name_file = pika_image_get_imported_file (image); if (! name_file) name_file = pika_image_get_untitled_file (image); /* Priority of default type/extension for Save: * * 1. Type of last Save * 2. .xcf (which we don't explicitly append) */ ext_file = pika_image_get_file (image); if (ext_file) g_object_ref (ext_file); else ext_file = g_file_new_for_uri ("file:///we/only/care/about/extension.xcf"); pika_image_get_xcf_version (image, FALSE, &rle_version, &version_string, NULL); pika_image_get_xcf_version (image, TRUE, &zlib_version, NULL, NULL); if (rle_version != zlib_version) { GtkWidget *label; gchar *text; text = g_strdup_printf (_("Keep compression disabled to make the XCF " "file readable by %s and later."), version_string); label = gtk_label_new (text); gtk_label_set_xalign (GTK_LABEL (label), 0.0); pika_label_set_attributes (GTK_LABEL (label), PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC, -1); gtk_container_add (GTK_CONTAINER (dialog->compression_frame), label); gtk_widget_show (label); g_free (text); } compression_toggle = gtk_frame_get_label_widget (GTK_FRAME (dialog->compression_frame)); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (compression_toggle), pika_image_get_xcf_compression (image)); /* Force a "toggled" signal since gtk_toggle_button_set_active() won't * send it if the button status doesn't change. */ gtk_toggle_button_toggled (GTK_TOGGLE_BUTTON (compression_toggle)); if (ext_file) { GFile *tmp_file = pika_file_with_new_extension (name_file, ext_file); basename = g_path_get_basename (pika_file_get_utf8_name (tmp_file)); g_object_unref (tmp_file); g_object_unref (ext_file); } else { basename = g_path_get_basename (pika_file_get_utf8_name (name_file)); } if (g_file_query_file_type (dir_file, G_FILE_QUERY_INFO_NONE, NULL) == G_FILE_TYPE_DIRECTORY) { gtk_file_chooser_set_current_folder_file (GTK_FILE_CHOOSER (dialog), dir_file, NULL); } else { GFile *parent_file = g_file_get_parent (dir_file); gtk_file_chooser_set_current_folder_file (GTK_FILE_CHOOSER (dialog), parent_file, NULL); g_object_unref (parent_file); } gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), basename); g_free (basename); } /* private functions */ static void pika_save_dialog_add_extra_widgets (PikaSaveDialog *dialog) { GtkWidget *label; GtkWidget *reasons; GtkWidget *compression_toggle; /* Compression toggle. */ compression_toggle = gtk_check_button_new_with_mnemonic (_("Save this _XCF file with better but slower compression")); gtk_widget_set_tooltip_text (compression_toggle, _("On edge cases, better compression algorithms might still " "end up on bigger file size; manual check recommended")); dialog->compression_frame = pika_frame_new (NULL); gtk_frame_set_label_widget (GTK_FRAME (dialog->compression_frame), compression_toggle); gtk_widget_show (compression_toggle); pika_file_dialog_add_extra_widget (PIKA_FILE_DIALOG (dialog), dialog->compression_frame, FALSE, FALSE, 0); gtk_widget_show (dialog->compression_frame); /* Additional information explaining file compatibility things */ dialog->compat_info = gtk_expander_new (NULL); label = gtk_label_new (""); gtk_expander_set_label_widget (GTK_EXPANDER (dialog->compat_info), label); gtk_widget_show (label); gtk_label_set_xalign (GTK_LABEL (label), 0.0); reasons = gtk_text_view_new (); gtk_text_view_set_editable (GTK_TEXT_VIEW (reasons), FALSE); gtk_text_view_set_cursor_visible (GTK_TEXT_VIEW (reasons), FALSE); gtk_container_add (GTK_CONTAINER (dialog->compat_info), reasons); gtk_widget_show (reasons); pika_file_dialog_add_extra_widget (PIKA_FILE_DIALOG (dialog), dialog->compat_info, FALSE, FALSE, 0); gtk_widget_show (dialog->compat_info); g_signal_connect (compression_toggle, "toggled", G_CALLBACK (pika_save_dialog_compression_toggled), dialog); } static void pika_save_dialog_compression_toggled (GtkToggleButton *button, PikaSaveDialog *dialog) { const gchar *version_string = NULL; PikaFileDialog *file_dialog = PIKA_FILE_DIALOG (dialog); gchar *compat_hint = NULL; gchar *reason = NULL; GtkWidget *widget; GtkTextBuffer *text_buffer; gint version; if (! file_dialog->image) return; dialog->compression = gtk_toggle_button_get_active (button); if (dialog->compression) pika_image_get_xcf_version (file_dialog->image, TRUE, &version, &version_string, &reason); else pika_image_get_xcf_version (file_dialog->image, FALSE, &version, &version_string, &reason); /* Only show compatibility information for PIKA over 2.6. The reason * is mostly that we don't have details to make a compatibility list * with this older version. * It's anyway so prehistorical that we are not really caring about * compatibility with older version. */ if (version <= 206) gtk_widget_hide (dialog->compat_info); else gtk_widget_show (dialog->compat_info); /* Set the compatibility label. */ compat_hint = g_strdup_printf (_("The image uses features from %s and " "won't be readable by older PIKA versions."), version_string); if (pika_image_get_metadata (file_dialog->image)) { gchar *temp_hint; temp_hint = g_strconcat (compat_hint, "\n", _("Metadata won't be visible in PIKA " "older than version 2.10."), NULL); g_free (compat_hint); compat_hint = temp_hint; } widget = gtk_expander_get_label_widget (GTK_EXPANDER (dialog->compat_info)); gtk_label_set_text (GTK_LABEL (widget), compat_hint); g_free (compat_hint); /* Fill in the details (list of compatibility reasons). */ widget = gtk_bin_get_child (GTK_BIN (dialog->compat_info)); text_buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)); gtk_text_buffer_set_text (text_buffer, reason ? reason : "", -1); if (reason) g_free (reason); } static PikaSaveDialogState * pika_save_dialog_get_state (PikaSaveDialog *dialog) { PikaSaveDialogState *state; GtkFileFilter *filter; state = g_slice_new0 (PikaSaveDialogState); filter = gtk_file_chooser_get_filter (GTK_FILE_CHOOSER (dialog)); if (filter) state->filter_name = g_strdup (gtk_file_filter_get_name (filter)); state->compression = dialog->compression; return state; } static void pika_save_dialog_set_state (PikaSaveDialog *dialog, PikaSaveDialogState *state) { if (state->filter_name) { GSList *filters; GSList *list; filters = gtk_file_chooser_list_filters (GTK_FILE_CHOOSER (dialog)); for (list = filters; list; list = list->next) { GtkFileFilter *filter = GTK_FILE_FILTER (list->data); const gchar *name = gtk_file_filter_get_name (filter); if (name && strcmp (state->filter_name, name) == 0) { gtk_file_chooser_set_filter (GTK_FILE_CHOOSER (dialog), filter); break; } } g_slist_free (filters); } dialog->compression = state->compression; } static void pika_save_dialog_state_destroy (PikaSaveDialogState *state) { g_free (state->filter_name); g_slice_free (PikaSaveDialogState, state); }