gimp/plug-ins/common/sparkle.c

1131 lines
36 KiB
C
Raw Normal View History

/* Sparkle --- image filter plug-in for GIMP
1999-09-03 23:14:44 +00:00
* Copyright (C) 1996 by John Beale; ported to Gimp by Michael J. Hammel;
*
1999-09-03 23:14:44 +00:00
* It has been optimized a little, bugfixed and modified by Martin Weber
* for additional functionality. Also bugfixed by Seth Burgess (9/17/03)
* to take rowstrides into account when selections are present (bug #50911).
* Attempted reformatting.
1997-11-24 22:05:25 +00:00
*
* This program is free software: you can redistribute it and/or modify
1997-11-24 22:05:25 +00:00
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
1997-11-24 22:05:25 +00:00
* (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/>.
1997-11-24 22:05:25 +00:00
*
* You can contact Michael at mjhammel@csn.net
* You can contact Martin at martweb@gmx.net
* You can contact Seth at sjburges@gimp.org
1997-11-24 22:05:25 +00:00
*/
/*
* Sparkle 1.27 - simulate pixel bloom and diffraction effects
1997-11-24 22:05:25 +00:00
*/
#include "config.h"
2000-02-07 15:49:54 +00:00
#include <string.h>
#include <libgimp/gimp.h>
#include <libgimp/gimpui.h>
#include "libgimp/stdplugins-intl.h"
1997-11-24 22:05:25 +00:00
#define PLUG_IN_PROC "plug-in-sparkle"
#define PLUG_IN_BINARY "sparkle"
#define PLUG_IN_ROLE "gimp-sparkle"
#define MAX_CHANNELS 4
#define PSV 2 /* point spread value */
#define NATURAL 0
#define FOREGROUND 1
#define BACKGROUND 2
1997-11-24 22:05:25 +00:00
1999-09-03 23:14:44 +00:00
2019-09-03 22:15:37 +02:00
typedef struct _Sparkle Sparkle;
typedef struct _SparkleClass SparkleClass;
struct _Sparkle
{
GimpPlugIn parent_instance;
};
1999-09-03 23:14:44 +00:00
2019-09-03 22:15:37 +02:00
struct _SparkleClass
1997-11-24 22:05:25 +00:00
{
2019-09-03 22:15:37 +02:00
GimpPlugInClass parent_class;
1997-11-24 22:05:25 +00:00
};
2019-09-03 22:15:37 +02:00
#define SPARKLE_TYPE (sparkle_get_type ())
#define SPARKLE(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), SPARKLE_TYPE, Sparkle))
2019-09-03 22:15:37 +02:00
GType sparkle_get_type (void) G_GNUC_CONST;
static GList * sparkle_query_procedures (GimpPlugIn *plug_in);
static GimpProcedure * sparkle_create_procedure (GimpPlugIn *plug_in,
const gchar *name);
static GimpValueArray * sparkle_run (GimpProcedure *procedure,
GimpRunMode run_mode,
GimpImage *image,
gint n_drawables,
GimpDrawable **drawables,
GimpProcedureConfig *config,
2019-09-03 22:15:37 +02:00
gpointer run_data);
static gboolean sparkle_dialog (GimpProcedure *procedure,
GObject *config,
GimpDrawable *drawable);
2019-09-03 22:15:37 +02:00
static gint compute_luminosity (const guchar *pixel,
gboolean gray,
gboolean has_alpha,
gint inverse);
2019-09-03 22:15:37 +02:00
static gint compute_lum_threshold (GimpDrawable *drawable,
gdouble percentile,
gint inverse);
static void sparkle (GObject *config,
GimpDrawable *drawable,
2019-09-03 22:15:37 +02:00
GimpPreview *preview);
static void sparkle_preview (GtkWidget *widget,
GObject *config);
static void fspike (GObject *config,
GeglBuffer *src_buffer,
2019-09-03 22:15:37 +02:00
GeglBuffer *dest_buffer,
const Babl *format,
gint bytes,
gint x1,
gint y1,
gint x2,
gint y2,
gint xr,
gint yr,
gdouble inten,
gdouble length,
gdouble angle,
GRand *gr,
guchar *dest_buf);
static void rpnt (GObject *config,
GeglBuffer *dest_buffer,
2019-09-03 22:15:37 +02:00
const Babl *format,
gint x1,
gint y1,
gint x2,
gint y2,
gdouble xr,
gdouble yr,
gint bytes,
gdouble inten,
guchar color[MAX_CHANNELS],
guchar *dest_buf);
G_DEFINE_TYPE (Sparkle, sparkle, GIMP_TYPE_PLUG_IN)
GIMP_MAIN (SPARKLE_TYPE)
DEFINE_STD_SET_I18N
2019-09-03 22:15:37 +02:00
1997-11-24 22:05:25 +00:00
static gint num_sparkles;
static void
2019-09-03 22:15:37 +02:00
sparkle_class_init (SparkleClass *klass)
1997-11-24 22:05:25 +00:00
{
2019-09-03 22:15:37 +02:00
GimpPlugInClass *plug_in_class = GIMP_PLUG_IN_CLASS (klass);
plug_in_class->query_procedures = sparkle_query_procedures;
plug_in_class->create_procedure = sparkle_create_procedure;
plug_in_class->set_i18n = STD_SET_I18N;
1997-11-24 22:05:25 +00:00
}
static void
2019-09-03 22:15:37 +02:00
sparkle_init (Sparkle *sparkle)
1997-11-24 22:05:25 +00:00
{
2019-09-03 22:15:37 +02:00
}
1997-11-24 22:05:25 +00:00
2019-09-03 22:15:37 +02:00
static GList *
sparkle_query_procedures (GimpPlugIn *plug_in)
{
return g_list_append (NULL, g_strdup (PLUG_IN_PROC));
}
static GimpProcedure *
sparkle_create_procedure (GimpPlugIn *plug_in,
const gchar *name)
{
GimpProcedure *procedure = NULL;
2019-09-03 22:15:37 +02:00
if (! strcmp (name, PLUG_IN_PROC))
{
procedure = gimp_image_procedure_new (plug_in, name,
GIMP_PDB_PROC_TYPE_PLUGIN,
sparkle_run, NULL, NULL);
2019-09-03 22:15:37 +02:00
gimp_procedure_set_image_types (procedure, "RGB*, GRAY*");
gimp_procedure_set_sensitivity_mask (procedure,
GIMP_PROCEDURE_SENSITIVE_DRAWABLE);
2019-09-03 22:15:37 +02:00
gimp_procedure_set_menu_label (procedure, _("_Sparkle..."));
2019-09-03 22:15:37 +02:00
gimp_procedure_add_menu_path (procedure,
"<Image>/Filters/Light and Shadow/[Light]");
2019-09-03 22:15:37 +02:00
gimp_procedure_set_documentation (procedure,
_("Turn bright spots into "
"starry sparkles"),
2023-04-14 09:19:47 +02:00
_("Uses a percentage based luminosity "
"threshold to find candidate pixels "
"for adding some sparkles (spikes)."),
2019-09-03 22:15:37 +02:00
name);
gimp_procedure_set_attribution (procedure,
"John Beale, & (ported to GIMP v0.54) "
"Michael J. Hammel & ted to GIMP v1.0) "
"& Seth Burgess & Spencer Kimball",
"John Beale",
"Version 1.27, September 2003");
GIMP_PROC_ARG_DOUBLE (procedure, "lum-threshold",
_("Lu_minosity threshold"),
_("Adjust the luminosity threshold"),
0.0, 0.1, 0.01,
2019-09-03 22:15:37 +02:00
G_PARAM_READWRITE);
GIMP_PROC_ARG_DOUBLE (procedure, "flare-inten",
_("_Flare intensity"),
_("Adjust the flare intensity"),
2019-09-03 22:15:37 +02:00
0.0, 1.0, 0.5,
G_PARAM_READWRITE);
GIMP_PROC_ARG_INT (procedure, "spike-len",
_("Spi_ke length"),
_("Adjust the spike length (in pixels)"),
1, 100, 20,
2019-09-03 22:15:37 +02:00
G_PARAM_READWRITE);
GIMP_PROC_ARG_INT (procedure, "spike-points",
_("Spike _points"),
_("Adjust the number of spikes"),
1, 16, 4,
2019-09-03 22:15:37 +02:00
G_PARAM_READWRITE);
GIMP_PROC_ARG_INT (procedure, "spike-angle",
_("Spike angle (-_1: random)"),
_("Adjust the spike angle "
"(-1 causes a random angle to be chosen)"),
2019-09-03 22:15:37 +02:00
-1, 360, 15,
G_PARAM_READWRITE);
GIMP_PROC_ARG_DOUBLE (procedure, "density",
_("Spike _density"),
_("Adjust the spike density"),
2019-09-03 22:15:37 +02:00
0.0, 1.0, 1.0,
G_PARAM_READWRITE);
GIMP_PROC_ARG_DOUBLE (procedure, "transparency",
_("_Transparency"),
_("Adjust the opacity of the spikes"),
2019-09-03 22:15:37 +02:00
0.0, 1.0, 0.0,
G_PARAM_READWRITE);
GIMP_PROC_ARG_DOUBLE (procedure, "random-hue",
_("Random _hue"),
_("Adjust how much the hue should be "
"changed randomly"),
2019-09-03 22:15:37 +02:00
0.0, 1.0, 0.0,
G_PARAM_READWRITE);
GIMP_PROC_ARG_DOUBLE (procedure, "random-saturation",
_("R_andom saturation"),
_("Adjust how much the saturation should be "
"changed randomly"),
2019-09-03 22:15:37 +02:00
0.0, 1.0, 0.0,
G_PARAM_READWRITE);
GIMP_PROC_ARG_BOOLEAN (procedure, "preserve-luminosity",
_("Preserve l_uminosity"),
_("Should the luminosity be preserved?"),
2019-09-03 22:15:37 +02:00
FALSE,
G_PARAM_READWRITE);
GIMP_PROC_ARG_BOOLEAN (procedure, "inverse",
_("In_verse"),
_("Should the effect be inversed?"),
2019-09-03 22:15:37 +02:00
FALSE,
G_PARAM_READWRITE);
GIMP_PROC_ARG_BOOLEAN (procedure, "border",
_("Add _border"),
_("Draw a border of spikes around the image"),
2019-09-03 22:15:37 +02:00
FALSE,
G_PARAM_READWRITE);
GIMP_PROC_ARG_INT (procedure, "color-type",
_("Color type"),
_("Color of sparkles: { NATURAL (0), "
"FOREGROUND (1), BACKGROUND (2) }"),
2019-09-03 22:15:37 +02:00
0, 2, NATURAL,
G_PARAM_READWRITE);
}
return procedure;
}
1997-11-24 22:05:25 +00:00
2019-09-03 22:15:37 +02:00
static GimpValueArray *
sparkle_run (GimpProcedure *procedure,
GimpRunMode run_mode,
GimpImage *image,
gint n_drawables,
GimpDrawable **drawables,
GimpProcedureConfig *config,
2019-09-03 22:15:37 +02:00
gpointer run_data)
{
GimpDrawable *drawable;
gint x, y, w, h;
1997-11-24 22:05:25 +00:00
2019-09-03 22:15:37 +02:00
gegl_init (NULL, NULL);
if (n_drawables != 1)
{
GError *error = NULL;
g_set_error (&error, GIMP_PLUG_IN_ERROR, 0,
_("Procedure '%s' only works with one drawable."),
PLUG_IN_PROC);
return gimp_procedure_new_return_values (procedure,
GIMP_PDB_CALLING_ERROR,
error);
}
else
{
drawable = drawables[0];
}
if (! gimp_drawable_mask_intersect (drawable, &x, &y, &w, &h))
{
g_message (_("Region selected for filter is empty"));
2019-09-03 22:15:37 +02:00
return gimp_procedure_new_return_values (procedure,
GIMP_PDB_SUCCESS,
NULL);
}
if (run_mode == GIMP_RUN_INTERACTIVE && ! sparkle_dialog (procedure, G_OBJECT (config), drawable))
return gimp_procedure_new_return_values (procedure,
GIMP_PDB_CANCEL,
NULL);
1997-11-24 22:05:25 +00:00
if (gimp_drawable_is_rgb (drawable) ||
gimp_drawable_is_gray (drawable))
1997-11-24 22:05:25 +00:00
{
gimp_progress_init (_("Sparkling"));
sparkle (G_OBJECT (config), drawable, NULL);
1997-11-24 22:05:25 +00:00
if (run_mode != GIMP_RUN_NONINTERACTIVE)
gimp_displays_flush ();
1997-11-24 22:05:25 +00:00
}
else
{
2019-09-03 22:15:37 +02:00
return gimp_procedure_new_return_values (procedure,
GIMP_PDB_EXECUTION_ERROR,
NULL);
1997-11-24 22:05:25 +00:00
}
2019-09-03 22:15:37 +02:00
return gimp_procedure_new_return_values (procedure, GIMP_PDB_SUCCESS, NULL);
1997-11-24 22:05:25 +00:00
}
static gboolean
sparkle_dialog (GimpProcedure *procedure,
GObject *config,
GimpDrawable *drawable)
1997-11-24 22:05:25 +00:00
{
GtkWidget *dialog;
GtkWidget *preview;
GtkWidget *hbox;
GtkWidget *scale;
GtkListStore *store;
gboolean run;
1997-11-24 22:05:25 +00:00
gimp_ui_init (PLUG_IN_BINARY);
1997-11-24 22:05:25 +00:00
dialog = gimp_procedure_dialog_new (procedure,
GIMP_PROCEDURE_CONFIG (config),
_("Sparkle"));
1997-11-24 22:05:25 +00:00
gimp_dialog_set_alternative_button_order (GTK_DIALOG (dialog),
GTK_RESPONSE_OK,
GTK_RESPONSE_CANCEL,
-1);
Added parent window API to the GimpProgress interface and to the libgimp 2005-09-09 Michael Natterer <mitch@gimp.org> Added parent window API to the GimpProgress interface and to the libgimp progress stuff. Might look strange, but does the right thing in almost all cases (image window, file dialog, script-fu dialog etc). Fixes bug #62988. * app/core/gimpprogress.[ch]: added GimpProgress::get_window() which should return a toplevel window ID if the progress is in a window that wants to be the transient parent of plug-in dialogs. * app/widgets/gimpwidgets-utils.[ch] (gimp_window_get_native): new function which returns the window handle of a GtkWindow's GdkWindow. * app/widgets/gimpfiledialog.c: implement ::get_window(). * app/display/gimpdisplay.[ch]: ditto. Removed window handle API. * app/gui/gui-vtable.c: changed accordingly. * libgimpbase/gimpbaseenums.[ch] (enum GimpProgressCommand): added GIMP_PROGRESS_COMMAND_GET_WINDOW. * app/plug-in/plug-in-progress.[ch] (plug_in_progress_get_window): new function. Also renamed some functions to match the GimpProgress interface, and not the legacy PDB procedure names. * tools/pdbgen/pdb/progress.pdb * app/core/gimppdbprogress.c: implement get_window() on both sides of the wire, keeping backward compatibility (hopefully). * libgimp/gimpprogress.[ch]: deprecated gimp_progress_install() and added gimp_progress_install_vtable() which takes a vtable with padding to be extensible. Added get_window() vtable entry and dispatch it accordingly. Also added pulse() which was implemented in a hackish way before. Everything is of course backward compatible. * libgimp/gimpprogressbar.c: inmplement the get_window() stuff so a plug-in dialog containing a progress can be the transient parent of another dialog in another plug-in. * libgimp/gimpui.[ch] (gimp_ui_get_progress_window): new function which returns a foreign GdkWindow of this plug-ins progress window. Renamed gimp_window_set_transient_for_default_display() to gimp_window_set_transient() and make it use the progress' window handle instead of the display's (which is the right thing to do in almost all cases). * libgimp/gimp.def * libgimp/gimpui.def: add the new functions. * tools/pdbgen/enums.pl * app/pdb/internal_procs.c * app/pdb/progress_cmds.c * libgimp/gimpprogress_pdb.[ch]: regenerated. * libgimp/gimpexport.c * plug-ins/*/*.c: follow API change.
2005-09-09 18:07:31 +00:00
gimp_window_set_transient (GTK_WINDOW (dialog));
gtk_widget_set_size_request (dialog, 430, -1);
gtk_container_set_border_width (
GTK_CONTAINER (gtk_dialog_get_content_area (GTK_DIALOG (dialog))), 12);
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"lum-threshold", 1.0);
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"flare-inten", 1.0);
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"spike-len", 1.0);
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"spike-points", 1.0);
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"spike-angle", 1.0);
1999-09-03 23:14:44 +00:00
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"density", 1.0);
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"density", 1.0);;
1999-09-03 23:14:44 +00:00
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"transparency", 1.0);
gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"random-hue", 1.0);
1999-09-03 23:14:44 +00:00
scale = gimp_procedure_dialog_get_scale_entry (GIMP_PROCEDURE_DIALOG (dialog),
"random-saturation", 1.0);
gtk_widget_set_margin_bottom (scale, 12);
gimp_procedure_dialog_fill_box (GIMP_PROCEDURE_DIALOG (dialog),
"sparkle-bool-vbox",
"preserve-luminosity", "inverse",
"border", NULL);
gimp_procedure_dialog_get_label (GIMP_PROCEDURE_DIALOG (dialog),
"sparkle-bool-label",
_("Additional Options"),
FALSE, FALSE);
gimp_procedure_dialog_fill_frame (GIMP_PROCEDURE_DIALOG (dialog),
"sparkle-bool-frame", "sparkle-bool-label",
FALSE, "sparkle-bool-vbox");
/* colortype */
store = gimp_int_store_new (_("Natural color"), NATURAL,
_("Foreground color"), FOREGROUND,
_("Background color"), BACKGROUND,
NULL);
gimp_procedure_dialog_get_int_radio (GIMP_PROCEDURE_DIALOG (dialog),
"color-type", GIMP_INT_STORE (store));
hbox = gimp_procedure_dialog_fill_box (GIMP_PROCEDURE_DIALOG (dialog),
"sparkle-row", "sparkle-bool-frame",
"color-type", NULL);
gtk_orientable_set_orientation (GTK_ORIENTABLE (hbox),
GTK_ORIENTATION_HORIZONTAL);
gtk_box_set_spacing (GTK_BOX (hbox), 12);
gtk_box_set_homogeneous (GTK_BOX (hbox), TRUE);
gtk_widget_set_margin_bottom (hbox, 12);
preview = gimp_procedure_dialog_get_drawable_preview (GIMP_PROCEDURE_DIALOG (dialog),
"preview", drawable);
g_object_set_data (config, "drawable", drawable);
g_signal_connect (preview, "invalidated",
G_CALLBACK (sparkle_preview),
config);
g_signal_connect_swapped (config, "notify",
G_CALLBACK (gimp_preview_invalidate),
preview);
1999-09-03 23:14:44 +00:00
gimp_procedure_dialog_fill (GIMP_PROCEDURE_DIALOG (dialog),
"preview",
"lum-threshold", "flare-inten", "spike-len",
"spike-points", "spike-angle", "density",
"transparency", "random-hue",
"random-saturation", "sparkle-row",
NULL);
gtk_widget_show (dialog);
1997-11-24 22:05:25 +00:00
run = gimp_procedure_dialog_run (GIMP_PROCEDURE_DIALOG (dialog));
removed our own action_area API and use GtkDialog's one. Create all 2003-11-06 Michael Natterer <mitch@gimp.org> * libgimpwidgets/gimpdialog.[ch]: removed our own action_area API and use GtkDialog's one. Create all dialogs without separator. Changed almost everything else too. Fixes bug #125143. * libgimpwidgets/gimpquerybox.c * libgimpwidgets/gimpunitmenu.c: changed accordingly. * libgimp/gimpexport.[ch]: ditto. Renamed enum GimpExportReturnType to GimpExportReturn. * libgimp/gimpcompat.h: added a #define for the old name. * themes/Default/gtkrc: increased action_area border to 6 pixels. * app/display/gimpdisplayshell-filter-dialog.c * app/display/gimpdisplayshell-scale.c * app/display/gimpprogress.c * app/gui/brush-select.c * app/gui/channels-commands.c * app/gui/color-notebook.c * app/gui/convert-dialog.c * app/gui/file-new-dialog.c * app/gui/font-select.c * app/gui/gradient-editor-commands.c * app/gui/gradient-select.c * app/gui/grid-dialog.c * app/gui/image-commands.c * app/gui/info-window.c * app/gui/layers-commands.c * app/gui/module-browser.c * app/gui/offset-dialog.c * app/gui/palette-import-dialog.c * app/gui/palette-select.c * app/gui/pattern-select.c * app/gui/preferences-dialog.c * app/gui/qmask-commands.c * app/gui/resize-dialog.c * app/gui/resolution-calibrate-dialog.c * app/gui/stroke-dialog.c * app/gui/templates-commands.c * app/gui/user-install-dialog.c * app/gui/vectors-commands.c * app/tools/gimpcolorpickertool.c * app/tools/gimpcroptool.c * app/tools/gimpimagemaptool.c * app/tools/gimpmeasuretool.c * app/tools/gimptransformtool.c * app/widgets/gimptexteditor.c * app/widgets/gimptooldialog.[ch] * app/widgets/gimpviewabledialog.[ch] * app/widgets/gimpwidgets-utils.c: changed accordingly and increased the dialogs' outer borders to 6 pixels all over the place. * plug-ins/*/*.c: changed accordingly. The plug-ins may be arbitrarily broken, I tested none of them.
2003-11-06 15:27:05 +00:00
gtk_widget_destroy (dialog);
1997-11-24 22:05:25 +00:00
removed our own action_area API and use GtkDialog's one. Create all 2003-11-06 Michael Natterer <mitch@gimp.org> * libgimpwidgets/gimpdialog.[ch]: removed our own action_area API and use GtkDialog's one. Create all dialogs without separator. Changed almost everything else too. Fixes bug #125143. * libgimpwidgets/gimpquerybox.c * libgimpwidgets/gimpunitmenu.c: changed accordingly. * libgimp/gimpexport.[ch]: ditto. Renamed enum GimpExportReturnType to GimpExportReturn. * libgimp/gimpcompat.h: added a #define for the old name. * themes/Default/gtkrc: increased action_area border to 6 pixels. * app/display/gimpdisplayshell-filter-dialog.c * app/display/gimpdisplayshell-scale.c * app/display/gimpprogress.c * app/gui/brush-select.c * app/gui/channels-commands.c * app/gui/color-notebook.c * app/gui/convert-dialog.c * app/gui/file-new-dialog.c * app/gui/font-select.c * app/gui/gradient-editor-commands.c * app/gui/gradient-select.c * app/gui/grid-dialog.c * app/gui/image-commands.c * app/gui/info-window.c * app/gui/layers-commands.c * app/gui/module-browser.c * app/gui/offset-dialog.c * app/gui/palette-import-dialog.c * app/gui/palette-select.c * app/gui/pattern-select.c * app/gui/preferences-dialog.c * app/gui/qmask-commands.c * app/gui/resize-dialog.c * app/gui/resolution-calibrate-dialog.c * app/gui/stroke-dialog.c * app/gui/templates-commands.c * app/gui/user-install-dialog.c * app/gui/vectors-commands.c * app/tools/gimpcolorpickertool.c * app/tools/gimpcroptool.c * app/tools/gimpimagemaptool.c * app/tools/gimpmeasuretool.c * app/tools/gimptransformtool.c * app/widgets/gimptexteditor.c * app/widgets/gimptooldialog.[ch] * app/widgets/gimpviewabledialog.[ch] * app/widgets/gimpwidgets-utils.c: changed accordingly and increased the dialogs' outer borders to 6 pixels all over the place. * plug-ins/*/*.c: changed accordingly. The plug-ins may be arbitrarily broken, I tested none of them.
2003-11-06 15:27:05 +00:00
return run;
1997-11-24 22:05:25 +00:00
}
static gint
compute_luminosity (const guchar *pixel,
gboolean gray,
gboolean has_alpha,
gint inverse)
1997-11-24 22:05:25 +00:00
{
1999-09-03 23:14:44 +00:00
gint pixel0, pixel1, pixel2;
if (inverse)
1999-09-03 23:14:44 +00:00
{
pixel0 = 255 - pixel[0];
pixel1 = 255 - pixel[1];
pixel2 = 255 - pixel[2];
}
else
{
pixel0 = pixel[0];
pixel1 = pixel[1];
pixel2 = pixel[2];
}
1997-11-24 22:05:25 +00:00
if (gray)
{
if (has_alpha)
return (pixel0 * pixel1) / 255;
1997-11-24 22:05:25 +00:00
else
return (pixel0);
1997-11-24 22:05:25 +00:00
}
else
{
gint min, max;
1999-09-03 23:14:44 +00:00
min = MIN (pixel0, pixel1);
min = MIN (min, pixel2);
max = MAX (pixel0, pixel1);
max = MAX (max, pixel2);
1997-11-24 22:05:25 +00:00
if (has_alpha)
return ((min + max) * pixel[3]) / 510;
1997-11-24 22:05:25 +00:00
else
return (min + max) / 2;
1997-11-24 22:05:25 +00:00
}
}
static gint
compute_lum_threshold (GimpDrawable *drawable,
gdouble percentile,
gint inverse)
1997-11-24 22:05:25 +00:00
{
GeglBuffer *src_buffer;
GeglBufferIterator *iter;
const Babl *format;
gint bpp;
gint values[256];
gint total, sum;
gboolean gray;
gboolean has_alpha;
gint i;
gint x1, y1;
gint width, height;
1997-11-24 22:05:25 +00:00
/* zero out the luminosity values array */
memset (values, 0, sizeof (gint) * 256);
1997-11-24 22:05:25 +00:00
if (! gimp_drawable_mask_intersect (drawable,
&x1, &y1, &width, &height))
return 0;
gray = gimp_drawable_is_gray (drawable);
has_alpha = gimp_drawable_has_alpha (drawable);
1997-11-24 22:05:25 +00:00
if (gray)
{
if (has_alpha)
format = babl_format ("Y'A u8");
else
format = babl_format ("Y' u8");
}
else
{
if (has_alpha)
format = babl_format ("R'G'B'A u8");
else
format = babl_format ("R'G'B' u8");
}
bpp = babl_format_get_bytes_per_pixel (format);
1997-11-24 22:05:25 +00:00
src_buffer = gimp_drawable_get_buffer (drawable);
iter = gegl_buffer_iterator_new (src_buffer,
GEGL_RECTANGLE (x1, y1, width, height), 0,
format,
GEGL_ACCESS_READ, GEGL_ABYSS_NONE, 1);
while (gegl_buffer_iterator_next (iter))
{
const guchar *src = iter->items[0].data;
gint length = iter->length;
while (length--)
{
values [compute_luminosity (src, gray, has_alpha, inverse)]++;
src += bpp;
}
}
g_object_unref (src_buffer);
1997-11-24 22:05:25 +00:00
total = width * height;
1997-11-24 22:05:25 +00:00
sum = 0;
for (i = 255; i >= 0; i--)
{
sum += values[i];
if ((gdouble) sum > percentile * (gdouble) total)
{
num_sparkles = sum;
return i;
}
1997-11-24 22:05:25 +00:00
}
1997-11-24 22:05:25 +00:00
return 0;
}
static void
sparkle (GObject *config,
GimpDrawable *drawable,
GimpPreview *preview)
1997-11-24 22:05:25 +00:00
{
GeglBuffer *src_buffer;
GeglBuffer *dest_buffer;
GeglBufferIterator *iter;
const Babl *format;
gint d_width, d_height;
gdouble nfrac, length, inten, spike_angle;
gint cur_progress, max_progress;
gint x1, y1, x2, y2;
gint width, height;
gint threshold;
gint lum, x, y, b;
gboolean gray, has_alpha;
gint alpha;
gint bytes;
GRand *gr;
guchar *dest_buf = NULL;
gdouble lum_threshold;
gdouble flare_inten;
gint spike_len;
gint spike_pts;
gint spike_angle_config;
gdouble density;
gint inverse;
gint border;
g_object_get (config,
"lum-threshold", &lum_threshold,
"flare-inten", &flare_inten,
"spike-len", &spike_len,
"spike-points", &spike_pts,
"spike-angle", &spike_angle_config,
"density", &density,
"inverse", &inverse,
"border", &border,
NULL);
gray = gimp_drawable_is_gray (drawable);
has_alpha = gimp_drawable_has_alpha (drawable);
if (gray)
{
if (has_alpha)
format = babl_format ("Y'A u8");
else
format = babl_format ("Y' u8");
}
else
{
if (has_alpha)
format = babl_format ("R'G'B'A u8");
else
format = babl_format ("R'G'B' u8");
}
bytes = babl_format_get_bytes_per_pixel (format);
alpha = (has_alpha) ? bytes - 1 : bytes;
if (preview)
{
gimp_preview_get_position (preview, &x1, &y1);
gimp_preview_get_size (preview, &width, &height);
x2 = x1 + width;
y2 = y1 + height;
dest_buf = g_new0 (guchar, width * height * bytes);
}
else
{
if (! gimp_drawable_mask_intersect (drawable,
&x1, &y1, &width, &height))
2018-02-16 19:12:19 +01:00
return;
x2 = x1 + width;
y2 = y1 + height;
}
if (width < 1 || height < 1)
return;
d_width = gimp_drawable_get_width (drawable);
d_height = gimp_drawable_get_height (drawable);
gr = g_rand_new ();
if (border)
{
num_sparkles = 2 * (width + height);
threshold = 255;
}
else
{
/* compute the luminosity which exceeds the luminosity threshold */
threshold = compute_lum_threshold (drawable, lum_threshold, inverse);
}
1997-11-24 22:05:25 +00:00
/* initialize the progress dialog */
cur_progress = 0;
max_progress = num_sparkles;
/* copy what is already there */
src_buffer = gimp_drawable_get_buffer (drawable);
dest_buffer = gimp_drawable_get_shadow_buffer (drawable);
iter = gegl_buffer_iterator_new (src_buffer,
GEGL_RECTANGLE (x1, y1, width, height), 0,
format,
GEGL_ACCESS_READ, GEGL_ABYSS_NONE, 2);
gegl_buffer_iterator_add (iter, dest_buffer,
GEGL_RECTANGLE (x1, y1, width, height), 0,
format,
GEGL_ACCESS_WRITE, GEGL_ABYSS_NONE);
while (gegl_buffer_iterator_next (iter))
1997-11-24 22:05:25 +00:00
{
GeglRectangle roi = iter->items[0].roi;
const guchar *src, *s;
guchar *dest, *d;
src = iter->items[0].data;
if (preview)
dest = dest_buf + (((roi.y - y1) * width) + (roi.x - x1)) * bytes;
else
dest = iter->items[1].data;
1997-11-24 22:05:25 +00:00
for (y = 0; y < roi.height; y++)
1999-09-03 23:14:44 +00:00
{
s = src;
d = dest;
for (x = 0; x < roi.width; x++)
1999-09-03 23:14:44 +00:00
{
if (has_alpha && s[alpha] == 0)
{
memset (d, 0, alpha);
}
else
1999-09-03 23:14:44 +00:00
{
for (b = 0; b < alpha; b++)
d[b] = s[b];
1999-09-03 23:14:44 +00:00
}
if (has_alpha)
d[alpha] = s[alpha];
1999-09-03 23:14:44 +00:00
s += bytes;
d += bytes;
}
src += roi.width * bytes;
if (preview)
dest += width * bytes;
else
dest += roi.width * bytes;
1999-09-03 23:14:44 +00:00
}
1997-11-24 22:05:25 +00:00
}
1997-11-24 22:05:25 +00:00
/* add effects to new image based on intensity of old pixels */
iter = gegl_buffer_iterator_new (src_buffer,
GEGL_RECTANGLE (x1, y1, width, height), 0,
format,
GEGL_ACCESS_READ, GEGL_ABYSS_NONE, 2);
gegl_buffer_iterator_add (iter, dest_buffer,
GEGL_RECTANGLE (x1, y1, width, height), 0,
format,
GEGL_ACCESS_WRITE, GEGL_ABYSS_NONE);
while (gegl_buffer_iterator_next (iter))
1997-11-24 22:05:25 +00:00
{
GeglRectangle roi = iter->items[0].roi;
const guchar *src, *s;
src = iter->items[0].data;
for (y = 0; y < roi.height; y++)
{
s = src;
for (x = 0; x < roi.width; x++)
{
if (border)
{
if (x + roi.x == 0 ||
y + roi.y == 0 ||
x + roi.x == d_width - 1 ||
y + roi.y == d_height - 1)
{
lum = 255;
}
else
{
lum = 0;
}
}
else
{
lum = compute_luminosity (s, gray, has_alpha, inverse);
}
if (lum >= threshold)
{
nfrac = fabs ((gdouble) (lum + 1 - threshold) /
(gdouble) (256 - threshold));
length = ((gdouble) spike_len *
(gdouble) pow (nfrac, 0.8));
inten = flare_inten * nfrac;
/* fspike im x,y intens rlength angle */
if (spike_pts > 0)
{
/* major spikes */
if (spike_angle_config == -1)
spike_angle = g_rand_double_range (gr, 0, 360.0);
else
spike_angle = spike_angle_config;
if (g_rand_double (gr) <= density)
{
fspike (config, src_buffer, dest_buffer, format,
bytes, x1, y1, x2, y2,
x + roi.x, y + roi.y,
inten, length, spike_angle, gr, dest_buf);
/* minor spikes */
fspike (config, src_buffer, dest_buffer, format,
bytes, x1, y1, x2, y2,
x + roi.x, y + roi.y,
inten * 0.7, length * 0.7,
((gdouble) spike_angle + 180.0 / spike_pts),
gr, dest_buf);
}
}
if (!preview)
{
cur_progress ++;
if ((cur_progress % 5) == 0)
gimp_progress_update ((double) cur_progress /
(double) max_progress);
}
}
s += bytes;
}
src += roi.width * bytes;
}
}
g_object_unref (src_buffer);
g_object_unref (dest_buffer);
if (preview)
{
gimp_preview_draw_buffer (preview, dest_buf, width * bytes);
g_free (dest_buf);
}
else
{
gimp_progress_update (1.0);
1997-11-24 22:05:25 +00:00
gimp_drawable_merge_shadow (drawable, TRUE);
gimp_drawable_update (drawable, x1, y1, width, height);
}
configure.in app/core/gimpbrushpipe.c app/gui/about-dialog.c 2002-11-20 Dave Neary <bolsh@gimp.org> * configure.in * app/core/gimpbrushpipe.c * app/gui/about-dialog.c * app/paint-funcs/paint-funcs-generic.h * app/paint-funcs/paint-funcs.c * libgimpmath/gimpmath.h * libgimpwidgets/gimpwidgets.c * plug-ins/common/CML_explorer.c * plug-ins/common/blur.c * plug-ins/common/cubism.c * plug-ins/common/gee.c * plug-ins/common/gee_zoom.c * plug-ins/common/gqbist.c * plug-ins/common/jigsaw.c * plug-ins/common/lic.c * plug-ins/common/noisify.c * plug-ins/common/nova.c * plug-ins/common/papertile.c * plug-ins/common/plasma.c * plug-ins/common/randomize.c * plug-ins/common/sample_colorize.c * plug-ins/common/scatter_hsv.c * plug-ins/common/shift.c * plug-ins/common/sinus.c * plug-ins/common/smooth_palette.c * plug-ins/common/snoise.c * plug-ins/common/sparkle.c * plug-ins/common/spheredesigner.c * plug-ins/common/spread.c * plug-ins/common/warp.c * plug-ins/common/wind.c * plug-ins/flame/cmap.c * plug-ins/flame/flame.c * plug-ins/flame/libifs.c * plug-ins/gflare/gflare.c * plug-ins/gimpressionist/gimpressionist.c * plug-ins/gimpressionist/gimpressionist.h * plug-ins/gimpressionist/plasma.c * plug-ins/gimpressionist/repaint.c * plug-ins/ifscompose/ifscompose_utils.c * plug-ins/maze/algorithms.c * plug-ins/maze/maze.c * plug-ins/maze/maze.h * plug-ins/mosaic/mosaic.c: Change all occurrences of RAND_MAX, G_MAXRAND, rand(), srand(), lrand48(), srand48(), random(), srandom(), RAND_FUNC and SRAND_FUNC to the appropriate g_rand* equivalent. Programs which require seed setting for reproducible results, and anything in the core, gets a dedicated GRand * for the lifetime required. Programs which only ever used random numbers for tossing a coin, rolling a dice, etc use g_random functions. For the rest, judgement was used. Where it was easy, a GRand * object was used and g_rand_* functions were preferred. This fixes bug #67386 in HEAD.
2002-11-20 09:27:48 +00:00
g_rand_free (gr);
1997-11-24 22:05:25 +00:00
}
static void
sparkle_preview (GtkWidget *widget,
GObject *config)
{
GimpPreview *preview = GIMP_PREVIEW (widget);
GimpDrawable *drawable = g_object_get_data (config, "drawable");
sparkle (config, drawable, preview);
}
static inline void
rpnt (GObject *config,
GeglBuffer *dest_buffer,
const Babl *format,
gint x1,
gint y1,
gint x2,
gint y2,
gdouble xr,
gdouble yr,
gint bytes,
gdouble inten,
guchar color[MAX_CHANNELS],
guchar *dest_buf)
1997-11-24 22:05:25 +00:00
{
gint x, y, b;
gdouble dx, dy, rs, val;
guchar *pixel;
guchar pixel_buf[4];
gdouble new;
gdouble transparency;
gint preserve_luminosity;
gint inverse;
gint border;
g_object_get (config,
"transparency", &transparency,
"preserve-luminosity", &preserve_luminosity,
"inverse", &inverse,
"border", &border,
NULL);
1997-11-24 22:05:25 +00:00
x = (int) (xr); /* integer coord. to upper left of real point */
1997-11-24 22:05:25 +00:00
y = (int) (yr);
if (x >= x1 && y >= y1 && x < x2 && y < y2)
{
if (dest_buf)
{
pixel = dest_buf + ((y - y1) * (x2 - x1) + (x - x1)) * bytes;
}
else
{
gegl_buffer_sample (dest_buffer, x, y, NULL,
pixel_buf, format,
GEGL_SAMPLER_NEAREST, GEGL_ABYSS_NONE);
pixel = pixel_buf;
}
1997-11-24 22:05:25 +00:00
dx = xr - x; dy = yr - y;
rs = dx * dx + dy * dy;
1999-09-03 23:14:44 +00:00
val = inten * exp (-rs / PSV);
1997-11-24 22:05:25 +00:00
1999-09-03 23:14:44 +00:00
for (b = 0; b < bytes; b++)
{
if (inverse)
new = 255 - pixel[b];
else
new = pixel[b];
if (preserve_luminosity)
{
if (new < color[b])
{
new *= (1.0 - val * (1.0 - transparency));
}
else
{
new -= val * color[b] * (1.0 - transparency);
if (new < 0.0)
new = 0.0;
}
}
new *= 1.0 - val * transparency;
new += val * color[b];
1997-11-24 22:05:25 +00:00
if (new > 255)
new = 255;
if (inverse)
pixel[b] = 255 - new;
else
pixel[b] = new;
}
if (! dest_buf)
gegl_buffer_set (dest_buffer, GEGL_RECTANGLE (x, y, 1, 1), 0,
format, pixel_buf,
GEGL_AUTO_ROWSTRIDE);
}
1997-11-24 22:05:25 +00:00
}
static void
fspike (GObject *config,
GeglBuffer *src_buffer,
GeglBuffer *dest_buffer,
const Babl *format,
gint bytes,
gint x1,
gint y1,
gint x2,
gint y2,
gint xr,
gint yr,
gdouble inten,
gdouble length,
gdouble angle,
GRand *gr,
guchar *dest_buf)
1997-11-24 22:05:25 +00:00
{
const gdouble efac = 2.0;
gdouble xrt, yrt, dx, dy;
gdouble rpos;
gdouble in;
gdouble theta;
gdouble sfac;
gint i;
gboolean ok;
GeglColor *gegl_color;
guchar pixel[MAX_CHANNELS];
guchar chosen_color[MAX_CHANNELS];
guchar color[MAX_CHANNELS];
gint spike_pts;
gdouble random_hue;
gdouble random_saturation;
gint inverse;
gint colortype;
g_object_get (config,
"spike-points", &spike_pts,
"random-hue", &random_hue,
"random-saturation", &random_saturation,
"inverse", &inverse,
"color-type", &colortype,
NULL);
1997-11-24 22:05:25 +00:00
2000-01-02 22:30:20 +00:00
theta = angle;
1997-11-24 22:05:25 +00:00
switch (colortype)
2000-01-02 22:30:20 +00:00
{
case NATURAL:
break;
1997-11-24 22:05:25 +00:00
case FOREGROUND:
gegl_color = gimp_context_get_foreground ();
gegl_color_get_pixel (gegl_color, babl_format_with_space ("R'G'B' u8", NULL), chosen_color);
g_clear_object (&gegl_color);
break;
case BACKGROUND:
gegl_color = gimp_context_get_background ();
gegl_color_get_pixel (gegl_color, babl_format_with_space ("R'G'B' u8", NULL), chosen_color);
g_clear_object (&gegl_color);
break;
}
/* draw the major spikes */
for (i = 0; i < spike_pts; i++)
{
gegl_buffer_sample (dest_buffer, xr, yr, NULL, pixel, format,
GEGL_SAMPLER_NEAREST, GEGL_ABYSS_NONE);
if (colortype == NATURAL)
{
color[0] = pixel[0];
color[1] = pixel[1];
color[2] = pixel[2];
}
else
{
color[0] = chosen_color[0];
color[1] = chosen_color[1];
color[2] = chosen_color[2];
}
color[3] = pixel[3];
if (inverse)
{
color[0] = 255 - color[0];
color[1] = 255 - color[1];
color[2] = 255 - color[2];
2000-01-02 22:30:20 +00:00
}
if (random_hue > 0.0 || random_saturation > 0.0)
2000-01-02 22:30:20 +00:00
{
GimpRGB rgb;
GimpHSV hsv;
rgb.r = (gdouble) (255 - color[0]) / 255.0;
rgb.g = (gdouble) (255 - color[1]) / 255.0;
rgb.b = (gdouble) (255 - color[2]) / 255.0;
gimp_rgb_to_hsv (&rgb, &hsv);
hsv.h += random_hue * g_rand_double_range (gr, -0.5, 0.5);
if (hsv.h >= 1.0)
hsv.h -= 1.0;
else if (hsv.h < 0.0)
hsv.h += 1.0;
hsv.v += (random_saturation *
g_rand_double_range (gr, -1.0, 1.0));
hsv.v = CLAMP (hsv.v, 0.0, 1.0);
gimp_hsv_to_rgb (&hsv, &rgb);
color[0] = 255 - ROUND (rgb.r * 255.0);
color[1] = 255 - ROUND (rgb.g * 255.0);
color[2] = 255 - ROUND (rgb.b * 255.0);
}
1997-11-24 22:05:25 +00:00
2000-01-02 22:30:20 +00:00
dx = 0.2 * cos (theta * G_PI / 180.0);
dy = 0.2 * sin (theta * G_PI / 180.0);
xrt = (gdouble) xr; /* (gdouble) is needed because some */
yrt = (gdouble) yr; /* compilers optimize too much otherwise */
2000-01-02 22:30:20 +00:00
rpos = 0.2;
1997-11-24 22:05:25 +00:00
2000-01-02 22:30:20 +00:00
do
{
sfac = inten * exp (-pow (rpos / length, efac));
ok = FALSE;
2000-01-02 22:30:20 +00:00
in = 0.2 * sfac;
if (in > 0.01)
2000-01-02 22:30:20 +00:00
ok = TRUE;
rpnt (config, dest_buffer, format, x1, y1, x2, y2,
xrt, yrt,
bytes, in, color, dest_buf);
rpnt (config, dest_buffer, format, x1, y1, x2, y2,
xrt + 1.0, yrt,
bytes, in, color, dest_buf);
rpnt (config, dest_buffer, format, x1, y1, x2, y2,
xrt + 1.0, yrt + 1.0,
bytes, in, color, dest_buf);
rpnt (config, dest_buffer, format, x1, y1, x2, y2,
xrt, yrt + 1.0,
bytes, in, color, dest_buf);
xrt += dx;
yrt += dy;
rpos += 0.2;
} while (ok);
2000-01-02 22:30:20 +00:00
theta += 360.0 / spike_pts;
1997-11-24 22:05:25 +00:00
}
2000-01-02 22:30:20 +00:00
}