PIKApp/app/propgui/pikapropgui.c

713 lines
25 KiB
C
Raw Permalink Normal View History

2023-09-26 00:35:21 +02:00
/* 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-1997 Spencer Kimball and Peter Mattis
*
* pikapropgui.c
* Copyright (C) 2002-2017 Michael Natterer <mitch@gimp.org>
* Sven Neumann <sven@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 <string.h>
#include <gegl.h>
#include <gegl-paramspecs.h>
#include <gtk/gtk.h>
#include "libpikacolor/pikacolor.h"
#include "libpikaconfig/pikaconfig.h"
#include "libpikawidgets/pikawidgets.h"
#include "propgui-types.h"
#include "gegl/pika-gegl-utils.h"
#include "operations/pika-operation-config.h"
#include "core/pikacontext.h"
#include "widgets/pikacolorpanel.h"
#include "widgets/pikamessagebox.h"
#include "widgets/pikapropwidgets.h"
#include "pikapropgui.h"
#include "pikapropgui-channel-mixer.h"
#include "pikapropgui-color-balance.h"
#include "pikapropgui-color-rotate.h"
#include "pikapropgui-color-to-alpha.h"
#include "pikapropgui-convolution-matrix.h"
#include "pikapropgui-diffraction-patterns.h"
#include "pikapropgui-eval.h"
#include "pikapropgui-focus-blur.h"
#include "pikapropgui-generic.h"
#include "pikapropgui-hue-saturation.h"
#include "pikapropgui-motion-blur-circular.h"
#include "pikapropgui-motion-blur-linear.h"
#include "pikapropgui-motion-blur-zoom.h"
#include "pikapropgui-newsprint.h"
#include "pikapropgui-panorama-projection.h"
#include "pikapropgui-recursive-transform.h"
#include "pikapropgui-shadows-highlights.h"
#include "pikapropgui-spiral.h"
#include "pikapropgui-supernova.h"
#include "pikapropgui-utils.h"
#include "pikapropgui-vignette.h"
#include "pika-intl.h"
#define HAS_KEY(p,k,v) pika_gegl_param_spec_has_key (p, k, v)
static gboolean pika_prop_string_to_boolean (GBinding *binding,
const GValue *from_value,
GValue *to_value,
gpointer user_data);
static void pika_prop_config_notify (GObject *config,
GParamSpec *pspec,
GtkWidget *widget);
static void pika_prop_widget_show (GtkWidget *widget,
GObject *config);
static void pika_prop_free_label_ref (GWeakRef *label_ref);
/* public functions */
GtkWidget *
pika_prop_widget_new (GObject *config,
const gchar *property_name,
GeglRectangle *area,
PikaContext *context,
PikaCreatePickerFunc create_picker_func,
PikaCreateControllerFunc create_controller_func,
gpointer creator,
const gchar **label)
{
GParamSpec *pspec;
g_return_val_if_fail (G_IS_OBJECT (config), NULL);
pspec = g_object_class_find_property (G_OBJECT_GET_CLASS (config),
property_name);
return pika_prop_widget_new_from_pspec (config, pspec, area, context,
create_picker_func,
create_controller_func,
creator,
label);
}
GtkWidget *
pika_prop_widget_new_from_pspec (GObject *config,
GParamSpec *pspec,
GeglRectangle *area,
PikaContext *context,
PikaCreatePickerFunc create_picker_func,
PikaCreateControllerFunc create_controller_func,
gpointer creator,
const gchar **label)
{
GtkWidget *widget = NULL;
g_return_val_if_fail (G_IS_OBJECT (config), NULL);
g_return_val_if_fail (pspec != NULL, NULL);
g_return_val_if_fail (PIKA_IS_CONTEXT (context), NULL);
g_return_val_if_fail (label != NULL, NULL);
*label = NULL;
if (GEGL_IS_PARAM_SPEC_SEED (pspec))
{
widget = pika_prop_random_seed_new (config, pspec->name);
*label = g_param_spec_get_nick (pspec);
}
else if (G_IS_PARAM_SPEC_INT (pspec) ||
G_IS_PARAM_SPEC_UINT (pspec) ||
G_IS_PARAM_SPEC_FLOAT (pspec) ||
G_IS_PARAM_SPEC_DOUBLE (pspec))
{
gdouble lower;
gdouble upper;
gdouble step;
gdouble page;
gint digits;
if (GEGL_IS_PARAM_SPEC_DOUBLE (pspec))
{
GeglParamSpecDouble *gspec = GEGL_PARAM_SPEC_DOUBLE (pspec);
lower = gspec->ui_minimum;
upper = gspec->ui_maximum;
step = gspec->ui_step_small;
page = gspec->ui_step_big;
digits = gspec->ui_digits;
}
else if (GEGL_IS_PARAM_SPEC_INT (pspec))
{
GeglParamSpecInt *gspec = GEGL_PARAM_SPEC_INT (pspec);
lower = gspec->ui_minimum;
upper = gspec->ui_maximum;
step = gspec->ui_step_small;
page = gspec->ui_step_big;
digits = 0;
}
else
{
gdouble value;
/* Get the min and max for the given property. */
_pika_prop_widgets_get_numeric_values (config, pspec,
&value, &lower, &upper,
G_STRFUNC);
if ((upper - lower <= 1.0) &&
(G_IS_PARAM_SPEC_FLOAT (pspec) ||
G_IS_PARAM_SPEC_DOUBLE (pspec)))
{
step = 0.01;
page = 0.1;
digits = 4;
}
else if ((upper - lower <= 10.0) &&
(G_IS_PARAM_SPEC_FLOAT (pspec) ||
G_IS_PARAM_SPEC_DOUBLE (pspec)))
{
step = 0.1;
page = 1.0;
digits = 3;
}
else
{
step = 1.0;
page = 10.0;
digits = (G_IS_PARAM_SPEC_FLOAT (pspec) ||
G_IS_PARAM_SPEC_DOUBLE (pspec)) ? 2 : 0;
}
}
widget = pika_prop_spin_scale_new (config, pspec->name,
step, page, digits);
if (HAS_KEY (pspec, "unit", "degree") &&
(upper - lower) == 360.0)
{
GtkWidget *hbox;
GtkWidget *dial;
gtk_spin_button_set_wrap (GTK_SPIN_BUTTON (widget), TRUE);
hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
gtk_box_pack_start (GTK_BOX (hbox), widget, TRUE, TRUE, 0);
gtk_widget_show (widget);
dial = pika_prop_angle_dial_new (config, pspec->name);
g_object_set (dial,
"clockwise-angles", HAS_KEY (pspec, "direction", "cw"),
NULL);
gtk_box_pack_start (GTK_BOX (hbox), dial, FALSE, FALSE, 0);
gtk_widget_show (dial);
pika_help_set_help_data (hbox, g_param_spec_get_blurb (pspec), NULL);
pika_prop_gui_bind_label (hbox, widget);
widget = hbox;
}
else if (HAS_KEY (pspec, "unit", "kelvin"))
{
GtkWidget *hbox;
GtkWidget *button;
hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
gtk_box_pack_start (GTK_BOX (hbox), widget, TRUE, TRUE, 0);
gtk_widget_show (widget);
button = pika_prop_kelvin_presets_new (config, pspec->name);
gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, FALSE, 0);
gtk_widget_show (button);
pika_help_set_help_data (hbox, g_param_spec_get_blurb (pspec), NULL);
pika_prop_gui_bind_label (hbox, widget);
widget = hbox;
}
else
{
pika_prop_gui_bind_label (widget, widget);
if (area &&
(HAS_KEY (pspec, "unit", "pixel-coordinate") ||
HAS_KEY (pspec, "unit", "pixel-distance")) &&
(HAS_KEY (pspec, "axis", "x") ||
HAS_KEY (pspec, "axis", "y")))
{
gdouble min = lower;
gdouble max = upper;
if (HAS_KEY (pspec, "unit", "pixel-coordinate"))
{
/* limit pixel coordinate scales to the actual area */
gint off_x = area->x;
gint off_y = area->y;
if (HAS_KEY (pspec, "axis", "x"))
{
min = MAX (lower, off_x);
max = MIN (upper, off_x + area->width);
}
else if (HAS_KEY (pspec, "axis","y"))
{
min = MAX (lower, off_y);
max = MIN (upper, off_y + area->height);
}
}
else if (HAS_KEY (pspec, "unit", "pixel-distance"))
{
/* limit pixel distance scales to the same value on the
* x and y axes, so linked values have the same range,
* we use MAX (width, height), see issue #2540
*/
max = MIN (upper, MAX (area->width, area->height));
}
pika_spin_scale_set_scale_limits (PIKA_SPIN_SCALE (widget),
min, max);
}
}
}
else if (G_IS_PARAM_SPEC_STRING (pspec))
{
*label = g_param_spec_get_nick (pspec);
if (PIKA_IS_PARAM_SPEC_CONFIG_PATH (pspec))
{
widget =
pika_prop_file_chooser_button_new (config, pspec->name,
g_param_spec_get_nick (pspec),
GTK_FILE_CHOOSER_ACTION_OPEN);
}
else if (HAS_KEY (pspec, "multiline", "true"))
{
GtkTextBuffer *buffer;
GtkWidget *view;
buffer = pika_prop_text_buffer_new (config, pspec->name, -1);
view = gtk_text_view_new_with_buffer (buffer);
g_object_unref (buffer);
widget = gtk_scrolled_window_new (NULL, NULL);
gtk_widget_set_size_request (widget, -1, 150);
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (widget),
GTK_SHADOW_IN);
gtk_container_add (GTK_CONTAINER (widget), view);
gtk_widget_show (view);
}
else if (HAS_KEY (pspec, "error", "true"))
{
GtkWidget *l;
widget = pika_message_box_new (PIKA_ICON_MASCOT_EEK);
pika_message_box_set_primary_text (PIKA_MESSAGE_BOX (widget), "%s",
*label);
pika_message_box_set_text (PIKA_MESSAGE_BOX (widget), "%s", "");
l = PIKA_MESSAGE_BOX (widget)->label[1];
g_object_bind_property (config, pspec->name,
l, "label",
G_BINDING_SYNC_CREATE);
g_object_bind_property_full (config, pspec->name,
widget, "visible",
G_BINDING_SYNC_CREATE,
pika_prop_string_to_boolean,
NULL,
NULL, NULL);
*label = NULL;
}
else
{
widget = pika_prop_entry_new (config, pspec->name, -1);
}
}
else if (G_IS_PARAM_SPEC_BOOLEAN (pspec))
{
widget = pika_prop_check_button_new (config, pspec->name,
g_param_spec_get_nick (pspec));
pika_prop_gui_bind_label (widget, widget);
}
else if (G_IS_PARAM_SPEC_ENUM (pspec))
{
widget = pika_prop_enum_combo_box_new (config, pspec->name, 0, 0);
pika_int_combo_box_set_label (PIKA_INT_COMBO_BOX (widget),
g_param_spec_get_nick (pspec));
pika_prop_gui_bind_label (widget, widget);
}
else if (PIKA_IS_PARAM_SPEC_RGB (pspec))
{
gboolean has_alpha;
GtkWidget *button;
has_alpha = pika_param_spec_rgb_has_alpha (pspec);
widget = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
button = pika_prop_color_button_new (config, pspec->name,
g_param_spec_get_nick (pspec),
128, 24,
has_alpha ?
PIKA_COLOR_AREA_SMALL_CHECKS :
PIKA_COLOR_AREA_FLAT);
pika_color_button_set_update (PIKA_COLOR_BUTTON (button), TRUE);
pika_color_panel_set_context (PIKA_COLOR_PANEL (button), context);
gtk_box_pack_start (GTK_BOX (widget), button, TRUE, TRUE, 0);
gtk_widget_show (button);
pika_prop_gui_bind_tooltip (button, widget);
if (create_picker_func)
{
button = create_picker_func (creator,
pspec->name,
PIKA_ICON_COLOR_PICKER_GRAY,
_("Pick color from the image"),
/* pick_abyss = */ FALSE,
NULL, NULL);
gtk_box_pack_start (GTK_BOX (widget), button, FALSE, FALSE, 0);
gtk_widget_show (button);
}
*label = g_param_spec_get_nick (pspec);
}
else
{
g_warning ("%s: not supported: %s (%s)\n", G_STRFUNC,
g_type_name (G_TYPE_FROM_INSTANCE (pspec)), pspec->name);
}
/* if we have any keys for dynamic properties, listen to config's notify
* signal, and update the properties accordingly.
*/
if (gegl_param_spec_get_property_key (pspec, "sensitive") ||
gegl_param_spec_get_property_key (pspec, "visible") ||
gegl_param_spec_get_property_key (pspec, "label") ||
gegl_param_spec_get_property_key (pspec, "description"))
{
g_object_set_data (G_OBJECT (widget), "pika-prop-pspec", pspec);
g_signal_connect_object (config, "notify",
G_CALLBACK (pika_prop_config_notify),
widget, 0);
if (gegl_param_spec_get_property_key (pspec, "visible"))
{
/* a bit of a hack: if we have a dynamic "visible" property key,
* connect to the widget's "show" signal, so that we can intercept
* our caller's gtk_widget_show() call, and keep the widget hidden if
* necessary.
*/
g_signal_connect (widget, "show",
G_CALLBACK (pika_prop_widget_show),
config);
}
/* update all the properties now */
pika_prop_config_notify (config, NULL, widget);
}
gtk_widget_show (widget);
return widget;
}
typedef GtkWidget * (* PikaPropGuiNewFunc) (GObject *config,
GParamSpec **param_specs,
guint n_param_specs,
GeglRectangle *area,
PikaContext *context,
PikaCreatePickerFunc create_picker_func,
PikaCreateControllerFunc create_controller_func,
gpointer creator);
static const struct
{
const gchar *config_type;
PikaPropGuiNewFunc gui_new_func;
}
gui_new_funcs[] =
{
{ "PikaColorBalanceConfig",
_pika_prop_gui_new_color_balance },
{ "PikaHueSaturationConfig",
_pika_prop_gui_new_hue_saturation },
{ "PikaGegl-gegl-color-rotate-config",
_pika_prop_gui_new_color_rotate },
{ "PikaGegl-gegl-color-to-alpha-config",
_pika_prop_gui_new_color_to_alpha },
{ "PikaGegl-gegl-convolution-matrix-config",
_pika_prop_gui_new_convolution_matrix },
{ "PikaGegl-gegl-channel-mixer-config",
_pika_prop_gui_new_channel_mixer },
{ "PikaGegl-gegl-diffraction-patterns-config",
_pika_prop_gui_new_diffraction_patterns },
{ "PikaGegl-gegl-focus-blur-config",
_pika_prop_gui_new_focus_blur },
{ "PikaGegl-gegl-motion-blur-circular-config",
_pika_prop_gui_new_motion_blur_circular },
{ "PikaGegl-gegl-motion-blur-linear-config",
_pika_prop_gui_new_motion_blur_linear },
{ "PikaGegl-gegl-motion-blur-zoom-config",
_pika_prop_gui_new_motion_blur_zoom },
{ "PikaGegl-gegl-newsprint-config",
_pika_prop_gui_new_newsprint },
{ "PikaGegl-gegl-panorama-projection-config",
_pika_prop_gui_new_panorama_projection },
{ "PikaGegl-gegl-recursive-transform-config",
_pika_prop_gui_new_recursive_transform },
{ "PikaGegl-gegl-shadows-highlights-config",
_pika_prop_gui_new_shadows_highlights },
{ "PikaGegl-gegl-spiral-config",
_pika_prop_gui_new_spiral },
{ "PikaGegl-gegl-supernova-config",
_pika_prop_gui_new_supernova },
{ "PikaGegl-gegl-vignette-config",
_pika_prop_gui_new_vignette },
{ NULL,
_pika_prop_gui_new_generic }
};
GtkWidget *
pika_prop_gui_new (GObject *config,
GType owner_type,
GParamFlags flags,
GeglRectangle *area,
PikaContext *context,
PikaCreatePickerFunc create_picker_func,
PikaCreateControllerFunc create_controller_func,
gpointer creator)
{
GtkWidget *gui = NULL;
GParamSpec **param_specs;
guint n_param_specs;
g_return_val_if_fail (G_IS_OBJECT (config), NULL);
g_return_val_if_fail (PIKA_IS_CONTEXT (context), NULL);
param_specs = pika_operation_config_list_properties (config,
owner_type, flags,
&n_param_specs);
if (param_specs)
{
const gchar *config_type_name = G_OBJECT_TYPE_NAME (config);
gint i;
for (i = 0; i < G_N_ELEMENTS (gui_new_funcs); i++)
{
if (! gui_new_funcs[i].config_type ||
! strcmp (gui_new_funcs[i].config_type, config_type_name))
{
g_printerr ("GUI new func match: %s\n",
gui_new_funcs[i].config_type ?
gui_new_funcs[i].config_type : "generic fallback");
gui = gui_new_funcs[i].gui_new_func (config,
param_specs, n_param_specs,
area,
context,
create_picker_func,
create_controller_func,
creator);
break;
}
}
g_free (param_specs);
}
else
{
gui = gtk_label_new (_("This operation has no editable properties"));
pika_label_set_attributes (GTK_LABEL (gui),
PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC,
-1);
g_object_set (gui,
"margin-top", 4,
"margin-bottom", 4,
NULL);
}
gtk_widget_show (gui);
return gui;
}
void
pika_prop_gui_bind_container (GtkWidget *source,
GtkWidget *target)
{
g_object_bind_property (source, "sensitive",
target, "sensitive",
G_BINDING_SYNC_CREATE);
g_object_bind_property (source, "visible",
target, "visible",
G_BINDING_SYNC_CREATE);
}
void
pika_prop_gui_bind_label (GtkWidget *source,
GtkWidget *target)
{
GWeakRef *label_ref;
const gchar *label;
/* we want to update "target"'s "label" property whenever the label
* expression associated with "source" is reevaluated, however, "source"
* might not itself have a "label" property we can bind to. just keep around
* a reference to "target", and update its label manually.
*/
g_return_if_fail (g_object_get_data (G_OBJECT (source),
"pika-prop-label-ref") == NULL);
label_ref = g_slice_new (GWeakRef);
g_weak_ref_init (label_ref, target);
g_object_set_data_full (G_OBJECT (source),
"pika-prop-label-ref", label_ref,
(GDestroyNotify) pika_prop_free_label_ref);
label = g_object_get_data (G_OBJECT (source), "pika-prop-label");
if (label)
g_object_set (target, "label", label, NULL);
/* note that "source" might be its own label widget, in which case there's no
* need to bind the rest of the properties.
*/
if (source != target)
pika_prop_gui_bind_tooltip (source, target);
}
void
pika_prop_gui_bind_tooltip (GtkWidget *source,
GtkWidget *target)
{
g_object_bind_property (source, "tooltip-text",
target, "tooltip-text",
G_BINDING_SYNC_CREATE);
}
/* private functions */
static gboolean
pika_prop_string_to_boolean (GBinding *binding,
const GValue *from_value,
GValue *to_value,
gpointer user_data)
{
const gchar *string = g_value_get_string (from_value);
g_value_set_boolean (to_value, string && *string);
return TRUE;
}
static void
pika_prop_config_notify (GObject *config,
GParamSpec *pspec,
GtkWidget *widget)
{
GParamSpec *widget_pspec;
GWeakRef *label_ref;
GtkWidget *label_widget;
gboolean sensitive;
gboolean visible;
gchar *label;
gchar *description;
widget_pspec = g_object_get_data (G_OBJECT (widget), "pika-prop-pspec");
label_ref = g_object_get_data (G_OBJECT (widget), "pika-prop-label-ref");
if (label_ref)
label_widget = g_weak_ref_get (label_ref);
else
label_widget = NULL;
sensitive = pika_prop_eval_boolean (config, widget_pspec, "sensitive", TRUE);
visible = pika_prop_eval_boolean (config, widget_pspec, "visible", TRUE);
label = pika_prop_eval_string (config, widget_pspec, "label",
g_param_spec_get_nick (widget_pspec));
description = pika_prop_eval_string (config, widget_pspec, "description",
g_param_spec_get_blurb (widget_pspec));
/* we store the label in (and pass ownership over it to) the widget's
* "pika-prop-label" key, so that we can use it to initialize the label
* widget's label in pika_prop_gui_bind_label() upon binding.
*/
g_object_set_data_full (G_OBJECT (widget), "pika-prop-label", label, g_free);
g_signal_handlers_block_by_func (widget,
pika_prop_widget_show, config);
gtk_widget_set_sensitive (widget, sensitive);
gtk_widget_set_visible (widget, visible);
if (label_widget) g_object_set (label_widget, "label", label, NULL);
pika_help_set_help_data (widget, description, NULL);
g_signal_handlers_unblock_by_func (widget,
pika_prop_widget_show, config);
g_free (description);
if (label_widget)
g_object_unref (label_widget);
}
static void
pika_prop_widget_show (GtkWidget *widget,
GObject *config)
{
GParamSpec *widget_pspec;
gboolean visible;
widget_pspec = g_object_get_data (G_OBJECT (widget), "pika-prop-pspec");
visible = pika_prop_eval_boolean (config, widget_pspec, "visible", TRUE);
gtk_widget_set_visible (widget, visible);
}
static void
pika_prop_free_label_ref (GWeakRef *label_ref)
{
g_weak_ref_clear (label_ref);
g_slice_free (GWeakRef, label_ref);
}