summaryrefslogtreecommitdiffstats
path: root/app/propgui/gimppropgui-vignette.c
diff options
context:
space:
mode:
Diffstat (limited to 'app/propgui/gimppropgui-vignette.c')
-rw-r--r--app/propgui/gimppropgui-vignette.c202
1 files changed, 202 insertions, 0 deletions
diff --git a/app/propgui/gimppropgui-vignette.c b/app/propgui/gimppropgui-vignette.c
new file mode 100644
index 0000000..5d8ef81
--- /dev/null
+++ b/app/propgui/gimppropgui-vignette.c
@@ -0,0 +1,202 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995-1997 Spencer Kimball and Peter Mattis
+ *
+ * gimppropgui-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 "libgimpmath/gimpmath.h"
+#include "libgimpwidgets/gimpwidgets.h"
+
+#include "propgui-types.h"
+
+#include "core/gimpcontext.h"
+
+#include "gimppropgui.h"
+#include "gimppropgui-generic.h"
+#include "gimppropgui-vignette.h"
+
+#include "gimp-intl.h"
+
+
+#define MAX_GAMMA 1000.0
+
+
+static void
+focus_callback (GObject *config,
+ GeglRectangle *area,
+ GimpLimitType 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_memdup (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)
+{
+ GimpControllerFocusCallback set_func;
+ GeglRectangle *area;
+ GimpLimitType 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 *
+_gimp_prop_gui_new_vignette (GObject *config,
+ GParamSpec **param_specs,
+ guint n_param_specs,
+ GeglRectangle *area,
+ GimpContext *context,
+ GimpCreatePickerFunc create_picker_func,
+ GimpCreateControllerFunc 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 (GIMP_IS_CONTEXT (context), NULL);
+
+ vbox = _gimp_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,
+ GIMP_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_memdup (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;
+}