207 lines
6.7 KiB
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;
|
|
}
|