mirror of https://github.com/GNOME/gimp.git
391 lines
13 KiB
C
391 lines
13 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* gimpextensionlist.c
|
|
* Copyright (C) 2018 Jehan <jehan@gimp.org>
|
|
*
|
|
* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#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"));
|
|
}
|