PIKApp/app/propgui/pikapropgui-vignette.c

207 lines
6.7 KiB
C

/* 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-vignette.c
* Copyright (C) 2020 Ell
*
* 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 "libpikamath/pikamath.h"
#include "libpikawidgets/pikawidgets.h"
#include "propgui-types.h"
#include "core/pikacontext.h"
#include "pikapropgui.h"
#include "pikapropgui-generic.h"
#include "pikapropgui-vignette.h"
#include "pika-intl.h"
#define MAX_GAMMA 1000.0
static void
focus_callback (GObject *config,
GeglRectangle *area,
PikaLimitType type,
gdouble x,
gdouble y,
gdouble radius,
gdouble aspect_ratio,
gdouble angle,
gdouble inner_limit,
gdouble midpoint)
{
gdouble proportion;
gdouble squeeze;
gdouble scale;
g_object_set_data_full (G_OBJECT (config), "area",
g_memdup2 (area, sizeof (GeglRectangle)),
(GDestroyNotify) g_free);
g_object_get (config,
"proportion", &proportion,
NULL);
if (aspect_ratio >= 0.0)
scale = 1.0 - aspect_ratio;
else
scale = 1.0 / (1.0 + aspect_ratio);
scale /= 1.0 + proportion * ((gdouble) area->height / area->width - 1.0);
if (scale <= 1.0)
squeeze = +2.0 * atan (1.0 / scale - 1.0) / G_PI;
else
squeeze = -2.0 * atan (scale - 1.0) / G_PI;
g_object_set (config,
"shape", type,
"x", x / area->width,
"y", y / area->height,
"radius", 2.0 * radius / area->width,
"proportion", proportion,
"squeeze", squeeze,
"rotation", fmod (fmod (angle * 180.0 / G_PI, 360.0) + 360.0,
360.0),
"softness", 1.0 - inner_limit,
"gamma", midpoint < 1.0 ?
MIN (log (0.5) / log (midpoint), MAX_GAMMA) :
MAX_GAMMA,
NULL);
}
static void
config_notify (GObject *config,
const GParamSpec *pspec,
gpointer set_data)
{
PikaControllerFocusCallback set_func;
GeglRectangle *area;
PikaLimitType shape;
gdouble x, y;
gdouble radius;
gdouble proportion;
gdouble squeeze;
gdouble rotation;
gdouble softness;
gdouble gamma;
gdouble aspect_ratio;
set_func = g_object_get_data (G_OBJECT (config), "set-func");
area = g_object_get_data (G_OBJECT (config), "area");
g_object_get (config,
"shape", &shape,
"x", &x,
"y", &y,
"radius", &radius,
"proportion", &proportion,
"squeeze", &squeeze,
"rotation", &rotation,
"softness", &softness,
"gamma", &gamma,
NULL);
aspect_ratio = 1.0 + ((gdouble) area->height / area->width - 1.0) *
proportion;
if (squeeze >= 0.0)
aspect_ratio /= tan (+squeeze * G_PI / 2.0) + 1.0;
else
aspect_ratio *= tan (-squeeze * G_PI / 2.0) + 1.0;
if (aspect_ratio <= 1.0)
aspect_ratio = 1.0 - aspect_ratio;
else
aspect_ratio = 1.0 / aspect_ratio - 1.0;
set_func (set_data, area,
shape,
x * area->width,
y * area->height,
radius * area->width / 2.0,
aspect_ratio,
rotation / 180.0 * G_PI,
1.0 - softness,
pow (0.5, 1.0 / gamma));
}
GtkWidget *
_pika_prop_gui_new_vignette (GObject *config,
GParamSpec **param_specs,
guint n_param_specs,
GeglRectangle *area,
PikaContext *context,
PikaCreatePickerFunc create_picker_func,
PikaCreateControllerFunc create_controller_func,
gpointer creator)
{
GtkWidget *vbox;
g_return_val_if_fail (G_IS_OBJECT (config), NULL);
g_return_val_if_fail (param_specs != NULL, NULL);
g_return_val_if_fail (n_param_specs > 0, NULL);
g_return_val_if_fail (PIKA_IS_CONTEXT (context), NULL);
vbox = _pika_prop_gui_new_generic (config,
param_specs, n_param_specs,
area, context,
create_picker_func,
create_controller_func,
creator);
if (create_controller_func)
{
GCallback set_func;
gpointer set_data;
set_func = create_controller_func (creator,
PIKA_CONTROLLER_TYPE_FOCUS,
_("Vignette: "),
(GCallback) focus_callback,
config,
&set_data);
g_object_set_data (G_OBJECT (config), "set-func", set_func);
g_object_set_data_full (G_OBJECT (config), "area",
g_memdup2 (area, sizeof (GeglRectangle)),
(GDestroyNotify) g_free);
config_notify (config, NULL, set_data);
g_signal_connect (config, "notify",
G_CALLBACK (config_notify),
set_data);
}
return vbox;
}