mirror of https://github.com/GNOME/gimp.git
554 lines
15 KiB
C
554 lines
15 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* 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 <string.h>
|
|
|
|
#include <gegl.h>
|
|
#include <gegl-plugin.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpcolor/gimpcolor.h"
|
|
#include "libgimpconfig/gimpconfig.h"
|
|
#include "libgimpwidgets/gimpwidgets.h"
|
|
|
|
#include "tools-types.h"
|
|
|
|
#include "widgets/gimphelp-ids.h"
|
|
#include "widgets/gimppropwidgets.h"
|
|
|
|
#include "gimpfilteroptions.h"
|
|
#include "gimpgegltool.h"
|
|
|
|
#include "gimp-intl.h"
|
|
|
|
|
|
enum
|
|
{
|
|
COLUMN_NAME,
|
|
COLUMN_LABEL,
|
|
COLUMN_ICON_NAME,
|
|
N_COLUMNS
|
|
};
|
|
|
|
|
|
/* local function prototypes */
|
|
|
|
static void gimp_gegl_tool_control (GimpTool *tool,
|
|
GimpToolAction action,
|
|
GimpDisplay *display);
|
|
|
|
static void gimp_gegl_tool_dialog (GimpFilterTool *filter_tool);
|
|
|
|
static void gimp_gegl_tool_halt (GimpGeglTool *gegl_tool);
|
|
|
|
static void gimp_gegl_tool_operation_changed (GtkWidget *widget,
|
|
GimpGeglTool *gegl_tool);
|
|
|
|
|
|
G_DEFINE_TYPE (GimpGeglTool, gimp_gegl_tool, GIMP_TYPE_OPERATION_TOOL)
|
|
|
|
#define parent_class gimp_gegl_tool_parent_class
|
|
|
|
|
|
void
|
|
gimp_gegl_tool_register (GimpToolRegisterCallback callback,
|
|
gpointer data)
|
|
{
|
|
(* callback) (GIMP_TYPE_GEGL_TOOL,
|
|
GIMP_TYPE_FILTER_OPTIONS,
|
|
gimp_color_options_gui,
|
|
0,
|
|
"gimp-gegl-tool",
|
|
_("GEGL Operation"),
|
|
_("GEGL Tool: Use an arbitrary GEGL operation"),
|
|
N_("_GEGL Operation..."), NULL,
|
|
NULL, GIMP_HELP_TOOL_GEGL,
|
|
GIMP_ICON_GEGL,
|
|
data);
|
|
}
|
|
|
|
static void
|
|
gimp_gegl_tool_class_init (GimpGeglToolClass *klass)
|
|
{
|
|
GimpToolClass *tool_class = GIMP_TOOL_CLASS (klass);
|
|
GimpFilterToolClass *filter_tool_class = GIMP_FILTER_TOOL_CLASS (klass);
|
|
|
|
tool_class->control = gimp_gegl_tool_control;
|
|
|
|
filter_tool_class->dialog = gimp_gegl_tool_dialog;
|
|
}
|
|
|
|
static void
|
|
gimp_gegl_tool_init (GimpGeglTool *tool)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gimp_gegl_tool_control (GimpTool *tool,
|
|
GimpToolAction action,
|
|
GimpDisplay *display)
|
|
{
|
|
GimpGeglTool *gegl_tool = GIMP_GEGL_TOOL (tool);
|
|
|
|
switch (action)
|
|
{
|
|
case GIMP_TOOL_ACTION_PAUSE:
|
|
case GIMP_TOOL_ACTION_RESUME:
|
|
break;
|
|
|
|
case GIMP_TOOL_ACTION_HALT:
|
|
gimp_gegl_tool_halt (gegl_tool);
|
|
break;
|
|
|
|
case GIMP_TOOL_ACTION_COMMIT:
|
|
break;
|
|
}
|
|
|
|
GIMP_TOOL_CLASS (parent_class)->control (tool, action, display);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_gegl_tool_operation_blacklisted (const gchar *name,
|
|
const gchar *categories_str)
|
|
{
|
|
static const gchar * const category_blacklist[] =
|
|
{
|
|
"compositors",
|
|
"core",
|
|
"debug",
|
|
"display",
|
|
"hidden",
|
|
"input",
|
|
"output",
|
|
"programming",
|
|
"transform",
|
|
"video"
|
|
};
|
|
static const gchar * const name_blacklist[] =
|
|
{
|
|
/* these ops are already added to the menus via
|
|
* filters-actions or drawable-actions
|
|
*/
|
|
"gegl:alien-map",
|
|
"gegl:antialias",
|
|
"gegl:apply-lens",
|
|
"gegl:bump-map",
|
|
"gegl:c2g",
|
|
"gegl:cartoon",
|
|
"gegl:cell-noise",
|
|
"gegl:channel-mixer",
|
|
"gegl:checkerboard",
|
|
"gegl:color",
|
|
"gegl:color-enhance",
|
|
"gegl:color-exchange",
|
|
"gegl:color-rotate",
|
|
"gegl:color-temperature",
|
|
"gegl:color-to-alpha",
|
|
"gegl:component-extract",
|
|
"gegl:convolution-matrix",
|
|
"gegl:cubism",
|
|
"gegl:deinterlace",
|
|
"gegl:difference-of-gaussians",
|
|
"gegl:diffraction-patterns",
|
|
"gegl:displace",
|
|
"gegl:distance-transform",
|
|
"gegl:dither",
|
|
"gegl:dropshadow",
|
|
"gegl:edge",
|
|
"gegl:edge-laplace",
|
|
"gegl:edge-neon",
|
|
"gegl:edge-sobel",
|
|
"gegl:emboss",
|
|
"gegl:engrave",
|
|
"gegl:exposure",
|
|
"gegl:fattal02",
|
|
"gegl:fractal-trace",
|
|
"gegl:gaussian-blur",
|
|
"gegl:gaussian-blur-selective",
|
|
"gegl:gegl",
|
|
"gegl:grid",
|
|
"gegl:high-pass",
|
|
"gegl:hue-chroma",
|
|
"gegl:illusion",
|
|
"gegl:image-gradient",
|
|
"gegl:invert-linear",
|
|
"gegl:invert-gamma",
|
|
"gegl:lens-distortion",
|
|
"gegl:lens-flare",
|
|
"gegl:long-shadow",
|
|
"gegl:mantiuk06",
|
|
"gegl:maze",
|
|
"gegl:median-blur",
|
|
"gegl:mirrors",
|
|
"gegl:mono-mixer",
|
|
"gegl:mosaic",
|
|
"gegl:motion-blur-circular",
|
|
"gegl:motion-blur-linear",
|
|
"gegl:motion-blur-zoom",
|
|
"gegl:noise-cie-lch",
|
|
"gegl:noise-hsv",
|
|
"gegl:noise-hurl",
|
|
"gegl:noise-pick",
|
|
"gegl:noise-reduction",
|
|
"gegl:noise-rgb",
|
|
"gegl:noise-slur",
|
|
"gegl:noise-solid",
|
|
"gegl:noise-spread",
|
|
"gegl:oilify",
|
|
"gegl:panorama-projection",
|
|
"gegl:perlin-noise",
|
|
"gegl:photocopy",
|
|
"gegl:pixelize",
|
|
"gegl:plasma",
|
|
"gegl:polar-coordinates",
|
|
"gegl:recursive-transform",
|
|
"gegl:red-eye-removal",
|
|
"gegl:reinhard05",
|
|
"gegl:rgb-clip",
|
|
"gegl:ripple",
|
|
"gegl:saturation",
|
|
"gegl:sepia",
|
|
"gegl:shadows-highlights",
|
|
"gegl:shift",
|
|
"gegl:simplex-noise",
|
|
"gegl:sinus",
|
|
"gegl:slic",
|
|
"gegl:snn-mean",
|
|
"gegl:softglow",
|
|
"gegl:spherize",
|
|
"gegl:spiral",
|
|
"gegl:stereographic-projection",
|
|
"gegl:stretch-contrast",
|
|
"gegl:stretch-contrast-hsv",
|
|
"gegl:stress",
|
|
"gegl:supernova",
|
|
"gegl:texturize-canvas",
|
|
"gegl:tile-glass",
|
|
"gegl:tile-paper",
|
|
"gegl:tile-seamless",
|
|
"gegl:unsharp-mask",
|
|
"gegl:value-invert",
|
|
"gegl:value-propagate",
|
|
"gegl:video-degradation",
|
|
"gegl:vignette",
|
|
"gegl:waterpixels",
|
|
"gegl:wavelet-blur",
|
|
"gegl:waves",
|
|
"gegl:whirl-pinch",
|
|
"gegl:wind",
|
|
|
|
/* these ops are blacklisted for other reasons */
|
|
"gegl:contrast-curve",
|
|
"gegl:convert-format", /* pointless */
|
|
"gegl:ditto", /* pointless */
|
|
"gegl:fill-path",
|
|
"gegl:gray", /* we use gimp's op */
|
|
"gegl:hstack", /* pointless */
|
|
"gegl:introspect", /* pointless */
|
|
"gegl:layer", /* we use gimp's ops */
|
|
"gegl:lcms-from-profile", /* not usable here */
|
|
"gegl:linear-gradient", /* we use the blend tool */
|
|
"gegl:map-absolute", /* pointless */
|
|
"gegl:map-relative", /* pointless */
|
|
"gegl:matting-global", /* used in the foreground select tool */
|
|
"gegl:matting-levin", /* used in the foreground select tool */
|
|
"gegl:opacity", /* poinless */
|
|
"gegl:path",
|
|
"gegl:posterize", /* we use gimp's op */
|
|
"gegl:radial-gradient", /* we use the blend tool */
|
|
"gegl:rectangle", /* pointless */
|
|
"gegl:seamless-clone", /* used in the seamless clone tool */
|
|
"gegl:text", /* we use gimp's text rendering */
|
|
"gegl:threshold", /* we use gimp's op */
|
|
"gegl:tile", /* pointless */
|
|
"gegl:unpremul", /* pointless */
|
|
"gegl:vector-stroke",
|
|
};
|
|
|
|
gchar **categories;
|
|
gint i;
|
|
|
|
/* Operations with no name are abstract base classes */
|
|
if (! name)
|
|
return TRUE;
|
|
|
|
/* use this flag to include all ops for testing */
|
|
if (g_getenv ("GIMP_TESTING_NO_GEGL_BLACKLIST"))
|
|
return FALSE;
|
|
|
|
if (g_str_has_prefix (name, "gimp"))
|
|
return TRUE;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (name_blacklist); i++)
|
|
{
|
|
if (! strcmp (name, name_blacklist[i]))
|
|
return TRUE;
|
|
}
|
|
|
|
if (! categories_str)
|
|
return FALSE;
|
|
|
|
categories = g_strsplit (categories_str, ":", 0);
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (category_blacklist); i++)
|
|
{
|
|
gint j;
|
|
|
|
for (j = 0; categories[j]; j++)
|
|
if (! strcmp (categories[j], category_blacklist[i]))
|
|
{
|
|
g_strfreev (categories);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
g_strfreev (categories);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/* Builds a GList of the class structures of all subtypes of type.
|
|
*/
|
|
static GList *
|
|
gimp_get_subtype_classes (GType type,
|
|
GList *classes)
|
|
{
|
|
GeglOperationClass *klass;
|
|
GType *ops;
|
|
const gchar *categories;
|
|
guint n_ops;
|
|
gint i;
|
|
|
|
if (! type)
|
|
return classes;
|
|
|
|
klass = GEGL_OPERATION_CLASS (g_type_class_ref (type));
|
|
ops = g_type_children (type, &n_ops);
|
|
|
|
categories = gegl_operation_class_get_key (klass, "categories");
|
|
|
|
if (! gimp_gegl_tool_operation_blacklisted (klass->name, categories))
|
|
classes = g_list_prepend (classes, klass);
|
|
|
|
for (i = 0; i < n_ops; i++)
|
|
classes = gimp_get_subtype_classes (ops[i], classes);
|
|
|
|
if (ops)
|
|
g_free (ops);
|
|
|
|
return classes;
|
|
}
|
|
|
|
static gint
|
|
gimp_gegl_tool_compare_operation_names (GeglOperationClass *a,
|
|
GeglOperationClass *b)
|
|
{
|
|
const gchar *name_a = gegl_operation_class_get_key (a, "title");
|
|
const gchar *name_b = gegl_operation_class_get_key (b, "title");
|
|
|
|
if (! name_a) name_a = a->name;
|
|
if (! name_b) name_b = b->name;
|
|
|
|
return strcmp (name_a, name_b);
|
|
}
|
|
|
|
static GList *
|
|
gimp_get_geglopclasses (void)
|
|
{
|
|
GList *opclasses;
|
|
|
|
opclasses = gimp_get_subtype_classes (GEGL_TYPE_OPERATION, NULL);
|
|
|
|
opclasses = g_list_sort (opclasses,
|
|
(GCompareFunc)
|
|
gimp_gegl_tool_compare_operation_names);
|
|
|
|
return opclasses;
|
|
}
|
|
|
|
|
|
/*****************/
|
|
/* Gegl dialog */
|
|
/*****************/
|
|
|
|
static void
|
|
gimp_gegl_tool_dialog (GimpFilterTool *filter_tool)
|
|
{
|
|
GimpGeglTool *tool = GIMP_GEGL_TOOL (filter_tool);
|
|
GimpOperationTool *o_tool = GIMP_OPERATION_TOOL (filter_tool);
|
|
GtkListStore *store;
|
|
GtkCellRenderer *cell;
|
|
GtkWidget *main_vbox;
|
|
GtkWidget *hbox;
|
|
GtkWidget *combo;
|
|
GtkWidget *options_gui;
|
|
GtkWidget *options_box;
|
|
GList *opclasses;
|
|
GList *iter;
|
|
|
|
GIMP_FILTER_TOOL_CLASS (parent_class)->dialog (filter_tool);
|
|
|
|
options_box = g_weak_ref_get (&o_tool->options_box_ref);
|
|
g_return_if_fail (options_box);
|
|
|
|
main_vbox = gimp_filter_tool_dialog_get_vbox (filter_tool);
|
|
|
|
/* The operation combo box */
|
|
hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
|
|
gtk_box_pack_start (GTK_BOX (main_vbox), hbox, FALSE, FALSE, 0);
|
|
gtk_box_reorder_child (GTK_BOX (main_vbox), hbox, 0);
|
|
gtk_widget_show (hbox);
|
|
|
|
store = gtk_list_store_new (N_COLUMNS,
|
|
G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
|
|
|
|
opclasses = gimp_get_geglopclasses ();
|
|
|
|
for (iter = opclasses; iter; iter = iter->next)
|
|
{
|
|
GeglOperationClass *opclass = GEGL_OPERATION_CLASS (iter->data);
|
|
const gchar *icon_name = NULL;
|
|
const gchar *op_name = opclass->name;
|
|
const gchar *title;
|
|
gchar *label;
|
|
|
|
if (g_str_has_prefix (opclass->name, "gegl:"))
|
|
icon_name = GIMP_ICON_GEGL;
|
|
|
|
if (g_str_has_prefix (op_name, "gegl:"))
|
|
op_name += strlen ("gegl:");
|
|
|
|
title = gegl_operation_class_get_key (opclass, "title");
|
|
|
|
if (title)
|
|
label = g_strdup_printf ("%s (%s)", title, op_name);
|
|
else
|
|
label = g_strdup (op_name);
|
|
|
|
gtk_list_store_insert_with_values (store, NULL, -1,
|
|
COLUMN_NAME, opclass->name,
|
|
COLUMN_LABEL, label,
|
|
COLUMN_ICON_NAME, icon_name,
|
|
-1);
|
|
|
|
g_free (label);
|
|
}
|
|
|
|
g_list_free (opclasses);
|
|
|
|
combo = gtk_combo_box_new_with_model (GTK_TREE_MODEL (store));
|
|
g_object_unref (store);
|
|
gtk_box_pack_start (GTK_BOX (hbox), combo, TRUE, TRUE, 0);
|
|
gtk_widget_show (combo);
|
|
|
|
cell = gtk_cell_renderer_pixbuf_new ();
|
|
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), cell, FALSE);
|
|
gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), cell,
|
|
"icon-name", COLUMN_ICON_NAME);
|
|
|
|
cell = gtk_cell_renderer_text_new ();
|
|
g_object_set (cell,
|
|
"ellipsize", PANGO_ELLIPSIZE_MIDDLE,
|
|
NULL);
|
|
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), cell, TRUE);
|
|
gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), cell,
|
|
"text", COLUMN_LABEL);
|
|
|
|
g_signal_connect (combo, "changed",
|
|
G_CALLBACK (gimp_gegl_tool_operation_changed),
|
|
tool);
|
|
|
|
tool->operation_combo = combo;
|
|
|
|
tool->description_label = gtk_label_new ("");
|
|
gtk_label_set_line_wrap (GTK_LABEL (tool->description_label), TRUE);
|
|
gtk_label_set_xalign (GTK_LABEL (tool->description_label), 0.0);
|
|
gtk_box_pack_start (GTK_BOX (main_vbox), tool->description_label,
|
|
FALSE, FALSE, 0);
|
|
gtk_box_reorder_child (GTK_BOX (main_vbox), tool->description_label, 1);
|
|
|
|
/* The options vbox */
|
|
options_gui = gtk_label_new (_("Select an operation from the list above"));
|
|
gimp_label_set_attributes (GTK_LABEL (options_gui),
|
|
PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC,
|
|
-1);
|
|
g_object_set (options_gui,
|
|
"margin-top", 4,
|
|
"margin-bottom", 4,
|
|
NULL);
|
|
gtk_container_add (GTK_CONTAINER (options_box), options_gui);
|
|
g_object_unref (options_box);
|
|
g_weak_ref_set (&o_tool->options_gui_ref, options_gui);
|
|
gtk_widget_show (options_gui);
|
|
}
|
|
|
|
static void
|
|
gimp_gegl_tool_halt (GimpGeglTool *gegl_tool)
|
|
{
|
|
GimpOperationTool *op_tool = GIMP_OPERATION_TOOL (gegl_tool);
|
|
|
|
gimp_operation_tool_set_operation (op_tool, NULL,
|
|
NULL, NULL, NULL, NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
gimp_gegl_tool_operation_changed (GtkWidget *widget,
|
|
GimpGeglTool *tool)
|
|
{
|
|
GtkTreeModel *model;
|
|
GtkTreeIter iter;
|
|
gchar *operation;
|
|
|
|
if (! gtk_combo_box_get_active_iter (GTK_COMBO_BOX (widget), &iter))
|
|
return;
|
|
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX (widget));
|
|
|
|
gtk_tree_model_get (model, &iter,
|
|
COLUMN_NAME, &operation,
|
|
-1);
|
|
|
|
if (operation)
|
|
{
|
|
const gchar *description;
|
|
|
|
description = gegl_operation_get_key (operation, "description");
|
|
|
|
if (description)
|
|
{
|
|
gtk_label_set_text (GTK_LABEL (tool->description_label), description);
|
|
gtk_widget_show (tool->description_label);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_hide (tool->description_label);
|
|
}
|
|
|
|
gimp_operation_tool_set_operation (GIMP_OPERATION_TOOL (tool),
|
|
operation,
|
|
_("GEGL Operation"),
|
|
_("GEGL Operation"),
|
|
NULL,
|
|
GIMP_ICON_GEGL,
|
|
GIMP_HELP_TOOL_GEGL);
|
|
g_free (operation);
|
|
}
|
|
}
|