/* GIMP - The GNU Image Manipulation Program * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * gimppickablechooser.c * Copyright (C) 2023 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 #include "libgimpbase/gimpbase.h" #include "libgimpwidgets/gimpwidgets.h" #include "widgets-types.h" #include "core/gimp.h" #include "core/gimpchannel.h" #include "core/gimpcontext.h" #include "core/gimpimage.h" #include "core/gimplayer.h" #include "core/gimppickable.h" #include "core/gimpviewable.h" #include "gimpcontainertreeview.h" #include "gimpcontainerview.h" #include "gimppickablechooser.h" #include "gimpviewrenderer.h" #include "gimp-intl.h" enum { ACTIVATE, LAST_SIGNAL }; enum { PROP_0, PROP_CONTEXT, PROP_PICKABLE_TYPE, PROP_PICKABLE, PROP_VIEW_SIZE, PROP_VIEW_BORDER_WIDTH }; struct _GimpPickableChooserPrivate { GType pickable_type; GimpPickable *pickable; GimpContext *context; gint view_size; gint view_border_width; GtkWidget *image_view; GtkWidget *layer_view; GtkWidget *channel_view; GtkWidget *layer_label; }; static void gimp_pickable_chooser_constructed (GObject *object); static void gimp_pickable_chooser_finalize (GObject *object); static void gimp_pickable_chooser_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec); static void gimp_pickable_chooser_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec); static void gimp_pickable_chooser_image_changed (GimpContext *context, GimpImage *image, GimpPickableChooser *chooser); static void gimp_pickable_chooser_item_activate (GimpContainerView *view, GimpPickable *pickable, gpointer unused, GimpPickableChooser *chooser); static void gimp_pickable_chooser_items_selected (GimpContainerView *view, GList *items, GList *paths, GimpPickableChooser *chooser); G_DEFINE_TYPE_WITH_PRIVATE (GimpPickableChooser, gimp_pickable_chooser, GTK_TYPE_FRAME) #define parent_class gimp_pickable_chooser_parent_class static guint signals[LAST_SIGNAL]; static void gimp_pickable_chooser_class_init (GimpPickableChooserClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->constructed = gimp_pickable_chooser_constructed; object_class->finalize = gimp_pickable_chooser_finalize; object_class->get_property = gimp_pickable_chooser_get_property; object_class->set_property = gimp_pickable_chooser_set_property; /** * GimpPickableChooser::activate: * @chooser: * * Emitted when a pickable is activated, which is mostly forwarding when * "activate-item" signal is emitted from any of either the image, layer or * channel view. E.g. this happens when one double-click on one of the * pickables. */ signals[ACTIVATE] = g_signal_new ("activate", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (GimpPickableChooserClass, activate), NULL, NULL, NULL, G_TYPE_NONE, 1, GIMP_TYPE_OBJECT); g_object_class_install_property (object_class, PROP_CONTEXT, g_param_spec_object ("context", NULL, NULL, GIMP_TYPE_CONTEXT, GIMP_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (object_class, PROP_PICKABLE_TYPE, g_param_spec_gtype ("pickable-type", NULL, NULL, GIMP_TYPE_PICKABLE, GIMP_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (object_class, PROP_PICKABLE, g_param_spec_object ("pickable", NULL, NULL, GIMP_TYPE_PICKABLE, GIMP_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY)); g_object_class_install_property (object_class, PROP_VIEW_SIZE, g_param_spec_int ("view-size", NULL, NULL, 1, GIMP_VIEWABLE_MAX_PREVIEW_SIZE, GIMP_VIEW_SIZE_MEDIUM, GIMP_PARAM_READWRITE | G_PARAM_CONSTRUCT)); g_object_class_install_property (object_class, PROP_VIEW_BORDER_WIDTH, g_param_spec_int ("view-border-width", NULL, NULL, 0, GIMP_VIEW_MAX_BORDER_WIDTH, 1, GIMP_PARAM_READWRITE | G_PARAM_CONSTRUCT)); } static void gimp_pickable_chooser_init (GimpPickableChooser *chooser) { chooser->priv = gimp_pickable_chooser_get_instance_private (chooser); chooser->priv->view_size = GIMP_VIEW_SIZE_SMALL; chooser->priv->view_border_width = 1; chooser->priv->layer_view = NULL; chooser->priv->channel_view = NULL; } static void gimp_pickable_chooser_constructed (GObject *object) { GimpPickableChooser *chooser = GIMP_PICKABLE_CHOOSER (object); GtkWidget *hbox; GtkWidget *vbox; GtkWidget *label; GtkWidget *notebook; GimpImage *image; G_OBJECT_CLASS (parent_class)->constructed (object); gimp_assert (GIMP_IS_CONTEXT (chooser->priv->context)); gtk_frame_set_shadow_type (GTK_FRAME (chooser), GTK_SHADOW_OUT); hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6); gtk_container_set_border_width (GTK_CONTAINER (hbox), 6); gtk_container_add (GTK_CONTAINER (chooser), hbox); gtk_widget_show (hbox); vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 4); gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0); gtk_widget_show (vbox); label = gtk_label_new (_("Images")); gtk_label_set_xalign (GTK_LABEL (label), 0.0); gtk_box_pack_start (GTK_BOX (vbox), label, FALSE, FALSE, 0); gtk_widget_show (label); chooser->priv->image_view = gimp_container_tree_view_new (chooser->priv->context->gimp->images, chooser->priv->context, chooser->priv->view_size, chooser->priv->view_border_width); gimp_container_box_set_size_request (GIMP_CONTAINER_BOX (chooser->priv->image_view), 4 * (chooser->priv->view_size + 2 * chooser->priv->view_border_width), 4 * (chooser->priv->view_size + 2 * chooser->priv->view_border_width)); gtk_box_pack_start (GTK_BOX (vbox), chooser->priv->image_view, TRUE, TRUE, 0); gtk_widget_show (chooser->priv->image_view); g_signal_connect_object (chooser->priv->image_view, "activate-item", G_CALLBACK (gimp_pickable_chooser_item_activate), G_OBJECT (chooser), 0); g_signal_connect_object (chooser->priv->image_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 4); gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0); gtk_widget_show (vbox); chooser->priv->layer_label = label = gtk_label_new (_("Select an image in the left pane")); gtk_label_set_xalign (GTK_LABEL (label), 0.0); gtk_label_set_ellipsize (GTK_LABEL (label), PANGO_ELLIPSIZE_END); gtk_box_pack_start (GTK_BOX (vbox), label, FALSE, FALSE, 0); gtk_widget_show (label); notebook = gtk_notebook_new (); gtk_box_pack_start (GTK_BOX (vbox), notebook, TRUE, TRUE, 0); gtk_widget_show (notebook); if (g_type_is_a (GIMP_TYPE_LAYER, chooser->priv->pickable_type)) { chooser->priv->layer_view = gimp_container_tree_view_new (NULL, chooser->priv->context, chooser->priv->view_size, chooser->priv->view_border_width); gtk_tree_view_set_show_expanders (GTK_TREE_VIEW (GIMP_CONTAINER_TREE_VIEW (chooser->priv->layer_view)->view), TRUE); gimp_container_box_set_size_request (GIMP_CONTAINER_BOX (chooser->priv->layer_view), 4 * (chooser->priv->view_size + 2 * chooser->priv->view_border_width), 4 * (chooser->priv->view_size + 2 * chooser->priv->view_border_width)); gtk_notebook_append_page (GTK_NOTEBOOK (notebook), chooser->priv->layer_view, gtk_label_new (_("Layers"))); gtk_widget_show (chooser->priv->layer_view); g_signal_connect_object (chooser->priv->layer_view, "activate-item", G_CALLBACK (gimp_pickable_chooser_item_activate), G_OBJECT (chooser), 0); g_signal_connect_object (chooser->priv->layer_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); } if (g_type_is_a (GIMP_TYPE_CHANNEL, chooser->priv->pickable_type)) { chooser->priv->channel_view = gimp_container_tree_view_new (NULL, chooser->priv->context, chooser->priv->view_size, chooser->priv->view_border_width); gimp_container_box_set_size_request (GIMP_CONTAINER_BOX (chooser->priv->channel_view), 4 * (chooser->priv->view_size + 2 * chooser->priv->view_border_width), 4 * (chooser->priv->view_size + 2 * chooser->priv->view_border_width)); gtk_notebook_append_page (GTK_NOTEBOOK (notebook), chooser->priv->channel_view, gtk_label_new (_("Channels"))); gtk_widget_show (chooser->priv->channel_view); g_signal_connect_object (chooser->priv->channel_view, "activate-item", G_CALLBACK (gimp_pickable_chooser_item_activate), G_OBJECT (chooser), 0); g_signal_connect_object (chooser->priv->channel_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); } g_signal_connect_object (chooser->priv->context, "image-changed", G_CALLBACK (gimp_pickable_chooser_image_changed), G_OBJECT (chooser), 0); image = gimp_context_get_image (chooser->priv->context); gimp_pickable_chooser_image_changed (chooser->priv->context, image, chooser); } static void gimp_pickable_chooser_finalize (GObject *object) { GimpPickableChooser *chooser = GIMP_PICKABLE_CHOOSER (object); g_clear_object (&chooser->priv->pickable); g_clear_object (&chooser->priv->context); G_OBJECT_CLASS (parent_class)->finalize (object); } static void gimp_pickable_chooser_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { GimpPickableChooser *chooser = GIMP_PICKABLE_CHOOSER (object); switch (property_id) { case PROP_CONTEXT: if (chooser->priv->context) g_object_unref (chooser->priv->context); chooser->priv->context = g_value_dup_object (value); break; case PROP_VIEW_SIZE: chooser->priv->view_size = g_value_get_int (value); break; case PROP_VIEW_BORDER_WIDTH: chooser->priv->view_border_width = g_value_get_int (value); break; case PROP_PICKABLE_TYPE: g_return_if_fail (g_value_get_gtype (value) == GIMP_TYPE_LAYER || g_value_get_gtype (value) == GIMP_TYPE_CHANNEL || g_value_get_gtype (value) == GIMP_TYPE_DRAWABLE || g_value_get_gtype (value) == GIMP_TYPE_IMAGE || g_value_get_gtype (value) == GIMP_TYPE_PICKABLE); chooser->priv->pickable_type = g_value_get_gtype (value); break; case PROP_PICKABLE: g_return_if_fail (g_value_get_object (value) == NULL || g_type_is_a (G_TYPE_FROM_INSTANCE (g_value_get_object (value)), chooser->priv->pickable_type)); gimp_pickable_chooser_set_pickable (chooser, g_value_get_object (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } static void gimp_pickable_chooser_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { GimpPickableChooser *chooser = GIMP_PICKABLE_CHOOSER (object); switch (property_id) { case PROP_CONTEXT: g_value_set_object (value, chooser->priv->context); break; case PROP_PICKABLE_TYPE: g_value_set_gtype (value, chooser->priv->pickable_type); break; case PROP_PICKABLE: g_value_set_object (value, chooser->priv->pickable); break; case PROP_VIEW_SIZE: g_value_set_int (value, chooser->priv->view_size); break; case PROP_VIEW_BORDER_WIDTH: g_value_set_int (value, chooser->priv->view_border_width); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } GtkWidget * gimp_pickable_chooser_new (GimpContext *context, GType pickable_type, gint view_size, gint view_border_width) { g_return_val_if_fail (GIMP_IS_CONTEXT (context), NULL); g_return_val_if_fail (view_size > 0 && view_size <= GIMP_VIEWABLE_MAX_POPUP_SIZE, NULL); g_return_val_if_fail (view_border_width >= 0 && view_border_width <= GIMP_VIEW_MAX_BORDER_WIDTH, NULL); return g_object_new (GIMP_TYPE_PICKABLE_CHOOSER, "context", context, "pickable-type", pickable_type, "view-size", view_size, "view-border-width", view_border_width, NULL); } GimpPickable * gimp_pickable_chooser_get_pickable (GimpPickableChooser *chooser) { g_return_val_if_fail (GIMP_IS_PICKABLE_CHOOSER (chooser), NULL); return chooser->priv->pickable; } void gimp_pickable_chooser_set_pickable (GimpPickableChooser *chooser, GimpPickable *pickable) { if (! gtk_widget_in_destruction (GTK_WIDGET (chooser))) { g_signal_handlers_disconnect_by_func (chooser->priv->image_view, G_CALLBACK (gimp_pickable_chooser_items_selected), chooser); if (chooser->priv->layer_view != NULL) g_signal_handlers_disconnect_by_func (chooser->priv->layer_view, G_CALLBACK (gimp_pickable_chooser_items_selected), chooser); else g_return_if_fail (! GIMP_IS_LAYER (pickable)); if (chooser->priv->channel_view != NULL) g_signal_handlers_disconnect_by_func (chooser->priv->channel_view, G_CALLBACK (gimp_pickable_chooser_items_selected), chooser); else g_return_if_fail (! GIMP_IS_CHANNEL (pickable)); if (GIMP_IS_IMAGE (pickable)) { gimp_container_view_select_item (GIMP_CONTAINER_VIEW (chooser->priv->image_view), GIMP_VIEWABLE (pickable)); gimp_context_set_image (chooser->priv->context, GIMP_IMAGE (pickable)); } else if (GIMP_IS_LAYER (pickable)) { gimp_context_set_image (chooser->priv->context, gimp_item_get_image (GIMP_ITEM (pickable))); gimp_container_view_select_item (GIMP_CONTAINER_VIEW (chooser->priv->layer_view), GIMP_VIEWABLE (pickable)); } else if (GIMP_IS_CHANNEL (pickable)) { gimp_context_set_image (chooser->priv->context, gimp_item_get_image (GIMP_ITEM (pickable))); gimp_container_view_select_item (GIMP_CONTAINER_VIEW (chooser->priv->channel_view), GIMP_VIEWABLE (pickable)); } else { g_return_if_fail (pickable == NULL); gimp_container_view_select_item (GIMP_CONTAINER_VIEW (chooser->priv->image_view), NULL); gimp_context_set_image (chooser->priv->context, NULL); } g_signal_connect_object (chooser->priv->image_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); if (chooser->priv->layer_view != NULL) g_signal_connect_object (chooser->priv->layer_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); if (chooser->priv->channel_view != NULL) g_signal_connect_object (chooser->priv->channel_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); } if (pickable != chooser->priv->pickable) { g_clear_object (&chooser->priv->pickable); chooser->priv->pickable = (pickable != NULL ? g_object_ref (pickable) : NULL); g_object_notify (G_OBJECT (chooser), "pickable"); } } /* private functions */ static void gimp_pickable_chooser_image_changed (GimpContext *context, GimpImage *image, GimpPickableChooser *chooser) { GimpContainer *layers = NULL; GimpContainer *channels = NULL; if (image) { gchar *desc; layers = gimp_image_get_layers (image); channels = gimp_image_get_channels (image); desc = gimp_viewable_get_description (GIMP_VIEWABLE (image), NULL); gtk_label_set_text (GTK_LABEL (chooser->priv->layer_label), desc); g_free (desc); } else { gtk_label_set_text (GTK_LABEL (chooser->priv->layer_label), _("Select an image in the left pane")); } g_signal_handlers_disconnect_by_func (chooser->priv->image_view, G_CALLBACK (gimp_pickable_chooser_items_selected), chooser); if (chooser->priv->layer_view != NULL) { g_signal_handlers_disconnect_by_func (chooser->priv->layer_view, G_CALLBACK (gimp_pickable_chooser_items_selected), chooser); gimp_container_view_set_container (GIMP_CONTAINER_VIEW (chooser->priv->layer_view), layers); g_signal_connect_object (chooser->priv->layer_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); } if (chooser->priv->channel_view != NULL) { g_signal_handlers_disconnect_by_func (chooser->priv->channel_view, G_CALLBACK (gimp_pickable_chooser_items_selected), chooser); gimp_container_view_set_container (GIMP_CONTAINER_VIEW (chooser->priv->channel_view), channels); g_signal_connect_object (chooser->priv->channel_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); } g_signal_connect_object (chooser->priv->image_view, "select-items", G_CALLBACK (gimp_pickable_chooser_items_selected), G_OBJECT (chooser), 0); } static void gimp_pickable_chooser_item_activate (GimpContainerView *view, GimpPickable *pickable, gpointer unused, GimpPickableChooser *chooser) { g_signal_emit (chooser, signals[ACTIVATE], 0, pickable); } static void gimp_pickable_chooser_items_selected (GimpContainerView *view, GList *items, GList *paths, GimpPickableChooser *chooser) { GimpPickable *pickable = NULL; g_return_if_fail (g_list_length (items) <= 1); if (items != NULL) pickable = items->data; gimp_pickable_chooser_set_pickable (chooser, pickable); }