/* GIMP - The GNU Image Manipulation Program * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * gimpextensionlist.c * Copyright (C) 2018 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 "libgimpwidgets/gimpwidgets.h" #include "widgets-types.h" #include "core/gimpextension.h" #include "core/gimpextensionmanager.h" #include "gimpextensionlist.h" #include "gimp-intl.h" enum { EXTENSION_ACTIVATED, LAST_SIGNAL }; typedef enum { GIMP_EXT_LIST_USER, GIMP_EXT_LIST_SYSTEM, GIMP_EXT_LIST_SEARCH, } GimpExtensionListContents; struct _GimpExtensionListPrivate { GimpExtensionManager *manager; GimpExtensionListContents contents; }; static void gimp_extension_list_set (GimpExtensionList *list, const GList *extensions, gboolean is_system); static void gimp_extension_list_ext_installed (GimpExtensionManager *manager, GimpExtension *extension, gboolean is_system_ext, GimpExtensionList *list); static void gimp_extension_list_ext_removed (GimpExtensionManager *manager, gchar *extension_id, GimpExtensionList *list); static void gimp_extension_list_delete_clicked (GtkButton *delbutton, GimpExtensionList *list); static void gimp_extension_switch_active (GObject *onoff, GParamSpec *spec, gpointer extension); static void gimp_extension_row_activated (GtkListBox *box, GtkListBoxRow *row, gpointer user_data); G_DEFINE_TYPE_WITH_PRIVATE (GimpExtensionList, gimp_extension_list, GTK_TYPE_LIST_BOX) #define parent_class gimp_extension_list_parent_class static guint signals[LAST_SIGNAL] = { 0, }; static void gimp_extension_list_class_init (GimpExtensionListClass *klass) { signals[EXTENSION_ACTIVATED] = g_signal_new ("extension-activated", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (GimpExtensionListClass, extension_activated), NULL, NULL, NULL, G_TYPE_NONE, 1, GIMP_TYPE_OBJECT); } static void gimp_extension_list_init (GimpExtensionList *list) { gtk_list_box_set_selection_mode (GTK_LIST_BOX (list), GTK_SELECTION_SINGLE); gtk_list_box_set_activate_on_single_click (GTK_LIST_BOX (list), FALSE); list->p = gimp_extension_list_get_instance_private (list); } static void gimp_extension_list_disable (GtkWidget *row, gchar *extension_id) { GimpExtension *extension; GtkWidget *outframe = gtk_bin_get_child (GTK_BIN (row)); extension = g_object_get_data (G_OBJECT (outframe), "extension"); g_return_if_fail (extension); if (g_strcmp0 (gimp_object_get_name (extension), extension_id) == 0) { GtkWidget *button = gtk_frame_get_label_widget (GTK_FRAME (outframe)); GtkWidget *image = gtk_bin_get_child (GTK_BIN (button)); gtk_widget_set_sensitive (gtk_bin_get_child (GTK_BIN (outframe)), FALSE); gtk_image_set_from_icon_name (GTK_IMAGE (image), GIMP_ICON_EDIT_UNDO, GTK_ICON_SIZE_MENU); gtk_image_set_pixel_size (GTK_IMAGE (image), 12); } } GtkWidget * gimp_extension_list_new (GimpExtensionManager *manager) { GimpExtensionList *list; g_return_val_if_fail (GIMP_IS_EXTENSION_MANAGER (manager), NULL); list = g_object_new (GIMP_TYPE_EXTENSION_LIST, NULL); list->p->manager = manager; g_signal_connect (manager, "extension-installed", G_CALLBACK (gimp_extension_list_ext_installed), list); g_signal_connect (manager, "extension-removed", G_CALLBACK (gimp_extension_list_ext_removed), list); return GTK_WIDGET (list); } void gimp_extension_list_show_system (GimpExtensionList *list) { list->p->contents = GIMP_EXT_LIST_SYSTEM; gimp_extension_list_set (list, gimp_extension_manager_get_system_extensions (list->p->manager), TRUE); } void gimp_extension_list_show_user (GimpExtensionList *list) { list->p->contents = GIMP_EXT_LIST_USER; gimp_extension_list_set (list, gimp_extension_manager_get_user_extensions (list->p->manager), FALSE); } void gimp_extension_list_show_search (GimpExtensionList *list, const gchar *search_terms) { list->p->contents = GIMP_EXT_LIST_SEARCH; /* TODO */ gimp_extension_list_set (list, NULL, FALSE); } static void gimp_extension_list_set (GimpExtensionList *list, const GList *extensions, gboolean is_system) { GList *iter = (GList *) extensions; gtk_container_foreach (GTK_CONTAINER (list), (GtkCallback) gtk_widget_destroy, NULL); for (; iter; iter = iter->next) { GimpExtension *extension = iter->data; gimp_extension_list_ext_installed (list->p->manager, extension, is_system, list); } gtk_container_foreach (GTK_CONTAINER (list), (GtkCallback) gtk_list_box_row_set_activatable, GUINT_TO_POINTER (TRUE)); g_signal_connect (list, "row-activated", G_CALLBACK (gimp_extension_row_activated), NULL); } static void gimp_extension_list_ext_installed (GimpExtensionManager *manager, GimpExtension *extension, gboolean is_system_ext, GimpExtensionList *list) { GList *rows; GList *iter; GtkWidget *outframe; GtkWidget *grid; GtkWidget *onoff; GtkWidget *delbutton; GtkWidget *image; if (list->p->contents == GIMP_EXT_LIST_SEARCH || (list->p->contents == GIMP_EXT_LIST_USER && is_system_ext) || (list->p->contents == GIMP_EXT_LIST_SYSTEM && ! is_system_ext)) return; /* Check if extension already listed (i.e. removed then reinstalled). */ rows = gtk_container_get_children (GTK_CONTAINER (list)); for (iter = rows; iter; iter = iter->next) { GimpExtension *row_ext; GtkWidget *outframe = gtk_bin_get_child (GTK_BIN (iter->data)); row_ext = g_object_get_data (G_OBJECT (outframe), "extension"); g_return_if_fail (row_ext); if (extension == row_ext) { GtkWidget *button = gtk_frame_get_label_widget (GTK_FRAME (outframe)); image = gtk_bin_get_child (GTK_BIN (button)); gtk_widget_set_sensitive (gtk_bin_get_child (GTK_BIN (outframe)), TRUE); gtk_image_set_from_icon_name (GTK_IMAGE (image), GIMP_ICON_EDIT_DELETE, GTK_ICON_SIZE_MENU); gtk_image_set_pixel_size (GTK_IMAGE (image), 12); g_list_free (rows); return; } } g_list_free (rows); outframe = gtk_frame_new (gimp_extension_get_name (extension)); gtk_container_add (GTK_CONTAINER (list), outframe); g_object_set_data (G_OBJECT (outframe), "extension", extension); gtk_widget_show (outframe); grid = gtk_grid_new (); gtk_grid_set_column_homogeneous (GTK_GRID (grid), FALSE); gtk_grid_set_row_homogeneous (GTK_GRID (grid), FALSE); gtk_container_add (GTK_CONTAINER (outframe), grid); gtk_widget_show (grid); /* On/Off switch. */ onoff = gtk_switch_new (); gtk_widget_set_vexpand (onoff, FALSE); gtk_widget_set_hexpand (onoff, FALSE); gtk_widget_set_halign (onoff, GTK_ALIGN_CENTER); gtk_widget_set_valign (onoff, GTK_ALIGN_CENTER); gtk_switch_set_active (GTK_SWITCH (onoff), gimp_extension_manager_is_running (list->p->manager, extension)); gtk_widget_set_sensitive (onoff, gimp_extension_manager_can_run (list->p->manager, extension)); g_signal_connect (onoff, "notify::active", G_CALLBACK (gimp_extension_switch_active), extension); gtk_grid_attach (GTK_GRID (grid), onoff, 0, 0, 1, 1); gtk_widget_show (onoff); /* Short description. */ if (gimp_extension_get_comment (extension)) { GtkWidget *desc = gtk_text_view_new (); GtkTextBuffer *buffer; buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (desc)); gtk_text_buffer_set_text (buffer, gimp_extension_get_comment (extension), -1); gtk_text_view_set_editable (GTK_TEXT_VIEW (desc), FALSE); gtk_widget_set_vexpand (desc, TRUE); gtk_widget_set_hexpand (desc, TRUE); gtk_grid_attach (GTK_GRID (grid), desc, 1, 0, 1, 1); gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (desc), GTK_WRAP_WORD_CHAR); gtk_widget_show (desc); } /* Delete button. */ delbutton = gtk_button_new (); g_object_set_data (G_OBJECT (delbutton), "extension", extension); g_signal_connect (delbutton, "clicked", G_CALLBACK (gimp_extension_list_delete_clicked), list); gtk_button_set_relief (GTK_BUTTON (delbutton), GTK_RELIEF_NONE); image = gtk_image_new_from_icon_name (GIMP_ICON_EDIT_DELETE, GTK_ICON_SIZE_MENU); gtk_image_set_pixel_size (GTK_IMAGE (image), 12); gtk_widget_set_vexpand (delbutton, FALSE); gtk_widget_set_hexpand (delbutton, FALSE); gtk_widget_set_halign (delbutton, GTK_ALIGN_END); gtk_widget_set_valign (delbutton, GTK_ALIGN_START); gtk_container_add (GTK_CONTAINER (delbutton), image); gtk_widget_show (image); gtk_grid_attach (GTK_GRID (grid), delbutton, 2, 0, 1, 1); gtk_widget_show (delbutton); } static void gimp_extension_list_ext_removed (GimpExtensionManager *manager, gchar *extension_id, GimpExtensionList *list) { gtk_container_foreach (GTK_CONTAINER (list), (GtkCallback) gimp_extension_list_disable, extension_id); } static void gimp_extension_list_delete_clicked (GtkButton *delbutton, GimpExtensionList *list) { GimpExtensionManager *manager = list->p->manager; GimpExtension *extension; GError *error = NULL; extension = g_object_get_data (G_OBJECT (delbutton), "extension"); g_return_if_fail (extension); if (gimp_extension_manager_is_removed (manager, extension)) gimp_extension_manager_undo_remove (manager, extension, &error); else gimp_extension_manager_remove (manager, extension, &error); if (error) { g_warning ("%s: %s\n", G_STRFUNC, error->message); g_error_free (error); } } static void gimp_extension_switch_active (GObject *onoff, GParamSpec *pspec, gpointer extension) { GimpExtension *ext = (GimpExtension *) extension; if (gtk_switch_get_active (GTK_SWITCH (onoff))) { GError *error = NULL; gimp_extension_run (ext, &error); if (error) { g_signal_handlers_block_by_func (onoff, G_CALLBACK (gimp_extension_switch_active), extension); gtk_switch_set_active (GTK_SWITCH (onoff), FALSE); g_signal_handlers_unblock_by_func (onoff, G_CALLBACK (gimp_extension_switch_active), extension); g_printerr ("Extension '%s' failed to run: %s\n", gimp_object_get_name (ext), error->message); g_error_free (error); } } else gimp_extension_stop (ext); } static void gimp_extension_row_activated (GtkListBox *box, GtkListBoxRow *row, gpointer user_data) { GtkWidget *frame = gtk_bin_get_child (GTK_BIN (row)); g_signal_emit (box, signals[EXTENSION_ACTIVATED], 0, g_object_get_data (G_OBJECT (frame), "extension")); }