/* 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-1999 Spencer Kimball and Peter Mattis * * pikaiconpicker.c * Copyright (C) 2011 Michael Natterer * 2012 Daniel Sabo * * 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/pikalist.h" #include "core/pikacontext.h" #include "core/pikatemplate.h" #include "core/pikaviewable.h" #include "pikacontainerpopup.h" #include "pikaiconpicker.h" #include "pikaview.h" #include "pikaviewablebutton.h" #include "pika-intl.h" enum { PROP_0, PROP_PIKA, PROP_ICON_NAME, PROP_ICON_PIXBUF }; typedef struct _PikaIconPickerPrivate PikaIconPickerPrivate; struct _PikaIconPickerPrivate { Pika *pika; gchar *icon_name; GdkPixbuf *icon_pixbuf; PikaViewable *preview; PikaContainer *icon_name_container; PikaContext *icon_name_context; PikaObject *null_template_object; GtkWidget *right_click_menu; GtkWidget *menu_item_file_icon; GtkWidget *menu_item_name_icon; GtkWidget *menu_item_copy; GtkWidget *menu_item_paste; }; #define GET_PRIVATE(picker) \ ((PikaIconPickerPrivate *) pika_icon_picker_get_instance_private ((PikaIconPicker *) (picker))) static void pika_icon_picker_constructed (GObject *object); static void pika_icon_picker_finalize (GObject *object); static void pika_icon_picker_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec); static void pika_icon_picker_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec); static void pika_icon_picker_icon_changed (PikaContext *context, PikaTemplate *template, PikaIconPicker *picker); static void pika_icon_picker_clicked (GtkWidget *widget, GdkEventButton *event, gpointer data); static void pika_icon_picker_menu_from_file (GtkWidget *widget, GdkEventButton *event, gpointer data); static void pika_icon_picker_menu_from_name (GtkWidget *widget, GdkEventButton *event, gpointer data); static void pika_icon_picker_menu_paste (GtkWidget *widget, GdkEventButton *event, gpointer data); static void pika_icon_picker_menu_copy (GtkWidget *widget, GdkEventButton *event, gpointer data); G_DEFINE_TYPE_WITH_PRIVATE (PikaIconPicker, pika_icon_picker, GTK_TYPE_BOX) #define parent_class pika_icon_picker_parent_class static void pika_icon_picker_class_init (PikaIconPickerClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->constructed = pika_icon_picker_constructed; object_class->finalize = pika_icon_picker_finalize; object_class->set_property = pika_icon_picker_set_property; object_class->get_property = pika_icon_picker_get_property; g_object_class_install_property (object_class, PROP_PIKA, g_param_spec_object ("pika", NULL, NULL, PIKA_TYPE_PIKA, PIKA_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (object_class, PROP_ICON_NAME, g_param_spec_string ("icon-name", NULL, NULL, "pika-toilet-paper", PIKA_PARAM_READWRITE | G_PARAM_CONSTRUCT)); g_object_class_install_property (object_class, PROP_ICON_PIXBUF, g_param_spec_object ("icon-pixbuf", NULL, NULL, GDK_TYPE_PIXBUF, PIKA_PARAM_READWRITE)); } static void pika_icon_picker_init (PikaIconPicker *picker) { PikaIconPickerPrivate *private = GET_PRIVATE (picker); gtk_orientable_set_orientation (GTK_ORIENTABLE (picker), GTK_ORIENTATION_HORIZONTAL); private->preview = g_object_new (PIKA_TYPE_VIEWABLE, "icon-name", private->icon_name, "icon-pixbuf", private->icon_pixbuf, NULL); } static void pika_icon_picker_constructed (GObject *object) { PikaIconPicker *picker = PIKA_ICON_PICKER (object); PikaIconPickerPrivate *private = GET_PRIVATE (object); GtkWidget *button; GtkWidget *viewable_view; GList *icon_list; GList *list; G_OBJECT_CLASS (parent_class)->constructed (object); pika_assert (PIKA_IS_PIKA (private->pika)); /* Set up the icon picker */ private->icon_name_container = pika_list_new (PIKA_TYPE_TEMPLATE, FALSE); private->icon_name_context = pika_context_new (private->pika, "foo", NULL); g_signal_connect (private->icon_name_context, "template-changed", G_CALLBACK (pika_icon_picker_icon_changed), picker); icon_list = gtk_icon_theme_list_icons (gtk_icon_theme_get_default (), NULL); icon_list = g_list_sort (icon_list, (GCompareFunc) g_strcmp0); icon_list = g_list_reverse (icon_list); for (list = icon_list; list; list = g_list_next (list)) { PikaObject *object = g_object_new (PIKA_TYPE_TEMPLATE, "name", list->data, "icon-name", list->data, NULL); pika_container_add (private->icon_name_container, object); g_object_unref (object); if (private->icon_name && strcmp (list->data, private->icon_name) == 0) pika_context_set_template (private->icon_name_context, PIKA_TEMPLATE (object)); } /* An extra template object, use to make all icons clickable when a * pixbuf icon is set. */ private->null_template_object = g_object_new (PIKA_TYPE_TEMPLATE, "name", "", "icon-name", "", NULL); if (private->icon_pixbuf) { pika_context_set_template (private->icon_name_context, PIKA_TEMPLATE (private->null_template_object)); } g_list_free_full (icon_list, (GDestroyNotify) g_free); /* Set up preview button */ button = gtk_button_new (); gtk_box_pack_start (GTK_BOX (picker), button, FALSE, FALSE, 0); gtk_widget_show (button); g_signal_connect (button, "button-press-event", G_CALLBACK (pika_icon_picker_clicked), object); viewable_view = pika_view_new (private->icon_name_context, private->preview, PIKA_VIEW_SIZE_SMALL, 0, FALSE); gtk_container_add (GTK_CONTAINER (button), GTK_WIDGET (viewable_view)); gtk_widget_show (viewable_view); /* Set up button menu */ private->right_click_menu = gtk_menu_new (); gtk_menu_attach_to_widget (GTK_MENU (private->right_click_menu), button, NULL); private->menu_item_file_icon = gtk_menu_item_new_with_label (_("From File...")); gtk_menu_shell_append (GTK_MENU_SHELL (private->right_click_menu), GTK_WIDGET (private->menu_item_file_icon)); g_signal_connect (private->menu_item_file_icon, "button-press-event", G_CALLBACK (pika_icon_picker_menu_from_file), object); private->menu_item_name_icon = gtk_menu_item_new_with_label (_("From Named Icons...")); gtk_menu_shell_append (GTK_MENU_SHELL (private->right_click_menu), GTK_WIDGET (private->menu_item_name_icon)); g_signal_connect (private->menu_item_name_icon, "button-press-event", G_CALLBACK (pika_icon_picker_menu_from_name), object); private->menu_item_copy = gtk_menu_item_new_with_label (_("Copy Icon to Clipboard")); gtk_menu_shell_append (GTK_MENU_SHELL (private->right_click_menu), GTK_WIDGET (private->menu_item_copy)); g_signal_connect (private->menu_item_copy, "button-press-event", G_CALLBACK (pika_icon_picker_menu_copy), object); private->menu_item_paste = gtk_menu_item_new_with_label (_("Paste Icon from Clipboard")); gtk_menu_shell_append (GTK_MENU_SHELL (private->right_click_menu), GTK_WIDGET (private->menu_item_paste)); g_signal_connect (private->menu_item_paste, "button-press-event", G_CALLBACK (pika_icon_picker_menu_paste), object); gtk_widget_show_all (GTK_WIDGET (private->right_click_menu)); } static void pika_icon_picker_finalize (GObject *object) { PikaIconPickerPrivate *private = GET_PRIVATE (object); g_clear_pointer (&private->icon_name, g_free); g_clear_object (&private->icon_name_container); g_clear_object (&private->icon_name_context); g_clear_object (&private->icon_pixbuf); g_clear_object (&private->preview); g_clear_object (&private->null_template_object); G_OBJECT_CLASS (parent_class)->finalize (object); } static void pika_icon_picker_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { PikaIconPickerPrivate *private = GET_PRIVATE (object); switch (property_id) { case PROP_PIKA: private->pika = g_value_get_object (value); /* don't ref */ break; case PROP_ICON_NAME: pika_icon_picker_set_icon_name (PIKA_ICON_PICKER (object), g_value_get_string (value)); break; case PROP_ICON_PIXBUF: pika_icon_picker_set_icon_pixbuf (PIKA_ICON_PICKER (object), g_value_get_object (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } static void pika_icon_picker_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { PikaIconPickerPrivate *private = GET_PRIVATE (object); switch (property_id) { case PROP_PIKA: g_value_set_object (value, private->pika); break; case PROP_ICON_NAME: g_value_set_string (value, private->icon_name); break; case PROP_ICON_PIXBUF: g_value_set_object (value, private->icon_pixbuf); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } GtkWidget * pika_icon_picker_new (Pika *pika) { g_return_val_if_fail (PIKA_IS_PIKA (pika), NULL); return g_object_new (PIKA_TYPE_ICON_PICKER, "pika", pika, NULL); } const gchar * pika_icon_picker_get_icon_name (PikaIconPicker *picker) { g_return_val_if_fail (PIKA_IS_ICON_PICKER (picker), NULL); return GET_PRIVATE (picker)->icon_name; } void pika_icon_picker_set_icon_name (PikaIconPicker *picker, const gchar *icon_name) { PikaIconPickerPrivate *private; g_return_if_fail (PIKA_IS_ICON_PICKER (picker)); g_return_if_fail (icon_name != NULL); private = GET_PRIVATE (picker); g_free (private->icon_name); private->icon_name = g_strdup (icon_name); if (private->icon_name_container) { PikaObject *object; object = pika_container_get_child_by_name (private->icon_name_container, icon_name); if (object) pika_context_set_template (private->icon_name_context, PIKA_TEMPLATE (object)); } g_object_set (private->preview, "icon-name", private->icon_name, NULL); g_object_notify (G_OBJECT (picker), "icon-name"); } GdkPixbuf * pika_icon_picker_get_icon_pixbuf (PikaIconPicker *picker) { g_return_val_if_fail (PIKA_IS_ICON_PICKER (picker), NULL); return GET_PRIVATE (picker)->icon_pixbuf; } void pika_icon_picker_set_icon_pixbuf (PikaIconPicker *picker, GdkPixbuf *value) { PikaIconPickerPrivate *private; g_return_if_fail (PIKA_IS_ICON_PICKER (picker)); g_return_if_fail (value == NULL || GDK_IS_PIXBUF (value)); private = GET_PRIVATE (picker); if (private->icon_pixbuf) g_object_unref (private->icon_pixbuf); private->icon_pixbuf = value; if (private->icon_pixbuf) { g_object_ref (private->icon_pixbuf); pika_context_set_template (private->icon_name_context, PIKA_TEMPLATE (private->null_template_object)); } else { PikaObject *object; object = pika_container_get_child_by_name (private->icon_name_container, private->icon_name); if (object) pika_context_set_template (private->icon_name_context, PIKA_TEMPLATE (object)); } g_object_set (private->preview, "icon-pixbuf", private->icon_pixbuf, NULL); g_object_notify (G_OBJECT (picker), "icon-pixbuf"); } /* private functions */ static void pika_icon_picker_icon_changed (PikaContext *context, PikaTemplate *template, PikaIconPicker *picker) { PikaIconPickerPrivate *private = GET_PRIVATE (picker); if (PIKA_OBJECT (template) != private->null_template_object) { pika_icon_picker_set_icon_pixbuf (picker, NULL); pika_icon_picker_set_icon_name (picker, pika_object_get_name (template)); } } static void pika_icon_picker_menu_from_file (GtkWidget *widget, GdkEventButton *event, gpointer object) { PikaIconPicker *picker = PIKA_ICON_PICKER (object); GtkWidget *dialog; GtkFileFilter *filter; dialog = gtk_file_chooser_dialog_new (_("Load Icon Image"), NULL, GTK_FILE_CHOOSER_ACTION_OPEN, _("_Cancel"), GTK_RESPONSE_CANCEL, _("_Open"), GTK_RESPONSE_ACCEPT, NULL); filter = gtk_file_filter_new (); gtk_file_filter_add_pixbuf_formats (filter); gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter); if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT) { gchar *filename; GdkPixbuf *icon_pixbuf = NULL; filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog)); icon_pixbuf = gdk_pixbuf_new_from_file (filename, NULL); if (icon_pixbuf) { pika_icon_picker_set_icon_pixbuf (picker, icon_pixbuf); g_object_unref (icon_pixbuf); } g_free (filename); } gtk_widget_destroy (dialog); } static void pika_icon_picker_menu_copy (GtkWidget *widget, GdkEventButton *event, gpointer object) { PikaIconPicker *picker = PIKA_ICON_PICKER (object); PikaIconPickerPrivate *private = GET_PRIVATE (picker); GtkClipboard *clipboard = NULL; clipboard = gtk_clipboard_get_for_display (gtk_widget_get_display (widget), GDK_SELECTION_CLIPBOARD); if (private->icon_pixbuf) { gtk_clipboard_set_image (clipboard, private->icon_pixbuf); } } static void pika_icon_picker_menu_paste (GtkWidget *widget, GdkEventButton *event, gpointer object) { PikaIconPicker *picker = PIKA_ICON_PICKER (object); GtkClipboard *clipboard = NULL; GdkPixbuf *clipboard_pixbuf = NULL; clipboard = gtk_clipboard_get_for_display (gtk_widget_get_display (widget), GDK_SELECTION_CLIPBOARD); clipboard_pixbuf = gtk_clipboard_wait_for_image (clipboard); if (clipboard_pixbuf) { pika_icon_picker_set_icon_pixbuf (picker, clipboard_pixbuf); g_object_unref (clipboard_pixbuf); } } static void pika_icon_picker_clicked (GtkWidget *widget, GdkEventButton *event, gpointer object) { PikaIconPicker *picker = PIKA_ICON_PICKER (object); PikaIconPickerPrivate *private = GET_PRIVATE (picker); GtkClipboard *clipboard = NULL; clipboard = gtk_clipboard_get_for_display (gtk_widget_get_display (widget), GDK_SELECTION_CLIPBOARD); if (gtk_clipboard_wait_is_image_available (clipboard)) gtk_widget_set_sensitive (private->menu_item_paste, TRUE); else gtk_widget_set_sensitive (private->menu_item_paste, FALSE); if (private->icon_pixbuf) gtk_widget_set_sensitive (private->menu_item_copy, TRUE); else gtk_widget_set_sensitive (private->menu_item_copy, FALSE); gtk_menu_popup_at_widget (GTK_MENU (private->right_click_menu), widget, GDK_GRAVITY_EAST, GDK_GRAVITY_NORTH_WEST, (GdkEvent *) event); } static void pika_icon_picker_menu_from_name (GtkWidget *widget, GdkEventButton *event, gpointer object) { PikaIconPicker *picker = PIKA_ICON_PICKER (object); PikaIconPickerPrivate *private = GET_PRIVATE (picker); GtkWidget *popup; /* FIXME: Right clicking on this popup can cause a crash */ popup = pika_container_popup_new (private->icon_name_container, private->icon_name_context, PIKA_VIEW_TYPE_LIST, PIKA_VIEW_SIZE_SMALL, PIKA_VIEW_SIZE_SMALL, 0, NULL, NULL, NULL, NULL); pika_container_popup_set_view_type (PIKA_CONTAINER_POPUP (popup), PIKA_VIEW_TYPE_GRID); pika_popup_show (PIKA_POPUP (popup), GTK_WIDGET (picker)); }