2007-06-06 16:44:52 +08:00
|
|
|
/* Warp --- image filter plug-in for GIMP
|
1999-09-04 07:14:44 +08:00
|
|
|
* Copyright (C) 1997 John P. Beale
|
|
|
|
* Much of the 'warp' is from the Displace plug-in: 1996 Stephen Robert Norris
|
2003-11-06 23:27:05 +08:00
|
|
|
* Much of the 'displace' code taken in turn from the pinch plug-in
|
1999-09-04 07:14:44 +08:00
|
|
|
* which is by 1996 Federico Mena Quintero
|
|
|
|
*
|
2009-01-18 06:28:01 +08:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
1999-09-04 07:14:44 +08:00
|
|
|
* it under the terms of the GNU General Public License as published by
|
2009-01-18 06:28:01 +08:00
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
1999-09-04 07:14:44 +08: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
|
2018-07-12 05:27:07 +08:00
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
1999-09-04 07:14:44 +08:00
|
|
|
*
|
|
|
|
* You can contact me (the warp author) at beale@best.com
|
|
|
|
* Please send me any patches or enhancements to this code.
|
2007-06-06 16:44:52 +08:00
|
|
|
* You can contact the original GIMP authors at gimp@xcf.berkeley.edu
|
1999-09-04 07:14:44 +08:00
|
|
|
*
|
|
|
|
* --------------------------------------------------------------------
|
|
|
|
* Warp Program structure: after running the user interface and setting the
|
2003-11-06 23:27:05 +08:00
|
|
|
* parameters, warp generates a brand-new image (later to be deleted
|
1999-09-04 07:14:44 +08:00
|
|
|
* before the user ever sees it) which contains two grayscale layers,
|
|
|
|
* representing the X and Y gradients of the "control" image. For this
|
|
|
|
* purpose, all channels of the control image are summed for a scalar
|
|
|
|
* value at each pixel coordinate for the gradient operation.
|
|
|
|
*
|
2005-03-04 23:12:29 +08:00
|
|
|
* The X,Y components of the calculated gradient are then used to
|
|
|
|
* displace pixels from the source image into the destination
|
|
|
|
* image. The displacement vector is rotated a user-specified amount
|
|
|
|
* first. This displacement operation happens iteratively, generating
|
|
|
|
* a new displaced image from each prior image.
|
1999-09-04 07:14:44 +08:00
|
|
|
* -------------------------------------------------------------------
|
|
|
|
*
|
|
|
|
* Revision History:
|
|
|
|
* Version 0.37 12/19/98 Fixed Tooltips and freeing memory
|
|
|
|
* Version 0.36 11/9/97 Changed XY vector layers back to own image
|
|
|
|
* fixed 'undo' problem (hopefully)
|
|
|
|
*
|
|
|
|
* Version 0.35 11/3/97 Added vector-map, mag-map, grad-map to
|
|
|
|
* diff vector instead of separate operation
|
|
|
|
* further futzing with drawable updates
|
|
|
|
* starting adding tooltips
|
|
|
|
*
|
|
|
|
* Version 0.34 10/30/97 'Fixed' drawable update problem
|
|
|
|
* Added 16-bit resolution to differential map
|
2008-10-20 14:04:39 +08:00
|
|
|
* Added substep increments for finer control
|
1999-09-04 07:14:44 +08:00
|
|
|
*
|
|
|
|
* Version 0.33 10/26/97 Added 'angle increment' to user interface
|
|
|
|
*
|
|
|
|
* Version 0.32 10/25/97 Added magnitude control map (secondary control)
|
|
|
|
* Changed undo behavior to be one undo-step per warp call.
|
|
|
|
*
|
|
|
|
* Version 0.31 10/25/97 Fixed src/dest pixregions so program works
|
|
|
|
* with multiple-layer images. Still don't know
|
2005-03-04 23:12:29 +08:00
|
|
|
* exactly what I did to fix it :-/ Also, added 'color' option
|
|
|
|
* for border pixels to use the current selected foreground color.
|
1999-09-04 07:14:44 +08:00
|
|
|
*
|
|
|
|
* Version 0.3 10/20/97 Initial release for Gimp 0.99.xx
|
|
|
|
*/
|
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
#include "config.h"
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
#include <libgimp/gimp.h>
|
|
|
|
#include <libgimp/gimpui.h>
|
|
|
|
|
1999-09-26 13:16:19 +08:00
|
|
|
#include "libgimp/stdplugins-intl.h"
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-05-02 04:22:55 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
/* Some useful macros */
|
|
|
|
|
2005-08-16 06:42:34 +08:00
|
|
|
#define PLUG_IN_PROC "plug-in-warp"
|
|
|
|
#define PLUG_IN_BINARY "warp"
|
2011-04-09 02:31:34 +08:00
|
|
|
#define PLUG_IN_ROLE "gimp-warp"
|
2000-01-26 01:46:56 +08:00
|
|
|
#define ENTRY_WIDTH 75
|
|
|
|
#define MIN_ARGS 6 /* minimum number of arguments required */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-26 01:46:56 +08:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
WRAP,
|
|
|
|
SMEAR,
|
|
|
|
BLACK,
|
|
|
|
COLOR
|
|
|
|
};
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-18 01:02:26 +08:00
|
|
|
typedef struct
|
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
gdouble amount;
|
|
|
|
gint warp_map_id;
|
|
|
|
gint iter;
|
|
|
|
gdouble dither;
|
|
|
|
gdouble angle;
|
|
|
|
gint wrap_type;
|
|
|
|
gint mag_map_id;
|
|
|
|
gint mag_use;
|
|
|
|
gint substeps;
|
|
|
|
gint grad_map_id;
|
|
|
|
gdouble grad_scale;
|
|
|
|
gint vector_map_id;
|
|
|
|
gdouble vector_scale;
|
|
|
|
gdouble vector_angle;
|
1999-09-04 07:14:44 +08:00
|
|
|
} WarpVals;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Function prototypes.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void query (void);
|
2003-07-02 21:00:16 +08:00
|
|
|
static void run (const gchar *name,
|
2008-10-20 14:04:39 +08:00
|
|
|
gint nparams,
|
|
|
|
const GimpParam *param,
|
|
|
|
gint *nreturn_vals,
|
|
|
|
GimpParam **return_vals);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
static void blur16 (GimpDrawable *drawable);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
static void diff (GimpDrawable *drawable,
|
|
|
|
GimpDrawable **xl,
|
|
|
|
GimpDrawable **yl);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
static void diff_prepare_row (GeglBuffer *buffer,
|
|
|
|
const Babl *format,
|
2008-10-20 14:04:39 +08:00
|
|
|
guchar *data,
|
|
|
|
gint x,
|
|
|
|
gint y,
|
|
|
|
gint w);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
static void warp_one (GimpDrawable *draw,
|
|
|
|
GimpDrawable *newid,
|
|
|
|
GimpDrawable *map_x,
|
|
|
|
GimpDrawable *map_y,
|
|
|
|
GimpDrawable *mag_draw,
|
2008-10-20 14:04:39 +08:00
|
|
|
gboolean first_time,
|
|
|
|
gint step);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
static void warp (GimpDrawable *drawable);
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
static gboolean warp_dialog (GimpDrawable *drawable);
|
2019-07-19 19:10:42 +08:00
|
|
|
static void warp_pixel (GeglBuffer *buffer,
|
|
|
|
const Babl *format,
|
2008-10-20 14:04:39 +08:00
|
|
|
gint width,
|
|
|
|
gint height,
|
|
|
|
gint x1,
|
|
|
|
gint y1,
|
|
|
|
gint x2,
|
|
|
|
gint y2,
|
|
|
|
gint x,
|
|
|
|
gint y,
|
|
|
|
guchar *pixel);
|
2000-05-02 04:22:55 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
static gboolean warp_map_constrain (GimpImage *image,
|
|
|
|
GimpItem *item,
|
2008-10-20 14:04:39 +08:00
|
|
|
gpointer data);
|
2003-11-06 23:27:05 +08:00
|
|
|
static gdouble warp_map_mag_give_value (guchar *pt,
|
2008-10-20 14:04:39 +08:00
|
|
|
gint alpha,
|
|
|
|
gint bytes);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-18 01:02:26 +08:00
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
/* Variables global over entire plug-in scope */
|
|
|
|
/* -------------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-05-16 20:26:20 +08:00
|
|
|
const GimpPlugInInfo PLUG_IN_INFO =
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2000-01-26 01:46:56 +08:00
|
|
|
NULL, /* init_proc */
|
|
|
|
NULL, /* quit_proc */
|
|
|
|
query, /* query_proc */
|
|
|
|
run, /* run_proc */
|
1999-09-04 07:14:44 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static WarpVals dvals =
|
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
10.0, /* amount */
|
|
|
|
-1, /* warp_map_id */
|
|
|
|
5, /* iterations */
|
|
|
|
0.0, /* dither */
|
|
|
|
90.0, /* angle */
|
|
|
|
WRAP, /* wrap_type */
|
|
|
|
-1, /* mag_map_id */
|
|
|
|
FALSE, /* mag_use */
|
|
|
|
1, /* substeps */
|
|
|
|
-1, /* grad_map_id */
|
|
|
|
0.0, /* grad_scale */
|
|
|
|
-1, /* vector_map_id */
|
|
|
|
0.0, /* vector_scale */
|
|
|
|
0.0 /* vector_angle */
|
1999-09-04 07:14:44 +08:00
|
|
|
};
|
|
|
|
|
2000-01-18 01:02:26 +08:00
|
|
|
/* -------------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-26 01:46:56 +08:00
|
|
|
static gint progress = 0; /* progress indicator bar */
|
2006-04-27 19:33:51 +08:00
|
|
|
static GimpRunMode run_mode; /* interactive, non-, etc. */
|
2000-01-26 01:46:56 +08:00
|
|
|
static guchar color_pixel[4] = {0, 0, 0, 255}; /* current fg color */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-18 01:02:26 +08:00
|
|
|
/* -------------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/***** Functions *****/
|
|
|
|
|
|
|
|
MAIN ()
|
|
|
|
|
|
|
|
static void
|
2000-01-18 01:02:26 +08:00
|
|
|
query (void)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2006-05-16 20:26:20 +08:00
|
|
|
static const GimpParamDef args[] =
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2009-01-20 04:11:36 +08:00
|
|
|
{ GIMP_PDB_INT32, "run-mode", "The run mode { RUN-INTERACTIVE (0), RUN-NONINTERACTIVE (1) }" },
|
2005-08-16 06:42:34 +08:00
|
|
|
{ GIMP_PDB_IMAGE, "image", "Input image (unused)" },
|
|
|
|
{ GIMP_PDB_DRAWABLE, "drawable", "Input drawable" },
|
|
|
|
{ GIMP_PDB_FLOAT, "amount", "Pixel displacement multiplier" },
|
|
|
|
{ GIMP_PDB_DRAWABLE, "warp-map", "Displacement control map" },
|
|
|
|
{ GIMP_PDB_INT32, "iter", "Iteration count (last required argument)" },
|
|
|
|
{ GIMP_PDB_FLOAT, "dither", "Random dither amount (first optional argument)" },
|
|
|
|
{ GIMP_PDB_FLOAT, "angle", "Angle of gradient vector rotation" },
|
|
|
|
{ GIMP_PDB_INT32, "wrap-type", "Edge behavior: { WRAP (0), SMEAR (1), BLACK (2), COLOR (3) }" },
|
|
|
|
{ GIMP_PDB_DRAWABLE, "mag-map", "Magnitude control map" },
|
|
|
|
{ GIMP_PDB_INT32, "mag-use", "Use magnitude map: { FALSE (0), TRUE (1) }" },
|
|
|
|
{ GIMP_PDB_INT32, "substeps", "Substeps between image updates" },
|
|
|
|
{ GIMP_PDB_INT32, "grad-map", "Gradient control map" },
|
|
|
|
{ GIMP_PDB_FLOAT, "grad-scale", "Scaling factor for gradient map (0=don't use)" },
|
|
|
|
{ GIMP_PDB_INT32, "vector-map", "Fixed vector control map" },
|
|
|
|
{ GIMP_PDB_FLOAT, "vector-scale", "Scaling factor for fixed vector map (0=don't use)" },
|
|
|
|
{ GIMP_PDB_FLOAT, "vector-angle", "Angle for fixed vector map" }
|
1999-09-04 07:14:44 +08:00
|
|
|
};
|
|
|
|
|
2005-08-16 06:42:34 +08:00
|
|
|
gimp_install_procedure (PLUG_IN_PROC,
|
2008-10-20 14:04:39 +08:00
|
|
|
N_("Twist or smear image in many different ways"),
|
|
|
|
"Smears an image along vector paths calculated as "
|
|
|
|
"the gradient of a separate control matrix. The "
|
|
|
|
"effect can look like brushstrokes of acrylic or "
|
|
|
|
"watercolor paint, in some cases.",
|
|
|
|
"John P. Beale",
|
|
|
|
"John P. Beale",
|
|
|
|
"1997",
|
|
|
|
N_("_Warp..."),
|
|
|
|
"RGB*, GRAY*",
|
2019-08-31 08:43:48 +08:00
|
|
|
GIMP_PDB_PROC_TYPE_PLUGIN,
|
2008-10-20 14:04:39 +08:00
|
|
|
G_N_ELEMENTS (args), 0,
|
|
|
|
args, NULL);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
|
|
|
gimp_plugin_menu_register (PLUG_IN_PROC, "<Image>/Filters/Map");
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-07-02 21:00:16 +08:00
|
|
|
run (const gchar *name,
|
|
|
|
gint nparams,
|
|
|
|
const GimpParam *param,
|
|
|
|
gint *nreturn_vals,
|
|
|
|
GimpParam **return_vals)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2001-01-15 08:06:43 +08:00
|
|
|
static GimpParam values[1];
|
2006-06-21 18:22:14 +08:00
|
|
|
GimpPDBStatusType status = GIMP_PDB_SUCCESS;
|
2019-08-31 08:43:48 +08:00
|
|
|
GimpDrawable *drawable;
|
2001-01-15 08:06:43 +08:00
|
|
|
GimpRGB color;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2003-03-26 00:38:19 +08:00
|
|
|
INIT_I18N ();
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_init (NULL, NULL);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* get currently selected foreground pixel color */
|
2004-09-23 02:43:09 +08:00
|
|
|
gimp_context_get_foreground (&color);
|
2003-11-06 23:27:05 +08:00
|
|
|
gimp_rgb_get_uchar (&color,
|
2008-10-20 14:04:39 +08:00
|
|
|
&color_pixel[0],
|
|
|
|
&color_pixel[1],
|
|
|
|
&color_pixel[2]);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
run_mode = param[0].data.d_int32;
|
2019-09-03 16:48:32 +08:00
|
|
|
drawable = gimp_drawable_get_by_id (param[2].data.d_drawable);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
*nreturn_vals = 1;
|
2000-01-26 01:46:56 +08:00
|
|
|
*return_vals = values;
|
2002-01-16 02:35:29 +08:00
|
|
|
|
2000-08-22 09:26:57 +08:00
|
|
|
values[0].type = GIMP_PDB_STATUS;
|
1999-09-04 07:14:44 +08:00
|
|
|
values[0].data.d_status = status;
|
|
|
|
|
|
|
|
switch (run_mode)
|
|
|
|
{
|
2000-08-22 09:26:57 +08:00
|
|
|
case GIMP_RUN_INTERACTIVE:
|
2005-08-16 06:42:34 +08:00
|
|
|
gimp_get_data (PLUG_IN_PROC, &dvals);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
if (! warp_dialog (drawable))
|
2008-10-20 14:04:39 +08:00
|
|
|
return;
|
1999-09-04 07:14:44 +08:00
|
|
|
break;
|
|
|
|
|
2000-08-22 09:26:57 +08:00
|
|
|
case GIMP_RUN_NONINTERACTIVE:
|
2000-01-18 01:02:26 +08:00
|
|
|
/* Make sure minimum args
|
2003-11-06 23:27:05 +08:00
|
|
|
* (mode, image, draw, amount, warp_map, iter) are there
|
2000-01-18 01:02:26 +08:00
|
|
|
*/
|
1999-09-04 07:14:44 +08:00
|
|
|
if (nparams < MIN_ARGS)
|
2008-10-20 14:04:39 +08:00
|
|
|
{
|
|
|
|
status = GIMP_PDB_CALLING_ERROR;
|
|
|
|
}
|
2000-01-26 01:46:56 +08:00
|
|
|
else
|
2008-10-20 14:04:39 +08:00
|
|
|
{
|
2006-06-21 18:22:14 +08:00
|
|
|
gint pcnt = MIN_ARGS;
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
dvals.amount = param[3].data.d_float;
|
|
|
|
dvals.warp_map_id = param[4].data.d_int32;
|
|
|
|
dvals.iter = param[5].data.d_int32;
|
|
|
|
|
|
|
|
if (nparams > pcnt++) dvals.dither = param[6].data.d_float;
|
|
|
|
if (nparams > pcnt++) dvals.angle = param[7].data.d_float;
|
|
|
|
if (nparams > pcnt++) dvals.wrap_type = param[8].data.d_int32;
|
|
|
|
if (nparams > pcnt++) dvals.mag_map_id = param[9].data.d_int32;
|
|
|
|
if (nparams > pcnt++) dvals.mag_use = param[10].data.d_int32;
|
|
|
|
if (nparams > pcnt++) dvals.substeps = param[11].data.d_int32;
|
|
|
|
if (nparams > pcnt++) dvals.grad_map_id = param[12].data.d_int32;
|
|
|
|
if (nparams > pcnt++) dvals.grad_scale = param[13].data.d_float;
|
|
|
|
if (nparams > pcnt++) dvals.vector_map_id = param[14].data.d_int32;
|
|
|
|
if (nparams > pcnt++) dvals.vector_scale = param[15].data.d_float;
|
|
|
|
if (nparams > pcnt++) dvals.vector_angle = param[16].data.d_float;
|
2008-10-20 14:04:39 +08:00
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
break;
|
|
|
|
|
2000-08-22 09:26:57 +08:00
|
|
|
case GIMP_RUN_WITH_LAST_VALS:
|
2005-08-16 06:42:34 +08:00
|
|
|
gimp_get_data (PLUG_IN_PROC, &dvals);
|
1999-09-04 07:14:44 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2000-08-22 09:26:57 +08:00
|
|
|
if (status == GIMP_PDB_SUCCESS)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
warp (drawable);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-08-22 09:26:57 +08:00
|
|
|
if (run_mode == GIMP_RUN_INTERACTIVE)
|
2008-10-20 14:04:39 +08:00
|
|
|
gimp_set_data (PLUG_IN_PROC, &dvals, sizeof (WarpVals));
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
|
2000-08-22 09:26:57 +08:00
|
|
|
if (run_mode != GIMP_RUN_NONINTERACTIVE)
|
2000-01-18 01:02:26 +08:00
|
|
|
gimp_displays_flush ();
|
2006-06-21 18:22:14 +08:00
|
|
|
|
|
|
|
values[0].data.d_status = status;
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
static GtkWidget *
|
|
|
|
spin_button_new (GtkAdjustment **adjustment, /* return value */
|
|
|
|
gdouble value,
|
|
|
|
gdouble lower,
|
|
|
|
gdouble upper,
|
|
|
|
gdouble step_increment,
|
|
|
|
gdouble page_increment,
|
|
|
|
gdouble page_size,
|
|
|
|
gdouble climb_rate,
|
|
|
|
guint digits)
|
|
|
|
{
|
|
|
|
GtkWidget *spinbutton;
|
|
|
|
|
|
|
|
*adjustment = gtk_adjustment_new (value, lower, upper,
|
|
|
|
step_increment, page_increment, 0);
|
|
|
|
|
2019-03-09 20:25:19 +08:00
|
|
|
spinbutton = gimp_spin_button_new (*adjustment,
|
|
|
|
climb_rate, digits);
|
2018-05-11 02:44:08 +08:00
|
|
|
|
|
|
|
gtk_spin_button_set_numeric (GTK_SPIN_BUTTON (spinbutton), TRUE);
|
|
|
|
|
|
|
|
return spinbutton;
|
|
|
|
}
|
|
|
|
|
2004-04-22 00:50:13 +08:00
|
|
|
static gboolean
|
2019-08-31 08:43:48 +08:00
|
|
|
warp_dialog (GimpDrawable *drawable)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2010-10-25 07:26:00 +08:00
|
|
|
GtkWidget *dlg;
|
|
|
|
GtkWidget *vbox;
|
|
|
|
GtkWidget *label;
|
|
|
|
GtkWidget *toggle;
|
|
|
|
GtkWidget *toggle_hbox;
|
|
|
|
GtkWidget *frame;
|
2018-05-05 06:31:21 +08:00
|
|
|
GtkWidget *grid;
|
2010-10-25 07:26:00 +08:00
|
|
|
GtkWidget *spinbutton;
|
|
|
|
GtkAdjustment *adj;
|
|
|
|
GtkWidget *combo;
|
|
|
|
GtkSizeGroup *label_group;
|
|
|
|
GtkSizeGroup *spin_group;
|
|
|
|
GSList *group = NULL;
|
|
|
|
gboolean run;
|
2000-01-09 04:00:10 +08:00
|
|
|
|
2005-08-16 06:42:34 +08:00
|
|
|
gimp_ui_init (PLUG_IN_BINARY, FALSE);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2011-04-09 02:31:34 +08:00
|
|
|
dlg = gimp_dialog_new (_("Warp"), PLUG_IN_ROLE,
|
2003-11-06 23:27:05 +08:00
|
|
|
NULL, 0,
|
2008-10-20 14:04:39 +08:00
|
|
|
gimp_standard_help_func, PLUG_IN_PROC,
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2017-02-12 23:18:24 +08:00
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
|
|
_("_OK"), GTK_RESPONSE_OK,
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2003-11-06 23:27:05 +08:00
|
|
|
NULL);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-10 23:04:37 +08:00
|
|
|
gimp_dialog_set_alternative_button_order (GTK_DIALOG (dlg),
|
2005-08-16 06:42:34 +08:00
|
|
|
GTK_RESPONSE_OK,
|
|
|
|
GTK_RESPONSE_CANCEL,
|
|
|
|
-1);
|
2005-02-09 04:40:33 +08:00
|
|
|
|
2005-09-10 02:07:31 +08:00
|
|
|
gimp_window_set_transient (GTK_WINDOW (dlg));
|
2005-09-06 05:40:29 +08:00
|
|
|
|
2011-09-30 18:17:53 +08:00
|
|
|
vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
|
2004-05-20 09:19:47 +08:00
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (vbox), 12);
|
2009-07-16 00:57:12 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dlg))),
|
|
|
|
vbox, TRUE, TRUE, 0);
|
2000-01-09 04:00:10 +08:00
|
|
|
gtk_widget_show (vbox);
|
|
|
|
|
2004-05-20 09:19:47 +08:00
|
|
|
frame = gimp_frame_new (_("Basic Options"));
|
2000-01-09 04:00:10 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
|
2004-05-20 09:19:47 +08:00
|
|
|
gtk_widget_show (frame);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
grid = gtk_grid_new ();
|
|
|
|
gtk_grid_set_row_spacing (GTK_GRID (grid), 6);
|
|
|
|
gtk_grid_set_column_spacing (GTK_GRID (grid), 6);
|
|
|
|
gtk_container_add (GTK_CONTAINER (frame), grid);
|
|
|
|
gtk_widget_show (grid);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2004-05-28 08:12:01 +08:00
|
|
|
spin_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
|
|
|
|
label_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
|
|
|
|
|
2000-01-26 01:46:56 +08:00
|
|
|
/* amount, iter */
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.amount,
|
|
|
|
-1000, 1000, /* ??? */
|
|
|
|
1, 10, 0, 1, 2);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
g_object_unref (spin_group);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 0,
|
|
|
|
_("Step size:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
g_object_unref (label_group);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_double_adjustment_update),
|
|
|
|
&dvals.amount);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.iter,
|
|
|
|
1, 100, 1, 5, 0, 1, 0);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 1,
|
|
|
|
_("Iterations:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_int_adjustment_update),
|
|
|
|
&dvals.iter);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Displacement map menu */
|
2004-06-21 06:47:33 +08:00
|
|
|
label = gtk_label_new (_("Displacement map:"));
|
2016-09-09 01:11:20 +08:00
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
|
|
|
gtk_label_set_yalign (GTK_LABEL (label), 1.0);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_widget_set_margin_start (label, 12);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), label, 2, 0, 1, 1);
|
|
|
|
// GTK_FILL, GTK_FILL, 0, 0);
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_widget_show (label);
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
combo = gimp_drawable_combo_box_new (warp_map_constrain,
|
2019-08-31 08:43:48 +08:00
|
|
|
drawable,
|
2019-08-08 05:16:25 +08:00
|
|
|
NULL);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_widget_set_margin_start (combo, 12);
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_int_combo_box_connect (GIMP_INT_COMBO_BOX (combo),
|
|
|
|
dvals.warp_map_id,
|
2004-04-22 00:50:13 +08:00
|
|
|
G_CALLBACK (gimp_int_combo_box_get_active),
|
2019-08-31 08:43:48 +08:00
|
|
|
&dvals.warp_map_id, NULL);
|
2004-04-22 00:50:13 +08:00
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_grid_attach (GTK_GRID (grid), combo, 2, 1, 1, 1);
|
|
|
|
// GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
|
2004-04-22 00:50:13 +08:00
|
|
|
gtk_widget_show (combo);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
/* ======================================================================= */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Displacement Type */
|
2004-06-21 06:47:33 +08:00
|
|
|
label = gtk_label_new (_("On edges:"));
|
2016-09-09 01:11:20 +08:00
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_grid_attach (GTK_GRID (grid), label, 0, 2, 1, 1);
|
|
|
|
// GTK_FILL, GTK_FILL, 0, 0);
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_widget_show (label);
|
|
|
|
|
2011-09-30 18:17:53 +08:00
|
|
|
toggle_hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_grid_attach (GTK_GRID (grid), toggle_hbox, 1, 2, 2, 1);
|
|
|
|
// GTK_FILL, GTK_FILL, 0, 0);
|
2004-05-20 09:19:47 +08:00
|
|
|
gtk_widget_show (toggle_hbox);
|
2000-01-26 01:46:56 +08:00
|
|
|
|
1999-09-26 13:16:19 +08:00
|
|
|
toggle = gtk_radio_button_new_with_label (group, _("Wrap"));
|
2002-01-16 02:35:29 +08:00
|
|
|
group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (toggle));
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (toggle_hbox), toggle, FALSE, FALSE, 0);
|
2002-01-16 02:35:29 +08:00
|
|
|
gtk_widget_show (toggle);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (toggle), "gimp-item-data",
|
|
|
|
GINT_TO_POINTER (WRAP));
|
|
|
|
|
2003-01-07 14:16:02 +08:00
|
|
|
g_signal_connect (toggle, "toggled",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_radio_button_update),
|
|
|
|
&dvals.wrap_type);
|
|
|
|
|
2000-08-28 08:42:32 +08:00
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle),
|
2008-10-20 14:04:39 +08:00
|
|
|
dvals.wrap_type == WRAP);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
1999-09-26 13:16:19 +08:00
|
|
|
toggle = gtk_radio_button_new_with_label (group, _("Smear"));
|
2002-01-16 02:35:29 +08:00
|
|
|
group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (toggle));
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (toggle_hbox), toggle, FALSE, FALSE, 0);
|
2002-01-16 02:35:29 +08:00
|
|
|
gtk_widget_show (toggle);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (toggle), "gimp-item-data",
|
|
|
|
GINT_TO_POINTER (SMEAR));
|
|
|
|
|
2003-01-07 14:16:02 +08:00
|
|
|
g_signal_connect (toggle, "toggled",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_radio_button_update),
|
|
|
|
&dvals.wrap_type);
|
|
|
|
|
2000-08-28 08:42:32 +08:00
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle),
|
2008-10-20 14:04:39 +08:00
|
|
|
dvals.wrap_type == SMEAR);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
1999-09-26 13:16:19 +08:00
|
|
|
toggle = gtk_radio_button_new_with_label (group, _("Black"));
|
2002-01-16 02:35:29 +08:00
|
|
|
group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (toggle));
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (toggle_hbox), toggle, FALSE, FALSE, 0);
|
2002-01-16 02:35:29 +08:00
|
|
|
gtk_widget_show (toggle);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (toggle), "gimp-item-data",
|
|
|
|
GINT_TO_POINTER (BLACK));
|
|
|
|
|
2003-01-07 14:16:02 +08:00
|
|
|
g_signal_connect (toggle, "toggled",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_radio_button_update),
|
|
|
|
&dvals.wrap_type);
|
|
|
|
|
2000-08-28 08:42:32 +08:00
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle),
|
2008-10-20 14:04:39 +08:00
|
|
|
dvals.wrap_type == BLACK);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-03-24 19:17:55 +08:00
|
|
|
toggle = gtk_radio_button_new_with_label (group, _("Foreground color"));
|
2002-01-16 02:35:29 +08:00
|
|
|
group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (toggle));
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (toggle_hbox), toggle, FALSE, FALSE, 0);
|
2002-01-16 02:35:29 +08:00
|
|
|
gtk_widget_show (toggle);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (toggle), "gimp-item-data",
|
|
|
|
GINT_TO_POINTER (COLOR));
|
|
|
|
|
2003-01-07 14:16:02 +08:00
|
|
|
g_signal_connect (toggle, "toggled",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_radio_button_update),
|
|
|
|
&dvals.wrap_type);
|
|
|
|
|
2000-08-28 08:42:32 +08:00
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle),
|
2008-10-20 14:04:39 +08:00
|
|
|
dvals.wrap_type == COLOR);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
2018-05-05 06:31:21 +08:00
|
|
|
/* --------- The secondary grid -------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2004-05-20 09:19:47 +08:00
|
|
|
frame = gimp_frame_new (_("Advanced Options"));
|
2000-01-09 04:00:10 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
|
2004-05-20 09:19:47 +08:00
|
|
|
gtk_widget_show (frame);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
grid = gtk_grid_new ();
|
|
|
|
gtk_grid_set_row_spacing (GTK_GRID (grid), 6);
|
|
|
|
gtk_grid_set_column_spacing (GTK_GRID (grid), 6);
|
|
|
|
gtk_container_add (GTK_CONTAINER (frame), grid);
|
|
|
|
gtk_widget_show (grid);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.dither,
|
|
|
|
0, 100, 1, 10, 0, 1, 2);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 0,
|
|
|
|
_("Dither size:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_double_adjustment_update),
|
|
|
|
&dvals.dither);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.angle,
|
|
|
|
0, 360, 1, 15, 0, 1, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 1,
|
|
|
|
_("Rotation angle:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_double_adjustment_update),
|
|
|
|
&dvals.angle);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.substeps,
|
|
|
|
1, 100, 1, 5, 0, 1, 0);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 2,
|
|
|
|
_("Substeps:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_int_adjustment_update),
|
|
|
|
&dvals.substeps);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Magnitude map menu */
|
2004-06-21 06:47:33 +08:00
|
|
|
label = gtk_label_new (_("Magnitude map:"));
|
2016-09-09 01:11:20 +08:00
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
|
|
|
gtk_label_set_yalign (GTK_LABEL (label), 1.0);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_widget_set_margin_start (label, 12);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), label, 2, 0, 1, 1);
|
|
|
|
// GTK_FILL, GTK_FILL, 0, 0);
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_widget_show (label);
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
combo = gimp_drawable_combo_box_new (warp_map_constrain,
|
2019-08-31 08:43:48 +08:00
|
|
|
drawable,
|
2019-08-08 05:16:25 +08:00
|
|
|
NULL);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_widget_set_margin_start (combo, 12);
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_int_combo_box_connect (GIMP_INT_COMBO_BOX (combo),
|
|
|
|
dvals.mag_map_id,
|
2004-04-22 00:50:13 +08:00
|
|
|
G_CALLBACK (gimp_int_combo_box_get_active),
|
2019-08-31 08:43:48 +08:00
|
|
|
&dvals.mag_map_id, NULL);
|
2004-04-22 00:50:13 +08:00
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_grid_attach (GTK_GRID (grid), combo, 2, 1, 1, 1);
|
|
|
|
// GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
|
2004-04-22 00:50:13 +08:00
|
|
|
gtk_widget_show (combo);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Magnitude Usage */
|
2011-09-30 18:17:53 +08:00
|
|
|
toggle_hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
|
2000-08-28 08:42:32 +08:00
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (toggle_hbox), 1);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_grid_attach (GTK_GRID (grid), toggle_hbox, 2, 2, 1, 1);
|
|
|
|
// GTK_FILL, GTK_FILL, 0, 0);
|
2004-05-20 09:19:47 +08:00
|
|
|
gtk_widget_show (toggle_hbox);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2004-06-21 06:47:33 +08:00
|
|
|
toggle = gtk_check_button_new_with_label (_("Use magnitude map"));
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_widget_set_margin_start (toggle, 12);
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (toggle_hbox), toggle, FALSE, FALSE, 0);
|
2000-08-28 08:42:32 +08:00
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle), dvals.mag_use);
|
1999-09-04 07:14:44 +08:00
|
|
|
gtk_widget_show (toggle);
|
|
|
|
|
2003-01-07 14:16:02 +08:00
|
|
|
g_signal_connect (toggle, "toggled",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_toggle_button_update),
|
|
|
|
&dvals.mag_use);
|
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
2018-05-05 06:31:21 +08:00
|
|
|
/* --------- The "other" grid -------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2004-05-20 09:19:47 +08:00
|
|
|
frame = gimp_frame_new (_("More Advanced Options"));
|
2000-01-09 04:00:10 +08:00
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
|
2004-05-20 09:19:47 +08:00
|
|
|
gtk_widget_show (frame);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
grid = gtk_grid_new ();
|
|
|
|
gtk_grid_set_row_spacing (GTK_GRID (grid), 6);
|
|
|
|
gtk_grid_set_column_spacing (GTK_GRID (grid), 6);
|
|
|
|
gtk_container_add (GTK_CONTAINER (frame), grid);
|
|
|
|
gtk_widget_show (grid);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.grad_scale,
|
|
|
|
-1000, 1000, /* ??? */
|
|
|
|
0.01, 0.1, 0, 1, 3);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 0,
|
|
|
|
_("Gradient scale:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_double_adjustment_update),
|
|
|
|
&dvals.grad_scale);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* --------- Gradient map menu ---------------- */
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
combo = gimp_drawable_combo_box_new (warp_map_constrain,
|
2019-08-31 08:43:48 +08:00
|
|
|
drawable,
|
2019-08-08 05:16:25 +08:00
|
|
|
NULL);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_widget_set_margin_start (combo, 12);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), combo, 2, 0, 1, 1);
|
|
|
|
// GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
|
2007-11-09 19:17:00 +08:00
|
|
|
gtk_widget_show (combo);
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_int_combo_box_connect (GIMP_INT_COMBO_BOX (combo),
|
|
|
|
dvals.grad_map_id,
|
2004-04-22 00:50:13 +08:00
|
|
|
G_CALLBACK (gimp_int_combo_box_get_active),
|
2019-08-31 08:43:48 +08:00
|
|
|
&dvals.grad_map_id, NULL);
|
2004-04-22 00:50:13 +08:00
|
|
|
|
2007-11-09 19:17:00 +08:00
|
|
|
gimp_help_set_help_data (combo, _("Gradient map selection menu"), NULL);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* ---------------------------------------------- */
|
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.vector_scale,
|
|
|
|
-1000, 1000, /* ??? */
|
|
|
|
0.01, 0.1, 0, 1, 3);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 1,
|
|
|
|
_("Vector mag:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_double_adjustment_update),
|
|
|
|
&dvals.vector_scale);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* -------------------------------------------------------- */
|
2000-01-09 04:00:10 +08:00
|
|
|
|
2018-05-11 02:44:08 +08:00
|
|
|
spinbutton = spin_button_new (&adj, dvals.vector_angle,
|
|
|
|
0, 360, 1, 15, 0, 1, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (spin_group, spinbutton);
|
|
|
|
|
2018-05-05 06:31:21 +08:00
|
|
|
label = gimp_grid_attach_aligned (GTK_GRID (grid), 0, 2,
|
|
|
|
_("Angle:"), 0.0, 0.5,
|
|
|
|
spinbutton, 1);
|
2004-05-28 08:12:01 +08:00
|
|
|
gtk_size_group_add_widget (label_group, label);
|
|
|
|
|
2005-07-01 00:03:24 +08:00
|
|
|
g_signal_connect (adj, "value-changed",
|
2002-01-16 02:35:29 +08:00
|
|
|
G_CALLBACK (gimp_double_adjustment_update),
|
|
|
|
&dvals.vector_angle);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* --------- Vector map menu ---------------- */
|
2019-07-19 19:10:42 +08:00
|
|
|
combo = gimp_drawable_combo_box_new (warp_map_constrain,
|
2019-08-31 08:43:48 +08:00
|
|
|
drawable,
|
2019-08-08 05:16:25 +08:00
|
|
|
NULL);
|
2018-05-05 06:31:21 +08:00
|
|
|
gtk_widget_set_margin_start (combo, 12);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), combo, 2, 1, 1, 1);
|
|
|
|
// GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
|
2007-11-09 19:17:00 +08:00
|
|
|
gtk_widget_show (combo);
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_int_combo_box_connect (GIMP_INT_COMBO_BOX (combo),
|
|
|
|
dvals.vector_map_id,
|
2004-04-22 00:50:13 +08:00
|
|
|
G_CALLBACK (gimp_int_combo_box_get_active),
|
2019-08-31 08:43:48 +08:00
|
|
|
&dvals.vector_map_id, NULL);
|
2004-04-22 00:50:13 +08:00
|
|
|
|
2007-11-09 19:17:00 +08:00
|
|
|
gimp_help_set_help_data (combo,
|
2008-10-20 14:04:39 +08:00
|
|
|
_("Fixed-direction-vector map selection menu"),
|
|
|
|
NULL);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
gtk_widget_show (dlg);
|
|
|
|
|
2003-11-12 02:11:56 +08:00
|
|
|
run = (gimp_dialog_run (GIMP_DIALOG (dlg)) == GTK_RESPONSE_OK);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2003-11-06 23:27:05 +08:00
|
|
|
gtk_widget_destroy (dlg);
|
|
|
|
|
|
|
|
return run;
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
2003-11-06 23:27:05 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
/* ---------------------------------------------------------------------- */
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
static const Babl *
|
2019-08-31 08:43:48 +08:00
|
|
|
get_u8_format (GimpDrawable *drawable)
|
2019-07-19 19:10:42 +08:00
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
if (gimp_drawable_is_rgb (drawable))
|
2019-07-19 19:10:42 +08:00
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
if (gimp_drawable_has_alpha (drawable))
|
2019-07-19 19:10:42 +08:00
|
|
|
return babl_format ("R'G'B'A u8");
|
|
|
|
else
|
|
|
|
return babl_format ("R'G'B' u8");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
if (gimp_drawable_has_alpha (drawable))
|
2019-07-19 19:10:42 +08:00
|
|
|
return babl_format ("Y'A u8");
|
|
|
|
else
|
|
|
|
return babl_format ("Y' u8");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
static void
|
2019-08-31 08:43:48 +08:00
|
|
|
blur16 (GimpDrawable *drawable)
|
2000-01-26 01:46:56 +08:00
|
|
|
{
|
|
|
|
/* blur a 2-or-more byte-per-pixel drawable,
|
|
|
|
* 1st 2 bytes interpreted as a 16-bit height field.
|
|
|
|
*/
|
2019-07-19 19:10:42 +08:00
|
|
|
GeglBuffer *src_buffer;
|
|
|
|
GeglBuffer *dest_buffer;
|
|
|
|
const Babl *format;
|
1999-09-04 07:14:44 +08:00
|
|
|
gint width, height;
|
|
|
|
gint src_bytes;
|
|
|
|
gint dest_bytes;
|
|
|
|
gint dest_bytes_inc;
|
|
|
|
gint offb, off1;
|
|
|
|
|
|
|
|
guchar *dest, *d; /* pointers to rows of X and Y diff. data */
|
|
|
|
guchar *prev_row, *pr;
|
|
|
|
guchar *cur_row, *cr;
|
|
|
|
guchar *next_row, *nr;
|
|
|
|
guchar *tmp;
|
|
|
|
gint row, col; /* relating to indexing into pixel row arrays */
|
|
|
|
gint x1, y1, x2, y2;
|
|
|
|
gdouble pval; /* average pixel value of pixel & neighbors */
|
|
|
|
|
|
|
|
/* --------------------------------------- */
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
if (! gimp_drawable_mask_intersect (drawable,
|
2008-02-22 17:39:50 +08:00
|
|
|
&x1, &y1, &width, &height))
|
2005-03-22 07:59:12 +08:00
|
|
|
return;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-02-22 17:39:50 +08:00
|
|
|
x2 = x1 + width;
|
|
|
|
y2 = y1 + height;
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
width = gimp_drawable_width (drawable); /* size of input drawable*/
|
|
|
|
height = gimp_drawable_height (drawable);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
format = get_u8_format (drawable);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
|
|
|
/* bytes per pixel in SOURCE drawable, must be 2 or more */
|
|
|
|
src_bytes = babl_format_get_bytes_per_pixel (format);
|
|
|
|
|
|
|
|
dest_bytes = src_bytes; /* bytes per pixel in SOURCE drawable, >= 2 */
|
1999-09-04 07:14:44 +08:00
|
|
|
dest_bytes_inc = dest_bytes - 2; /* this is most likely zero, but I guess it's more conservative... */
|
|
|
|
|
|
|
|
/* allocate row buffers for source & dest. data */
|
|
|
|
|
2005-03-04 23:12:29 +08:00
|
|
|
prev_row = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
2019-07-19 19:10:42 +08:00
|
|
|
cur_row = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
2005-03-04 23:12:29 +08:00
|
|
|
next_row = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
2019-07-19 19:10:42 +08:00
|
|
|
dest = g_new (guchar, (x2 - x1) * src_bytes);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* initialize the pixel regions (read from source, write into dest) */
|
2019-08-31 08:43:48 +08:00
|
|
|
src_buffer = gimp_drawable_get_buffer (drawable);
|
|
|
|
dest_buffer = gimp_drawable_get_shadow_buffer (drawable);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
pr = prev_row + src_bytes; /* row arrays are prepared for indexing to -1 (!) */
|
2019-07-19 19:10:42 +08:00
|
|
|
cr = cur_row + src_bytes;
|
1999-09-04 07:14:44 +08:00
|
|
|
nr = next_row + src_bytes;
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (src_buffer, format, pr, x1, y1, (x2 - x1));
|
|
|
|
diff_prepare_row (src_buffer, format, cr, x1, y1+1, (x2 - x1));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* loop through the rows, applying the smoothing function */
|
|
|
|
for (row = y1; row < y2; row++)
|
|
|
|
{
|
|
|
|
/* prepare the next row */
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (src_buffer, format, nr, x1, row + 1, (x2 - x1));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
d = dest;
|
|
|
|
for (col = 0; col < (x2 - x1); col++) /* over columns of pixels */
|
2008-10-20 14:04:39 +08:00
|
|
|
{
|
|
|
|
offb = col*src_bytes; /* base of byte pointer offset */
|
|
|
|
off1 = offb+1; /* offset into row arrays */
|
|
|
|
|
|
|
|
pval = (256.0 * pr[offb - src_bytes] + pr[off1 - src_bytes] +
|
|
|
|
256.0 * pr[offb] + pr[off1] +
|
|
|
|
256.0 * pr[offb + src_bytes] + pr[off1 + src_bytes] +
|
|
|
|
256.0 * cr[offb - src_bytes] + cr[off1 - src_bytes] +
|
|
|
|
256.0 * cr[offb] + cr[off1] +
|
|
|
|
256.0 * cr[offb + src_bytes] + cr[off1 + src_bytes] +
|
|
|
|
256.0 * nr[offb - src_bytes] + nr[off1 - src_bytes] +
|
|
|
|
256.0 * nr[offb] + nr[off1] +
|
|
|
|
256.0 * nr[offb + src_bytes]) + nr[off1 + src_bytes];
|
|
|
|
|
|
|
|
pval /= 9.0; /* take the average */
|
|
|
|
*d++ = (guchar) (((gint) pval) >> 8); /* high-order byte */
|
|
|
|
*d++ = (guchar) (((gint) pval) % 256); /* low-order byte */
|
|
|
|
d += dest_bytes_inc; /* move data pointer on to next destination pixel */
|
2019-07-19 19:10:42 +08:00
|
|
|
}
|
2008-10-20 14:04:39 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
/* store the dest */
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_set (dest_buffer, GEGL_RECTANGLE (x1, row, (x2 - x1), 1), 0,
|
|
|
|
format, dest,
|
|
|
|
GEGL_AUTO_ROWSTRIDE);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* shuffle the row pointers */
|
|
|
|
tmp = pr;
|
|
|
|
pr = cr;
|
|
|
|
cr = nr;
|
2003-11-06 23:27:05 +08:00
|
|
|
nr = tmp;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if ((row % 8) == 0)
|
2008-10-20 14:04:39 +08:00
|
|
|
gimp_progress_update ((double) row / (double) (y2 - y1));
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
g_object_unref (src_buffer);
|
|
|
|
g_object_unref (dest_buffer);
|
|
|
|
|
2011-04-11 01:05:08 +08:00
|
|
|
gimp_progress_update (1.0);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_drawable_merge_shadow (drawable, TRUE);
|
|
|
|
gimp_drawable_update (drawable, x1, y1, (x2 - x1), (y2 - y1));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2005-03-04 23:12:29 +08:00
|
|
|
g_free (prev_row); /* row buffers allocated at top of fn. */
|
|
|
|
g_free (cur_row);
|
|
|
|
g_free (next_row);
|
|
|
|
g_free (dest);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
|
|
|
|
/* ====================================================================== */
|
|
|
|
/* Get one row of pixels from the PixelRegion and put them in 'data' */
|
|
|
|
|
|
|
|
static void
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (GeglBuffer *buffer,
|
|
|
|
const Babl *format,
|
|
|
|
guchar *data,
|
|
|
|
gint x,
|
|
|
|
gint y,
|
|
|
|
gint w)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-07-19 19:10:42 +08:00
|
|
|
gint bpp = babl_format_get_bytes_per_pixel (format);
|
2006-04-27 19:33:51 +08:00
|
|
|
gint b;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
/* y = CLAMP (y, 0, pixel_rgn->h - 1); FIXME? */
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_get (buffer, GEGL_RECTANGLE (x, y, w, 1), 1.0,
|
|
|
|
format, data,
|
|
|
|
GEGL_AUTO_ROWSTRIDE, GEGL_ABYSS_NONE);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Fill in edge pixels */
|
2019-07-19 19:10:42 +08:00
|
|
|
for (b = 0; b < bpp; b++)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-07-19 19:10:42 +08:00
|
|
|
data[b - (gint) bpp] = data[b];
|
|
|
|
data[w * bpp + b] = data[(w - 1) * bpp + b];
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
/* 'diff' combines the input drawables to prepare the two */
|
|
|
|
/* 16-bit (X,Y) vector displacement maps */
|
|
|
|
/* -------------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
static void
|
2019-08-31 08:43:48 +08:00
|
|
|
diff (GimpDrawable *drawable,
|
|
|
|
GimpDrawable **xl,
|
|
|
|
GimpDrawable **yl)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
GimpDrawable *draw_xd;
|
|
|
|
GimpDrawable *draw_yd; /* vector disp. drawables */
|
|
|
|
GimpDrawable *mdraw;
|
|
|
|
GimpDrawable *vdraw;
|
|
|
|
GimpDrawable *gdraw;
|
|
|
|
GimpImage *image; /* image holding X and Y diff. arrays */
|
|
|
|
GimpImage *new_image; /* image holding X and Y diff. layers */
|
|
|
|
GimpLayer *layer_active; /* currently active layer */
|
|
|
|
GimpLayer *xlayer;
|
|
|
|
GimpLayer *ylayer; /* individual X and Y layer ID numbers */
|
|
|
|
GeglBuffer *src_buffer;
|
|
|
|
GeglBuffer *destx_buffer;
|
|
|
|
const Babl *destx_format;
|
|
|
|
GeglBuffer *desty_buffer;
|
|
|
|
const Babl *desty_format;
|
|
|
|
GeglBuffer *vec_buffer;
|
|
|
|
GeglBuffer *mag_buffer = NULL;
|
|
|
|
GeglBuffer *grad_buffer;
|
|
|
|
gint width, height;
|
|
|
|
const Babl *src_format;
|
|
|
|
gint src_bytes;
|
|
|
|
const Babl *mformat = NULL;
|
|
|
|
gint mbytes = 0;
|
|
|
|
const Babl *vformat = NULL;
|
|
|
|
gint vbytes = 0;
|
|
|
|
const Babl *gformat = NULL;
|
|
|
|
gint gbytes = 0; /* bytes-per-pixel of various source drawables */
|
|
|
|
const Babl *dest_format;
|
|
|
|
gint dest_bytes;
|
|
|
|
gint dest_bytes_inc;
|
|
|
|
gint do_gradmap = FALSE; /* whether to add in gradient of gradmap to final diff. map */
|
|
|
|
gint do_vecmap = FALSE; /* whether to add in a fixed vector scaled by the vector map */
|
|
|
|
gint do_magmap = FALSE; /* whether to multiply result by the magnitude map */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
guchar *destx, *dx, *desty, *dy; /* pointers to rows of X and Y diff. data */
|
|
|
|
guchar *tmp;
|
|
|
|
guchar *prev_row, *pr;
|
|
|
|
guchar *cur_row, *cr;
|
|
|
|
guchar *next_row, *nr;
|
2006-06-21 18:22:14 +08:00
|
|
|
guchar *prev_row_g, *prg = NULL; /* pointers to gradient map data */
|
|
|
|
guchar *cur_row_g, *crg = NULL;
|
|
|
|
guchar *next_row_g, *nrg = NULL;
|
|
|
|
guchar *cur_row_v, *crv = NULL; /* pointers to vector map data */
|
|
|
|
guchar *cur_row_m, *crm = NULL; /* pointers to magnitude map data */
|
|
|
|
gint row, col, offb, off, bytes; /* relating to indexing into pixel row arrays */
|
1999-09-04 07:14:44 +08:00
|
|
|
gint x1, y1, x2, y2;
|
2006-06-21 18:22:14 +08:00
|
|
|
gint dvalx, dvaly; /* differential value at particular pixel */
|
|
|
|
gdouble tx, ty; /* temporary x,y differential value increments from gradmap, etc. */
|
|
|
|
gdouble rdx, rdy; /* x,y differential values: real #s */
|
|
|
|
gdouble rscalefac; /* scaling factor for x,y differential of 'curl' map */
|
|
|
|
gdouble gscalefac; /* scaling factor for x,y differential of 'gradient' map */
|
|
|
|
gdouble r, theta, dtheta; /* rectangular<-> spherical coordinate transform for vector rotation */
|
|
|
|
gdouble scale_vec_x, scale_vec_y; /* fixed vector X,Y component scaling factors */
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
/* ----------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
if (dvals.grad_scale != 0.0)
|
2006-06-21 18:22:14 +08:00
|
|
|
do_gradmap = TRUE; /* add in gradient of gradmap if scale != 0.000 */
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2006-06-21 18:22:14 +08:00
|
|
|
if (dvals.vector_scale != 0.0) /* add in gradient of vectormap if scale != 0.000 */
|
2006-04-27 19:33:51 +08:00
|
|
|
do_vecmap = TRUE;
|
|
|
|
|
2006-06-21 18:22:14 +08:00
|
|
|
do_magmap = (dvals.mag_use == TRUE); /* multiply by magnitude map if so requested */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Get the input area. This is the bounding box of the selection in
|
|
|
|
* the image (or the entire image if there is no selection). Only
|
|
|
|
* operating on the input area is simply an optimization. It doesn't
|
|
|
|
* need to be done for correct operation. (It simply makes it go
|
|
|
|
* faster, since fewer pixels need to be operated on).
|
|
|
|
*/
|
2019-08-31 08:43:48 +08:00
|
|
|
if (! gimp_drawable_mask_intersect (drawable,
|
2008-02-22 17:39:50 +08:00
|
|
|
&x1, &y1, &width, &height))
|
2005-03-22 07:59:12 +08:00
|
|
|
return;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-02-22 17:39:50 +08:00
|
|
|
x2 = x1 + width;
|
|
|
|
y2 = y1 + height;
|
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
/* Get the size of the input image. (This will/must be the same
|
|
|
|
* as the size of the output image.
|
|
|
|
*/
|
2019-08-31 08:43:48 +08:00
|
|
|
width = gimp_drawable_width (drawable);
|
|
|
|
height = gimp_drawable_height (drawable);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
src_format = get_u8_format (drawable);
|
2019-07-19 19:10:42 +08:00
|
|
|
src_bytes = babl_format_get_bytes_per_pixel (src_format);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* -- Add two layers: X and Y Displacement vectors -- */
|
2003-11-06 23:27:05 +08:00
|
|
|
/* -- I'm using a RGB drawable and using the first two bytes for a
|
|
|
|
16-bit pixel value. This is either clever, or a kluge,
|
1999-09-04 07:14:44 +08:00
|
|
|
depending on your point of view. */
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
image = gimp_item_get_image (GIMP_ITEM (drawable));
|
|
|
|
layer_active = gimp_image_get_active_layer (image);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-06-21 18:22:14 +08:00
|
|
|
/* create new image for X,Y diff */
|
2019-08-31 08:43:48 +08:00
|
|
|
new_image = gimp_image_new (width, height, GIMP_RGB);
|
|
|
|
|
|
|
|
xlayer = gimp_layer_new (new_image, "Warp_X_Vectors",
|
|
|
|
width, height,
|
|
|
|
GIMP_RGB_IMAGE,
|
|
|
|
100.0,
|
|
|
|
gimp_image_get_default_new_layer_mode (new_image));
|
|
|
|
|
|
|
|
ylayer = gimp_layer_new (new_image, "Warp_Y_Vectors",
|
|
|
|
width, height,
|
|
|
|
GIMP_RGB_IMAGE,
|
|
|
|
100.0,
|
|
|
|
gimp_image_get_default_new_layer_mode (new_image));
|
|
|
|
|
|
|
|
draw_yd = GIMP_DRAWABLE (ylayer);
|
|
|
|
draw_xd = GIMP_DRAWABLE (xlayer);
|
|
|
|
|
|
|
|
gimp_image_insert_layer (new_image, xlayer, NULL, 1);
|
|
|
|
gimp_image_insert_layer (new_image, ylayer, NULL, 1);
|
|
|
|
gimp_drawable_fill (GIMP_DRAWABLE (xlayer), GIMP_FILL_BACKGROUND);
|
|
|
|
gimp_drawable_fill (GIMP_DRAWABLE (ylayer), GIMP_FILL_BACKGROUND);
|
|
|
|
gimp_image_set_active_layer (image, layer_active);
|
|
|
|
|
|
|
|
dest_format = get_u8_format (draw_xd);
|
2019-07-19 19:10:42 +08:00
|
|
|
dest_bytes = babl_format_get_bytes_per_pixel (dest_format);
|
1999-09-04 07:14:44 +08:00
|
|
|
/* for a GRAYA drawable, I would expect this to be two bytes; any more would be excess */
|
|
|
|
dest_bytes_inc = dest_bytes - 2;
|
|
|
|
|
|
|
|
/* allocate row buffers for source & dest. data */
|
|
|
|
|
1999-12-31 02:54:17 +08:00
|
|
|
prev_row = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
|
|
|
cur_row = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
|
|
|
next_row = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
1999-12-31 02:54:17 +08:00
|
|
|
prev_row_g = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
|
|
|
cur_row_g = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
|
|
|
next_row_g = g_new (guchar, (x2 - x1 + 2) * src_bytes);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
1999-12-31 02:54:17 +08:00
|
|
|
cur_row_v = g_new (guchar, (x2 - x1 + 2) * src_bytes); /* vector map */
|
|
|
|
cur_row_m = g_new (guchar, (x2 - x1 + 2) * src_bytes); /* magnitude map */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
1999-12-31 02:54:17 +08:00
|
|
|
destx = g_new (guchar, (x2 - x1) * dest_bytes);
|
|
|
|
desty = g_new (guchar, (x2 - x1) * dest_bytes);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* initialize the source and destination pixel regions */
|
2006-04-27 19:33:51 +08:00
|
|
|
|
|
|
|
/* 'curl' vector-rotation input */
|
2019-08-31 08:43:48 +08:00
|
|
|
src_buffer = gimp_drawable_get_buffer (drawable);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
|
|
|
/* destination: X diff output */
|
2019-08-31 08:43:48 +08:00
|
|
|
destx_buffer = gimp_drawable_get_buffer (draw_xd);
|
|
|
|
destx_format = get_u8_format (draw_xd);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
|
|
|
/* Y diff output */
|
2019-08-31 08:43:48 +08:00
|
|
|
desty_buffer = gimp_drawable_get_buffer (draw_yd);
|
|
|
|
desty_format = get_u8_format (draw_yd);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
pr = prev_row + src_bytes;
|
2019-07-19 19:10:42 +08:00
|
|
|
cr = cur_row + src_bytes;
|
1999-09-04 07:14:44 +08:00
|
|
|
nr = next_row + src_bytes;
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (src_buffer, src_format, pr, x1, y1, (x2 - x1));
|
|
|
|
diff_prepare_row (src_buffer, src_format, cr, x1, y1+1, (x2 - x1));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* fixed-vector (x,y) component scale factors */
|
2006-04-27 19:33:51 +08:00
|
|
|
scale_vec_x = (dvals.vector_scale *
|
|
|
|
cos ((90 - dvals.vector_angle) * G_PI / 180.0) * 256.0 / 10);
|
|
|
|
scale_vec_y = (dvals.vector_scale *
|
|
|
|
sin ((90 - dvals.vector_angle) * G_PI / 180.0) * 256.0 / 10);
|
|
|
|
|
|
|
|
if (do_vecmap)
|
|
|
|
{
|
2019-09-03 16:48:32 +08:00
|
|
|
vdraw = gimp_drawable_get_by_id (dvals.vector_map_id);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
|
|
|
/* bytes per pixel in SOURCE drawable */
|
2019-08-31 08:43:48 +08:00
|
|
|
vformat = get_u8_format (vdraw);
|
2019-07-19 19:10:42 +08:00
|
|
|
vbytes = babl_format_get_bytes_per_pixel (vformat);
|
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
/* fixed-vector scale-map */
|
2019-08-31 08:43:48 +08:00
|
|
|
vec_buffer = gimp_drawable_get_buffer (vdraw);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
crv = cur_row_v + vbytes;
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (vec_buffer, vformat, crv, x1, y1, (x2 - x1));
|
2006-04-27 19:33:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (do_gradmap)
|
|
|
|
{
|
2019-09-03 16:48:32 +08:00
|
|
|
gdraw = gimp_drawable_get_by_id (dvals.grad_map_id);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gformat = get_u8_format (gdraw);
|
2019-07-19 19:10:42 +08:00
|
|
|
gbytes = babl_format_get_bytes_per_pixel (gformat);
|
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
/* fixed-vector scale-map */
|
2019-08-31 08:43:48 +08:00
|
|
|
grad_buffer = gimp_drawable_get_buffer (gdraw);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
prg = prev_row_g + gbytes;
|
|
|
|
crg = cur_row_g + gbytes;
|
|
|
|
nrg = next_row_g + gbytes;
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (grad_buffer, gformat, prg, x1, y1 - 1, (x2 - x1));
|
|
|
|
diff_prepare_row (grad_buffer, gformat, crg, x1, y1, (x2 - x1));
|
2006-04-27 19:33:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (do_magmap)
|
|
|
|
{
|
2019-09-03 16:48:32 +08:00
|
|
|
mdraw = gimp_drawable_get_by_id (dvals.mag_map_id);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
mformat = get_u8_format (mdraw);
|
2019-07-19 19:10:42 +08:00
|
|
|
mbytes = babl_format_get_bytes_per_pixel (mformat);
|
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
/* fixed-vector scale-map */
|
2019-08-31 08:43:48 +08:00
|
|
|
mag_buffer = gimp_drawable_get_buffer (mdraw);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
crm = cur_row_m + mbytes;
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (mag_buffer, mformat, crm, x1, y1, (x2 - x1));
|
2006-04-27 19:33:51 +08:00
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-07-19 05:49:50 +08:00
|
|
|
dtheta = dvals.angle * G_PI / 180.0;
|
2006-04-27 19:33:51 +08:00
|
|
|
/* note that '3' is rather arbitrary here. */
|
|
|
|
rscalefac = 256.0 / (3 * src_bytes);
|
|
|
|
/* scale factor for gradient map components */
|
|
|
|
gscalefac = dvals.grad_scale * 256.0 / (3 * gbytes);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* loop through the rows, applying the differential convolution */
|
|
|
|
for (row = y1; row < y2; row++)
|
|
|
|
{
|
|
|
|
/* prepare the next row */
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (src_buffer, src_format, nr, x1, row + 1, (x2 - x1));
|
2003-11-06 23:27:05 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
if (do_magmap)
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (mag_buffer, mformat, crm, x1, row + 1, (x2 - x1));
|
1999-09-04 07:14:44 +08:00
|
|
|
if (do_vecmap)
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (vec_buffer, vformat, crv, x1, row + 1, (x2 - x1));
|
1999-09-04 07:14:44 +08:00
|
|
|
if (do_gradmap)
|
2019-07-19 19:10:42 +08:00
|
|
|
diff_prepare_row (grad_buffer, gformat, crg, x1, row + 1, (x2 - x1));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
dx = destx;
|
|
|
|
dy = desty;
|
|
|
|
|
|
|
|
for (col = 0; col < (x2 - x1); col++) /* over columns of pixels */
|
2006-04-27 19:33:51 +08:00
|
|
|
{
|
|
|
|
rdx = 0.0;
|
|
|
|
rdy = 0.0;
|
|
|
|
ty = 0.0;
|
|
|
|
tx = 0.0;
|
|
|
|
|
|
|
|
offb = col * src_bytes; /* base of byte pointer offset */
|
|
|
|
for (bytes=0; bytes < src_bytes; bytes++) /* add all channels together */
|
|
|
|
{
|
|
|
|
off = offb+bytes; /* offset into row arrays */
|
|
|
|
rdx += ((gint) -pr[off - src_bytes] + (gint) pr[off + src_bytes] +
|
|
|
|
(gint) -2*cr[off - src_bytes] + (gint) 2*cr[off + src_bytes] +
|
|
|
|
(gint) -nr[off - src_bytes] + (gint) nr[off + src_bytes]);
|
|
|
|
|
|
|
|
rdy += ((gint) -pr[off - src_bytes] - (gint)2*pr[off] - (gint) pr[off + src_bytes] +
|
|
|
|
(gint) nr[off - src_bytes] + (gint)2*nr[off] + (gint) nr[off + src_bytes]);
|
|
|
|
}
|
|
|
|
|
|
|
|
rdx *= rscalefac; /* take average, then reduce. Assume max. rdx now 65535 */
|
|
|
|
rdy *= rscalefac; /* take average, then reduce */
|
|
|
|
|
|
|
|
theta = atan2(rdy,rdx); /* convert to polar, then back to rectang. coords */
|
|
|
|
r = sqrt(rdy*rdy + rdx*rdx);
|
|
|
|
theta += dtheta; /* rotate gradient vector by this angle (radians) */
|
|
|
|
rdx = r * cos(theta);
|
|
|
|
rdy = r * sin(theta);
|
|
|
|
|
|
|
|
if (do_gradmap)
|
|
|
|
{
|
|
|
|
offb = col*gbytes; /* base of byte pointer offset into pixel values (R,G,B,Alpha, etc.) */
|
|
|
|
for (bytes=0; bytes < src_bytes; bytes++) /* add all channels together */
|
|
|
|
{
|
2008-10-20 14:04:39 +08:00
|
|
|
off = offb+bytes; /* offset into row arrays */
|
|
|
|
tx += ((gint) -prg[off - gbytes] + (gint) prg[off + gbytes] +
|
2006-04-27 19:33:51 +08:00
|
|
|
(gint) -2*crg[off - gbytes] + (gint) 2*crg[off + gbytes] +
|
|
|
|
(gint) -nrg[off - gbytes] + (gint) nrg[off + gbytes]);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
ty += ((gint) -prg[off - gbytes] - (gint)2*prg[off] - (gint) prg[off + gbytes] +
|
2006-04-27 19:33:51 +08:00
|
|
|
(gint) nrg[off - gbytes] + (gint)2*nrg[off] + (gint) nrg[off + gbytes]);
|
2008-10-20 14:04:39 +08:00
|
|
|
}
|
2006-04-27 19:33:51 +08:00
|
|
|
tx *= gscalefac;
|
|
|
|
ty *= gscalefac;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
rdx += tx; /* add gradient component in to the other one */
|
|
|
|
rdy += ty;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
} /* if (do_gradmap) */
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if (do_vecmap)
|
|
|
|
{ /* add in fixed vector scaled by vec. map data */
|
|
|
|
tx = (gdouble) crv[col*vbytes]; /* use first byte only */
|
|
|
|
rdx += scale_vec_x * tx;
|
|
|
|
rdy += scale_vec_y * tx;
|
|
|
|
} /* if (do_vecmap) */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if (do_magmap)
|
|
|
|
{ /* multiply result by mag. map data */
|
|
|
|
tx = (gdouble) crm[col*mbytes];
|
|
|
|
rdx = (rdx * tx)/(255.0);
|
|
|
|
rdy = (rdy * tx)/(255.0);
|
|
|
|
} /* if do_magmap */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
dvalx = rdx + (2<<14); /* take zero point to be 2^15, since this is two bytes */
|
|
|
|
dvaly = rdy + (2<<14);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if (dvalx < 0)
|
|
|
|
dvalx = 0;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if (dvalx > 65535)
|
|
|
|
dvalx = 65535;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
*dx++ = (guchar) (dvalx >> 8); /* store high order byte in value channel */
|
|
|
|
*dx++ = (guchar) (dvalx % 256); /* store low order byte in alpha channel */
|
|
|
|
dx += dest_bytes_inc; /* move data pointer on to next destination pixel */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if (dvaly < 0)
|
|
|
|
dvaly = 0;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if (dvaly > 65535)
|
|
|
|
dvaly = 65535;
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
*dy++ = (guchar) (dvaly >> 8);
|
|
|
|
*dy++ = (guchar) (dvaly % 256);
|
|
|
|
dy += dest_bytes_inc;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
} /* ------------------------------- for (col...) ---------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* store the dest */
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_set (destx_buffer,
|
|
|
|
GEGL_RECTANGLE (x1, row, (x2 - x1), 1), 0,
|
|
|
|
destx_format, destx,
|
|
|
|
GEGL_AUTO_ROWSTRIDE);
|
|
|
|
|
|
|
|
gegl_buffer_set (desty_buffer,
|
|
|
|
GEGL_RECTANGLE (x1, row, (x2 - x1), 1), 0,
|
|
|
|
desty_format, desty,
|
|
|
|
GEGL_AUTO_ROWSTRIDE);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* swap around the pointers to row buffers */
|
|
|
|
tmp = pr;
|
|
|
|
pr = cr;
|
|
|
|
cr = nr;
|
|
|
|
nr = tmp;
|
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if (do_gradmap)
|
|
|
|
{
|
|
|
|
tmp = prg;
|
|
|
|
prg = crg;
|
|
|
|
crg = nrg;
|
|
|
|
nrg = tmp;
|
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
if ((row % 8) == 0)
|
2008-10-20 14:04:39 +08:00
|
|
|
gimp_progress_update ((gdouble) row / (gdouble) (y2 - y1));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
} /* for (row..) */
|
|
|
|
|
2011-04-11 01:05:08 +08:00
|
|
|
gimp_progress_update (1.0);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
g_object_unref (src_buffer);
|
|
|
|
g_object_unref (destx_buffer);
|
|
|
|
g_object_unref (desty_buffer);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_drawable_update (draw_xd, x1, y1, (x2 - x1), (y2 - y1));
|
|
|
|
gimp_drawable_update (draw_yd, x1, y1, (x2 - x1), (y2 - y1));
|
2019-07-19 19:10:42 +08:00
|
|
|
|
|
|
|
gimp_displays_flush (); /* make sure layer is visible */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2005-09-30 16:16:10 +08:00
|
|
|
gimp_progress_init (_("Smoothing X gradient"));
|
2019-08-31 08:43:48 +08:00
|
|
|
blur16 (draw_xd);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2005-09-30 16:16:10 +08:00
|
|
|
gimp_progress_init (_("Smoothing Y gradient"));
|
2019-08-31 08:43:48 +08:00
|
|
|
blur16 (draw_yd);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
1999-12-31 02:54:17 +08:00
|
|
|
g_free (prev_row); /* row buffers allocated at top of fn. */
|
|
|
|
g_free (cur_row);
|
|
|
|
g_free (next_row);
|
|
|
|
g_free (prev_row_g); /* row buffers allocated at top of fn. */
|
|
|
|
g_free (cur_row_g);
|
|
|
|
g_free (next_row_g);
|
|
|
|
g_free (cur_row_v);
|
|
|
|
g_free (cur_row_m);
|
|
|
|
|
|
|
|
g_free (destx);
|
|
|
|
g_free (desty);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
*xl = GIMP_DRAWABLE (xlayer); /* pass back the X and Y layer ID numbers */
|
|
|
|
*yl = GIMP_DRAWABLE (ylayer);
|
2019-07-19 19:10:42 +08:00
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
/* The Warp displacement is done here. */
|
|
|
|
/* -------------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2003-11-06 23:27:05 +08:00
|
|
|
static void
|
2019-08-31 08:43:48 +08:00
|
|
|
warp (GimpDrawable *orig_draw)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
GimpDrawable *disp_map; /* Displacement map, ie, control array */
|
|
|
|
GimpDrawable *mag_draw; /* Magnitude multiplier factor map */
|
|
|
|
GimpDrawable *map_x = NULL;
|
|
|
|
GimpDrawable *map_y = NULL;
|
|
|
|
gboolean first_time = TRUE;
|
|
|
|
gint width;
|
|
|
|
gint height;
|
|
|
|
gint x1, y1, x2, y2;
|
|
|
|
GimpImage *image;
|
2006-06-21 18:22:14 +08:00
|
|
|
|
|
|
|
/* index var. over all "warp" Displacement iterations */
|
|
|
|
gint warp_iter;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-09-03 16:48:32 +08:00
|
|
|
disp_map = gimp_drawable_get_by_id (dvals.warp_map_id);
|
|
|
|
mag_draw = gimp_drawable_get_by_id (dvals.mag_map_id);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* calculate new X,Y Displacement image maps */
|
|
|
|
|
2005-09-30 16:16:10 +08:00
|
|
|
gimp_progress_init (_("Finding XY gradient"));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Get selection area */
|
2019-08-31 08:43:48 +08:00
|
|
|
if (! gimp_drawable_mask_intersect (orig_draw,
|
2008-02-22 17:39:50 +08:00
|
|
|
&x1, &y1, &width, &height))
|
2005-03-22 07:59:12 +08:00
|
|
|
return;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-02-22 17:39:50 +08:00
|
|
|
x2 = x1 + width;
|
|
|
|
y2 = y1 + height;
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
width = gimp_drawable_width (orig_draw);
|
|
|
|
height = gimp_drawable_height (orig_draw);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-06-21 18:22:14 +08:00
|
|
|
/* generate x,y differential images (arrays) */
|
2019-08-31 08:43:48 +08:00
|
|
|
diff (disp_map, &map_x, &map_y);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2003-12-04 21:21:27 +08:00
|
|
|
for (warp_iter = 0; warp_iter < dvals.iter; warp_iter++)
|
|
|
|
{
|
2005-09-30 16:16:10 +08:00
|
|
|
gimp_progress_init_printf (_("Flow step %d"), warp_iter+1);
|
2003-12-04 21:21:27 +08:00
|
|
|
progress = 0;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
warp_one (orig_draw, orig_draw,
|
|
|
|
map_x, map_y, mag_draw,
|
2019-07-19 19:10:42 +08:00
|
|
|
first_time, warp_iter);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_drawable_update (orig_draw,
|
2003-12-04 21:21:27 +08:00
|
|
|
x1, y1, (x2 - x1), (y2 - y1));
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2003-12-04 21:21:27 +08:00
|
|
|
if (run_mode != GIMP_RUN_NONINTERACTIVE)
|
|
|
|
gimp_displays_flush ();
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2003-12-04 21:21:27 +08:00
|
|
|
first_time = FALSE;
|
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
image = gimp_item_get_image (GIMP_ITEM (map_x));
|
2006-06-21 18:22:14 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_image_delete (image);
|
2006-06-21 18:22:14 +08:00
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
/* -------------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
static void
|
2019-08-31 08:43:48 +08:00
|
|
|
warp_one (GimpDrawable *draw,
|
|
|
|
GimpDrawable *new,
|
|
|
|
GimpDrawable *map_x,
|
|
|
|
GimpDrawable *map_y,
|
|
|
|
GimpDrawable *mag_draw,
|
|
|
|
gboolean first_time,
|
|
|
|
gint step)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-07-19 19:10:42 +08:00
|
|
|
GeglBuffer *src_buffer;
|
|
|
|
GeglBuffer *dest_buffer;
|
|
|
|
GeglBuffer *map_x_buffer;
|
|
|
|
GeglBuffer *map_y_buffer;
|
|
|
|
GeglBuffer *mag_buffer = NULL;
|
|
|
|
|
|
|
|
GeglBufferIterator *iter;
|
|
|
|
|
|
|
|
gint width;
|
|
|
|
gint height;
|
|
|
|
|
|
|
|
const Babl *src_format;
|
|
|
|
gint src_bytes;
|
|
|
|
const Babl *dest_format;
|
|
|
|
gint dest_bytes;
|
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
guchar pixel[4][4];
|
|
|
|
gint x1, y1, x2, y2;
|
|
|
|
gint x, y;
|
|
|
|
gint max_progress;
|
|
|
|
|
|
|
|
gdouble needx, needy;
|
|
|
|
gdouble xval=0; /* initialize to quiet compiler grumbles */
|
|
|
|
gdouble yval=0; /* interpolated vector displacement */
|
|
|
|
gdouble scalefac; /* multiplier for vector displacement scaling */
|
|
|
|
gdouble dscalefac; /* multiplier for incremental displacement vectors */
|
|
|
|
gint xi, yi;
|
|
|
|
gint substep; /* loop variable counting displacement vector substeps */
|
|
|
|
|
|
|
|
guchar values[4];
|
2002-11-21 05:29:16 +08:00
|
|
|
guint32 ivalues[4];
|
1999-09-04 07:14:44 +08:00
|
|
|
guchar val;
|
|
|
|
|
|
|
|
gint k;
|
|
|
|
|
|
|
|
gdouble dx, dy; /* X and Y Displacement, integer from GRAY map */
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
const Babl *map_x_format;
|
|
|
|
gint map_x_bytes;
|
|
|
|
const Babl *map_y_format;
|
|
|
|
gint map_y_bytes;
|
|
|
|
const Babl *mag_format;
|
|
|
|
gint mag_bytes = 1;
|
|
|
|
gboolean mag_alpha = FALSE;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
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 17:27:48 +08:00
|
|
|
GRand *gr;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
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 17:27:48 +08:00
|
|
|
gr = g_rand_new (); /* Seed Pseudo Random Number Generator */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
/* ================ Outer Loop calculation ================================ */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Get selection area */
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
if (! gimp_drawable_mask_intersect (draw,
|
2009-06-08 05:52:37 +08:00
|
|
|
&x1, &y1, &width, &height))
|
2005-03-22 07:59:12 +08:00
|
|
|
return;
|
|
|
|
|
2009-06-08 05:52:37 +08:00
|
|
|
x2 = x1 + width;
|
|
|
|
y2 = y1 + height;
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
width = gimp_drawable_width (draw);
|
|
|
|
height = gimp_drawable_height (draw);
|
2019-07-19 19:10:42 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
max_progress = (x2 - x1) * (y2 - y1);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
/* --------- Register the (many) pixel regions ---------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
src_buffer = gimp_drawable_get_buffer (draw);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
src_format = get_u8_format (draw);
|
2019-07-19 19:10:42 +08:00
|
|
|
src_bytes = babl_format_get_bytes_per_pixel (src_format);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
iter = gegl_buffer_iterator_new (src_buffer,
|
|
|
|
GEGL_RECTANGLE (x1, y1, (x2 - x1), (y2 - y1)),
|
|
|
|
0, src_format,
|
|
|
|
GEGL_ACCESS_READ, GEGL_ABYSS_NONE, 5);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
dest_buffer = gimp_drawable_get_shadow_buffer (new);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
dest_format = get_u8_format (new);
|
2019-07-19 19:10:42 +08:00
|
|
|
dest_bytes = babl_format_get_bytes_per_pixel (dest_format);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_iterator_add (iter, dest_buffer,
|
|
|
|
GEGL_RECTANGLE (x1, y1, (x2 - x1), (y2 - y1)),
|
|
|
|
0, dest_format,
|
|
|
|
GEGL_ACCESS_WRITE, GEGL_ABYSS_NONE);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
map_x_buffer = gimp_drawable_get_buffer (map_x);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
map_x_format = get_u8_format (map_x);
|
2019-07-19 19:10:42 +08:00
|
|
|
map_x_bytes = babl_format_get_bytes_per_pixel (map_x_format);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_iterator_add (iter, map_x_buffer,
|
|
|
|
GEGL_RECTANGLE (x1, y1, (x2 - x1), (y2 - y1)),
|
|
|
|
0, map_x_format,
|
|
|
|
GEGL_ACCESS_READ, GEGL_ABYSS_NONE);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
map_y_buffer = gimp_drawable_get_buffer (map_y);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
map_y_format = get_u8_format (map_y);
|
2019-07-19 19:10:42 +08:00
|
|
|
map_y_bytes = babl_format_get_bytes_per_pixel (map_y_format);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_iterator_add (iter, map_y_buffer,
|
|
|
|
GEGL_RECTANGLE (x1, y1, (x2 - x1), (y2 - y1)),
|
|
|
|
0, map_y_format,
|
|
|
|
GEGL_ACCESS_READ, GEGL_ABYSS_NONE);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
if (dvals.mag_use)
|
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
mag_buffer = gimp_drawable_get_buffer (mag_draw);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
mag_format = get_u8_format (mag_draw);
|
2019-07-19 19:10:42 +08:00
|
|
|
mag_bytes = babl_format_get_bytes_per_pixel (mag_format);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
mag_alpha = gimp_drawable_has_alpha (mag_draw);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_iterator_add (iter, mag_buffer,
|
|
|
|
GEGL_RECTANGLE (x1, y1, (x2 - x1), (y2 - y1)),
|
|
|
|
0, mag_format,
|
|
|
|
GEGL_ACCESS_READ, GEGL_ABYSS_NONE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* substep displacement vector scale factor */
|
|
|
|
dscalefac = dvals.amount / (256 * 127.5 * dvals.substeps);
|
|
|
|
|
|
|
|
while (gegl_buffer_iterator_next (iter))
|
|
|
|
{
|
|
|
|
GeglRectangle roi = iter->items[1].roi;
|
|
|
|
guchar *srcrow = iter->items[0].data;
|
|
|
|
guchar *destrow = iter->items[1].data;
|
|
|
|
guchar *mxrow = iter->items[2].data;
|
|
|
|
guchar *myrow = iter->items[3].data;
|
|
|
|
guchar *mmagrow = NULL;
|
|
|
|
|
|
|
|
if (dvals.mag_use)
|
|
|
|
mmagrow = iter->items[4].data;
|
|
|
|
|
|
|
|
/* loop over destination pixels */
|
|
|
|
for (y = roi.y; y < (roi.y + roi.height); y++)
|
|
|
|
{
|
|
|
|
guchar *dest = destrow;
|
|
|
|
guchar *mx = mxrow;
|
|
|
|
guchar *my = myrow;
|
|
|
|
guchar *mmag = NULL;
|
|
|
|
|
|
|
|
if (dvals.mag_use == TRUE)
|
|
|
|
mmag = mmagrow;
|
|
|
|
|
|
|
|
for (x = roi.x; x < (roi.x + roi.width); x++)
|
|
|
|
{
|
|
|
|
/* ----- Find displacement vector (amnt_x, amnt_y) ------------ */
|
|
|
|
|
|
|
|
dx = dscalefac * ((256.0 * mx[0]) + mx[1] -32768); /* 16-bit values */
|
|
|
|
dy = dscalefac * ((256.0 * my[0]) + my[1] -32768);
|
|
|
|
|
|
|
|
if (dvals.mag_use)
|
|
|
|
{
|
|
|
|
scalefac = warp_map_mag_give_value (mmag,
|
|
|
|
mag_alpha,
|
|
|
|
mag_bytes) / 255.0;
|
|
|
|
dx *= scalefac;
|
|
|
|
dy *= scalefac;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dvals.dither != 0.0)
|
|
|
|
{ /* random dither is +/- dvals.dither pixels */
|
|
|
|
dx += g_rand_double_range (gr, -dvals.dither, dvals.dither);
|
|
|
|
dy += g_rand_double_range (gr, -dvals.dither, dvals.dither);
|
|
|
|
}
|
2006-04-27 19:33:51 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
if (dvals.substeps != 1)
|
2006-04-27 19:33:51 +08:00
|
|
|
{ /* trace (substeps) iterations of displacement vector */
|
|
|
|
for (substep = 1; substep < dvals.substeps; substep++)
|
|
|
|
{
|
|
|
|
/* In this (substep) loop, (x,y) remain fixed. (dx,dy) vary each step. */
|
|
|
|
needx = x + dx;
|
|
|
|
needy = y + dy;
|
|
|
|
|
|
|
|
if (needx >= 0.0)
|
|
|
|
xi = (gint) needx;
|
|
|
|
else
|
|
|
|
xi = -((gint) -needx + 1);
|
|
|
|
|
|
|
|
if (needy >= 0.0)
|
|
|
|
yi = (gint) needy;
|
|
|
|
else
|
|
|
|
yi = -((gint) -needy + 1);
|
|
|
|
|
|
|
|
/* get 4 neighboring DX values from DiffX drawable for linear interpolation */
|
2019-07-19 19:10:42 +08:00
|
|
|
warp_pixel (map_x_buffer, map_x_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi, yi,
|
|
|
|
pixel[0]);
|
|
|
|
warp_pixel (map_x_buffer, map_x_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi + 1, yi,
|
|
|
|
pixel[1]);
|
|
|
|
warp_pixel (map_x_buffer, map_x_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi, yi + 1,
|
|
|
|
pixel[2]);
|
|
|
|
warp_pixel (map_x_buffer, map_x_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi + 1, yi + 1,
|
|
|
|
pixel[3]);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
|
|
|
ivalues[0] = 256 * pixel[0][0] + pixel[0][1];
|
|
|
|
ivalues[1] = 256 * pixel[1][0] + pixel[1][1];
|
|
|
|
ivalues[2] = 256 * pixel[2][0] + pixel[2][1];
|
|
|
|
ivalues[3] = 256 * pixel[3][0] + pixel[3][1];
|
|
|
|
|
|
|
|
xval = gimp_bilinear_32 (needx, needy, ivalues);
|
|
|
|
|
|
|
|
/* get 4 neighboring DY values from DiffY drawable for linear interpolation */
|
2019-07-19 19:10:42 +08:00
|
|
|
warp_pixel (map_y_buffer, map_y_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi, yi,
|
|
|
|
pixel[0]);
|
|
|
|
warp_pixel (map_y_buffer, map_y_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi + 1, yi,
|
|
|
|
pixel[1]);
|
|
|
|
warp_pixel (map_y_buffer, map_y_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi, yi + 1,
|
|
|
|
pixel[2]);
|
|
|
|
warp_pixel (map_y_buffer, map_y_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi + 1, yi + 1,
|
|
|
|
pixel[3]);
|
2006-04-27 19:33:51 +08:00
|
|
|
|
|
|
|
ivalues[0] = 256 * pixel[0][0] + pixel[0][1];
|
|
|
|
ivalues[1] = 256 * pixel[1][0] + pixel[1][1];
|
|
|
|
ivalues[2] = 256 * pixel[2][0] + pixel[2][1];
|
|
|
|
ivalues[3] = 256 * pixel[3][0] + pixel[3][1];
|
|
|
|
|
|
|
|
yval = gimp_bilinear_32 (needx, needy, ivalues);
|
|
|
|
|
|
|
|
/* move displacement vector to this new value */
|
|
|
|
dx += dscalefac * (xval - 32768);
|
|
|
|
dy += dscalefac * (yval - 32768);
|
|
|
|
|
|
|
|
} /* for (substep) */
|
|
|
|
} /* if (substeps != 0) */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
/* --------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
needx = x + dx;
|
|
|
|
needy = y + dy;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
mx += map_x_bytes; /* pointers into x,y displacement maps */
|
|
|
|
my += map_y_bytes;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
if (dvals.mag_use == TRUE)
|
2019-07-19 19:10:42 +08:00
|
|
|
mmag += mag_bytes;
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
/* Calculations complete; now copy the proper pixel */
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
if (needx >= 0.0)
|
2006-04-27 19:33:51 +08:00
|
|
|
xi = (gint) needx;
|
2008-10-20 14:04:39 +08:00
|
|
|
else
|
2006-04-27 19:33:51 +08:00
|
|
|
xi = -((gint) -needx + 1);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
if (needy >= 0.0)
|
2006-04-27 19:33:51 +08:00
|
|
|
yi = (gint) needy;
|
2008-10-20 14:04:39 +08:00
|
|
|
else
|
2006-04-27 19:33:51 +08:00
|
|
|
yi = -((gint) -needy + 1);
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
/* get 4 neighboring pixel values from source drawable
|
|
|
|
* for linear interpolation
|
|
|
|
*/
|
|
|
|
warp_pixel (src_buffer, src_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi, yi,
|
|
|
|
pixel[0]);
|
|
|
|
warp_pixel (src_buffer, src_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi + 1, yi,
|
|
|
|
pixel[1]);
|
|
|
|
warp_pixel (src_buffer, src_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi, yi + 1,
|
|
|
|
pixel[2]);
|
|
|
|
warp_pixel (src_buffer, src_format,
|
|
|
|
width, height,
|
|
|
|
x1, y1, x2, y2,
|
|
|
|
xi + 1, yi + 1,
|
|
|
|
pixel[3]);
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
for (k = 0; k < dest_bytes; k++)
|
|
|
|
{
|
|
|
|
values[0] = pixel[0][k];
|
|
|
|
values[1] = pixel[1][k];
|
|
|
|
values[2] = pixel[2][k];
|
|
|
|
values[3] = pixel[3][k];
|
2006-04-27 19:33:51 +08:00
|
|
|
|
|
|
|
val = gimp_bilinear_8 (needx, needy, values);
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
*dest++ = val;
|
2019-07-19 19:10:42 +08:00
|
|
|
}
|
|
|
|
}
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2008-10-20 14:04:39 +08:00
|
|
|
/* srcrow += src_rgn.rowstride; */
|
2019-07-19 19:10:42 +08:00
|
|
|
srcrow += src_bytes * roi.width;
|
|
|
|
destrow += dest_bytes * roi.width;
|
|
|
|
mxrow += map_x_bytes * roi.width;
|
|
|
|
myrow += map_y_bytes * roi.width;
|
2006-04-27 19:33:51 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
if (dvals.mag_use == TRUE)
|
2019-07-19 19:10:42 +08:00
|
|
|
mmagrow += mag_bytes * roi.width;
|
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
progress += (roi.width * roi.height);
|
1999-09-04 07:14:44 +08:00
|
|
|
gimp_progress_update ((double) progress / (double) max_progress);
|
2019-07-19 19:10:42 +08:00
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
g_object_unref (src_buffer);
|
|
|
|
g_object_unref (dest_buffer);
|
|
|
|
g_object_unref (map_x_buffer);
|
|
|
|
g_object_unref (map_y_buffer);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
if (dvals.mag_use == TRUE)
|
|
|
|
g_object_unref (mag_buffer);
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
gimp_progress_update (1.0);
|
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 17:27:48 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
gimp_drawable_merge_shadow (draw, first_time);
|
2003-11-06 23:27:05 +08:00
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
g_rand_free (gr);
|
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2000-01-09 04:00:10 +08:00
|
|
|
/* ------------------------------------------------------------------------- */
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
static gdouble
|
2000-01-09 04:00:10 +08:00
|
|
|
warp_map_mag_give_value (guchar *pt,
|
2008-10-20 14:04:39 +08:00
|
|
|
gint alpha,
|
|
|
|
gint bytes)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
|
|
|
gdouble ret, val_alpha;
|
2003-11-06 23:27:05 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
if (bytes >= 3)
|
|
|
|
ret = (pt[0] + pt[1] + pt[2])/3.0;
|
|
|
|
else
|
|
|
|
ret = (gdouble) *pt;
|
2003-11-06 23:27:05 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
if (alpha)
|
|
|
|
{
|
|
|
|
val_alpha = pt[bytes - 1];
|
|
|
|
ret = (ret * val_alpha / 255.0);
|
|
|
|
};
|
2003-11-06 23:27:05 +08:00
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
return (ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
static void
|
|
|
|
warp_pixel (GeglBuffer *buffer,
|
|
|
|
const Babl *format,
|
|
|
|
gint width,
|
|
|
|
gint height,
|
|
|
|
gint x1,
|
|
|
|
gint y1,
|
|
|
|
gint x2,
|
|
|
|
gint y2,
|
|
|
|
gint x,
|
|
|
|
gint y,
|
|
|
|
guchar *pixel)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-07-19 19:10:42 +08:00
|
|
|
static guchar empty_pixel[4] = { 0, 0, 0, 0 };
|
|
|
|
guchar *data;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
/* Tile the image. */
|
|
|
|
if (dvals.wrap_type == WRAP)
|
|
|
|
{
|
|
|
|
if (x < 0)
|
2008-10-20 14:04:39 +08:00
|
|
|
x = width - (-x % width);
|
1999-09-04 07:14:44 +08:00
|
|
|
else
|
2008-10-20 14:04:39 +08:00
|
|
|
x %= width;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
if (y < 0)
|
2008-10-20 14:04:39 +08:00
|
|
|
y = height - (-y % height);
|
1999-09-04 07:14:44 +08:00
|
|
|
else
|
2008-10-20 14:04:39 +08:00
|
|
|
y %= height;
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
/* Smear out the edges of the image by repeating pixels. */
|
|
|
|
else if (dvals.wrap_type == SMEAR)
|
|
|
|
{
|
|
|
|
if (x < 0)
|
2008-10-20 14:04:39 +08:00
|
|
|
x = 0;
|
1999-09-04 07:14:44 +08:00
|
|
|
else if (x > width - 1)
|
2008-10-20 14:04:39 +08:00
|
|
|
x = width - 1;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
|
|
|
if (y < 0)
|
2008-10-20 14:04:39 +08:00
|
|
|
y = 0;
|
1999-09-04 07:14:44 +08:00
|
|
|
else if (y > height - 1)
|
2008-10-20 14:04:39 +08:00
|
|
|
y = height - 1;
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (x >= x1 && y >= y1 && x < x2 && y < y2)
|
|
|
|
{
|
2019-07-19 19:10:42 +08:00
|
|
|
gegl_buffer_sample (buffer, x, y, NULL, pixel, format,
|
|
|
|
GEGL_SAMPLER_NEAREST, GEGL_ABYSS_NONE);
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-07-19 19:10:42 +08:00
|
|
|
gint bpp = babl_format_get_bytes_per_pixel (format);
|
|
|
|
gint b;
|
|
|
|
|
1999-09-04 07:14:44 +08:00
|
|
|
if (dvals.wrap_type == BLACK)
|
|
|
|
data = empty_pixel;
|
|
|
|
else
|
|
|
|
data = color_pixel; /* must have selected COLOR type */
|
|
|
|
|
2019-07-19 19:10:42 +08:00
|
|
|
for (b = 0; b < bpp; b++)
|
|
|
|
pixel[b] = data[b];
|
|
|
|
}
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Warp interface functions */
|
|
|
|
|
2004-04-22 00:50:13 +08:00
|
|
|
static gboolean
|
2019-08-31 08:43:48 +08:00
|
|
|
warp_map_constrain (GimpImage *image,
|
|
|
|
GimpItem *item,
|
2008-10-20 14:04:39 +08:00
|
|
|
gpointer data)
|
1999-09-04 07:14:44 +08:00
|
|
|
{
|
2019-08-31 08:43:48 +08:00
|
|
|
GimpDrawable *d = data;
|
1999-09-04 07:14:44 +08:00
|
|
|
|
2019-08-31 08:43:48 +08:00
|
|
|
return (gimp_drawable_width (GIMP_DRAWABLE (item)) == gimp_drawable_width (d) &&
|
|
|
|
gimp_drawable_height (GIMP_DRAWABLE (item)) == gimp_drawable_height (d));
|
1999-09-04 07:14:44 +08:00
|
|
|
}
|