2014-06-13 08:57:26 +08:00
|
|
|
/* GIMP - The GNU Image Manipulation Program
|
|
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
|
|
*
|
2018-04-14 05:36:16 +08:00
|
|
|
* Largely based on gimpdrawable-gradient.c
|
2014-06-13 08:57:26 +08:00
|
|
|
*
|
2018-04-14 05:36:16 +08:00
|
|
|
* gimpoperationgradient.c
|
2014-06-13 08:57:26 +08:00
|
|
|
* Copyright (C) 2014 Michael Henning <drawoc@darkrefraction.com>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <cairo.h>
|
|
|
|
#include <gegl.h>
|
|
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
|
|
|
|
|
|
#include "libgimpcolor/gimpcolor.h"
|
|
|
|
#include "libgimpmath/gimpmath.h"
|
|
|
|
|
|
|
|
#include "operations-types.h"
|
|
|
|
|
|
|
|
#include "core/gimpgradient.h"
|
|
|
|
|
2018-04-14 05:36:16 +08:00
|
|
|
#include "gimpoperationgradient.h"
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2016-08-29 23:38:06 +08:00
|
|
|
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
#define GRADIENT_CACHE_N_SUPERSAMPLES 4
|
|
|
|
#define GRADIENT_CACHE_MAX_SIZE ((1 << 20) / sizeof (GimpRGB))
|
|
|
|
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
2014-06-30 08:40:47 +08:00
|
|
|
PROP_CONTEXT,
|
2014-06-13 08:57:26 +08:00
|
|
|
PROP_GRADIENT,
|
|
|
|
PROP_START_X,
|
|
|
|
PROP_START_Y,
|
|
|
|
PROP_END_X,
|
|
|
|
PROP_END_Y,
|
|
|
|
PROP_GRADIENT_TYPE,
|
|
|
|
PROP_GRADIENT_REPEAT,
|
|
|
|
PROP_OFFSET,
|
|
|
|
PROP_GRADIENT_REVERSE,
|
2018-04-14 04:33:16 +08:00
|
|
|
PROP_GRADIENT_BLEND_COLOR_SPACE,
|
2014-06-13 08:57:26 +08:00
|
|
|
PROP_SUPERSAMPLE,
|
|
|
|
PROP_SUPERSAMPLE_DEPTH,
|
|
|
|
PROP_SUPERSAMPLE_THRESHOLD,
|
|
|
|
PROP_DITHER
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
GimpGradient *gradient;
|
|
|
|
gboolean reverse;
|
|
|
|
GimpGradientBlendColorSpace blend_color_space;
|
|
|
|
GimpRGB *gradient_cache;
|
|
|
|
gint gradient_cache_size;
|
|
|
|
GimpGradientSegment *last_seg;
|
|
|
|
gdouble offset;
|
|
|
|
gdouble sx, sy;
|
|
|
|
GimpGradientType gradient_type;
|
|
|
|
gdouble dist;
|
|
|
|
gdouble vec[2];
|
|
|
|
GimpRepeatMode repeat;
|
2018-05-26 00:07:38 +08:00
|
|
|
GeglSampler *dist_sampler;
|
2014-06-13 08:57:26 +08:00
|
|
|
} RenderBlendData;
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
GeglBuffer *buffer;
|
|
|
|
gfloat *row_data;
|
|
|
|
gint roi_x;
|
|
|
|
gint width;
|
|
|
|
GRand *dither_rand;
|
2014-06-13 08:57:26 +08:00
|
|
|
} PutPixelData;
|
|
|
|
|
|
|
|
|
|
|
|
/* local function prototypes */
|
|
|
|
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
static void gimp_operation_gradient_dispose (GObject *gobject);
|
2018-04-26 01:05:29 +08:00
|
|
|
static void gimp_operation_gradient_finalize (GObject *gobject);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
static void gimp_operation_gradient_get_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec);
|
|
|
|
static void gimp_operation_gradient_set_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec);
|
|
|
|
|
|
|
|
static void gimp_operation_gradient_prepare (GeglOperation *operation);
|
|
|
|
|
|
|
|
static GeglRectangle gimp_operation_gradient_get_bounding_box (GeglOperation *operation);
|
|
|
|
|
|
|
|
static gdouble gradient_calc_conical_sym_factor (gdouble dist,
|
|
|
|
gdouble *axis,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
|
|
|
static gdouble gradient_calc_conical_asym_factor (gdouble dist,
|
|
|
|
gdouble *axis,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
|
|
|
static gdouble gradient_calc_square_factor (gdouble dist,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
|
|
|
static gdouble gradient_calc_radial_factor (gdouble dist,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
|
|
|
static gdouble gradient_calc_linear_factor (gdouble dist,
|
|
|
|
gdouble *vec,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
|
|
|
static gdouble gradient_calc_bilinear_factor (gdouble dist,
|
|
|
|
gdouble *vec,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
|
|
|
static gdouble gradient_calc_spiral_factor (gdouble dist,
|
|
|
|
gdouble *axis,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y,
|
|
|
|
gboolean clockwise);
|
|
|
|
|
2018-05-26 00:07:38 +08:00
|
|
|
static gdouble gradient_calc_shapeburst_angular_factor (GeglSampler *dist_sampler,
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
2018-05-26 00:07:38 +08:00
|
|
|
static gdouble gradient_calc_shapeburst_spherical_factor (GeglSampler *dist_sampler,
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
2018-05-26 00:07:38 +08:00
|
|
|
static gdouble gradient_calc_shapeburst_dimpled_factor (GeglSampler *dist_sampler,
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
gdouble x,
|
|
|
|
gdouble y);
|
|
|
|
|
|
|
|
static void gradient_render_pixel (gdouble x,
|
|
|
|
gdouble y,
|
|
|
|
GimpRGB *color,
|
|
|
|
gpointer render_data);
|
|
|
|
|
|
|
|
static void gradient_put_pixel (gint x,
|
|
|
|
gint y,
|
|
|
|
GimpRGB *color,
|
|
|
|
gpointer put_pixel_data);
|
|
|
|
|
|
|
|
static gboolean gimp_operation_gradient_process (GeglOperation *operation,
|
|
|
|
GeglBuffer *input,
|
|
|
|
GeglBuffer *output,
|
|
|
|
const GeglRectangle *result,
|
|
|
|
gint level);
|
|
|
|
|
|
|
|
static void gimp_operation_gradient_invalidate_cache (GimpOperationGradient *self);
|
|
|
|
static void gimp_operation_gradient_validate_cache (GimpOperationGradient *self);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2016-08-29 23:38:06 +08:00
|
|
|
|
2018-04-14 05:36:16 +08:00
|
|
|
G_DEFINE_TYPE (GimpOperationGradient, gimp_operation_gradient,
|
2015-04-27 10:39:41 +08:00
|
|
|
GEGL_TYPE_OPERATION_FILTER)
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2018-04-14 05:36:16 +08:00
|
|
|
#define parent_class gimp_operation_gradient_parent_class
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2016-08-29 23:38:06 +08:00
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
static void
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_class_init (GimpOperationGradientClass *klass)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
GeglOperationClass *operation_class = GEGL_OPERATION_CLASS (klass);
|
2015-04-27 10:39:41 +08:00
|
|
|
GeglOperationFilterClass *filter_class = GEGL_OPERATION_FILTER_CLASS (klass);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2018-04-14 05:36:16 +08:00
|
|
|
object_class->dispose = gimp_operation_gradient_dispose;
|
2018-04-26 01:05:29 +08:00
|
|
|
object_class->finalize = gimp_operation_gradient_finalize;
|
2018-04-14 05:36:16 +08:00
|
|
|
object_class->set_property = gimp_operation_gradient_set_property;
|
|
|
|
object_class->get_property = gimp_operation_gradient_get_property;
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2018-04-14 05:36:16 +08:00
|
|
|
operation_class->prepare = gimp_operation_gradient_prepare;
|
|
|
|
operation_class->get_bounding_box = gimp_operation_gradient_get_bounding_box;
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2018-04-14 05:36:16 +08:00
|
|
|
filter_class->process = gimp_operation_gradient_process;
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
gegl_operation_class_set_keys (operation_class,
|
2018-04-14 05:36:16 +08:00
|
|
|
"name", "gimp:gradient",
|
2014-06-13 08:57:26 +08:00
|
|
|
"categories", "gimp",
|
2018-04-14 05:36:16 +08:00
|
|
|
"description", "GIMP Gradient operation",
|
2014-06-13 08:57:26 +08:00
|
|
|
NULL);
|
|
|
|
|
2014-06-30 08:40:47 +08:00
|
|
|
g_object_class_install_property (object_class, PROP_CONTEXT,
|
|
|
|
g_param_spec_object ("context",
|
|
|
|
"Context",
|
|
|
|
"A GimpContext",
|
|
|
|
GIMP_TYPE_OBJECT,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
g_object_class_install_property (object_class, PROP_GRADIENT,
|
|
|
|
g_param_spec_object ("gradient",
|
|
|
|
"Gradient",
|
|
|
|
"A GimpGradient to render",
|
|
|
|
GIMP_TYPE_OBJECT,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_START_X,
|
|
|
|
g_param_spec_double ("start-x",
|
|
|
|
"Start X",
|
|
|
|
"X coordinate of the first point",
|
|
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, 0,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_START_Y,
|
|
|
|
g_param_spec_double ("start-y",
|
|
|
|
"Start Y",
|
|
|
|
"Y coordinate of the first point",
|
|
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, 0,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_END_X,
|
|
|
|
g_param_spec_double ("end-x",
|
|
|
|
"End X",
|
|
|
|
"X coordinate of the second point",
|
|
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, 200,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_END_Y,
|
|
|
|
g_param_spec_double ("end-y",
|
|
|
|
"End Y",
|
|
|
|
"Y coordinate of the second point",
|
|
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, 200,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_GRADIENT_TYPE,
|
|
|
|
g_param_spec_enum ("gradient-type",
|
|
|
|
"Gradient Type",
|
|
|
|
"The type of gradient to render",
|
|
|
|
GIMP_TYPE_GRADIENT_TYPE,
|
|
|
|
GIMP_GRADIENT_LINEAR,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_GRADIENT_REPEAT,
|
|
|
|
g_param_spec_enum ("gradient-repeat",
|
|
|
|
"Repeat mode",
|
|
|
|
"Repeat mode",
|
|
|
|
GIMP_TYPE_REPEAT_MODE,
|
|
|
|
GIMP_REPEAT_NONE,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_OFFSET,
|
|
|
|
g_param_spec_double ("offset",
|
|
|
|
"Offset",
|
|
|
|
"Offset relates to the starting and ending coordinates "
|
2016-12-21 11:05:32 +08:00
|
|
|
"specified for the blend. This parameter is mode dependent.",
|
2014-06-13 08:57:26 +08:00
|
|
|
0, G_MAXDOUBLE, 0,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_GRADIENT_REVERSE,
|
|
|
|
g_param_spec_boolean ("gradient-reverse",
|
|
|
|
"Reverse",
|
|
|
|
"Reverse the gradient",
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
2018-04-14 04:33:16 +08:00
|
|
|
g_object_class_install_property (object_class, PROP_GRADIENT_BLEND_COLOR_SPACE,
|
|
|
|
g_param_spec_enum ("gradient-blend-color-space",
|
|
|
|
"Blend Color Space",
|
|
|
|
"Which color space to use when blending RGB gradient segments",
|
|
|
|
GIMP_TYPE_GRADIENT_BLEND_COLOR_SPACE,
|
|
|
|
GIMP_GRADIENT_BLEND_RGB_PERCEPTUAL,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
g_object_class_install_property (object_class, PROP_SUPERSAMPLE,
|
|
|
|
g_param_spec_boolean ("supersample",
|
|
|
|
"Supersample",
|
|
|
|
"Do adaptive supersampling",
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_SUPERSAMPLE_DEPTH,
|
|
|
|
g_param_spec_int ("supersample-depth",
|
|
|
|
"Max depth",
|
|
|
|
"Maximum recursion levels for supersampling",
|
|
|
|
1, 9, 3,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_SUPERSAMPLE_THRESHOLD,
|
|
|
|
g_param_spec_double ("supersample-threshold",
|
|
|
|
"Threshold",
|
|
|
|
"Supersampling threshold",
|
|
|
|
0, 4, 0.20,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_DITHER,
|
|
|
|
g_param_spec_boolean ("dither",
|
|
|
|
"Dither",
|
|
|
|
"Use dithering to reduce banding",
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_init (GimpOperationGradient *self)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
2018-04-26 01:05:29 +08:00
|
|
|
g_mutex_init (&self->gradient_cache_mutex);
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_dispose (GObject *object)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
2018-04-14 05:36:16 +08:00
|
|
|
GimpOperationGradient *self = GIMP_OPERATION_GRADIENT (object);
|
2014-06-30 08:40:47 +08:00
|
|
|
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
g_clear_object (&self->gradient);
|
2014-06-30 08:40:47 +08:00
|
|
|
g_clear_object (&self->context);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
|
|
}
|
|
|
|
|
2018-04-26 01:05:29 +08:00
|
|
|
static void
|
|
|
|
gimp_operation_gradient_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
GimpOperationGradient *self = GIMP_OPERATION_GRADIENT (object);
|
|
|
|
|
|
|
|
g_mutex_clear (&self->gradient_cache_mutex);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
static void
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_get_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
2018-04-14 05:36:16 +08:00
|
|
|
GimpOperationGradient *self = GIMP_OPERATION_GRADIENT (object);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
2014-06-30 08:40:47 +08:00
|
|
|
case PROP_CONTEXT:
|
|
|
|
g_value_set_object (value, self->context);
|
|
|
|
break;
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
case PROP_GRADIENT:
|
|
|
|
g_value_set_object (value, self->gradient);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_START_X:
|
|
|
|
g_value_set_double (value, self->start_x);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_START_Y:
|
|
|
|
g_value_set_double (value, self->start_y);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_END_X:
|
|
|
|
g_value_set_double (value, self->end_x);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_END_Y:
|
|
|
|
g_value_set_double (value, self->end_y);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_GRADIENT_TYPE:
|
|
|
|
g_value_set_enum (value, self->gradient_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_GRADIENT_REPEAT:
|
|
|
|
g_value_set_enum (value, self->gradient_repeat);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_OFFSET:
|
|
|
|
g_value_set_double (value, self->offset);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_GRADIENT_REVERSE:
|
|
|
|
g_value_set_boolean (value, self->gradient_reverse);
|
|
|
|
break;
|
|
|
|
|
2018-04-14 04:33:16 +08:00
|
|
|
case PROP_GRADIENT_BLEND_COLOR_SPACE:
|
|
|
|
g_value_set_enum (value, self->gradient_blend_color_space);
|
|
|
|
break;
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
case PROP_SUPERSAMPLE:
|
|
|
|
g_value_set_boolean (value, self->supersample);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_SUPERSAMPLE_DEPTH:
|
|
|
|
g_value_set_int (value, self->supersample_depth);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_SUPERSAMPLE_THRESHOLD:
|
|
|
|
g_value_set_double (value, self->supersample_threshold);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_DITHER:
|
|
|
|
g_value_set_boolean (value, self->dither);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_set_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
2018-04-14 05:36:16 +08:00
|
|
|
GimpOperationGradient *self = GIMP_OPERATION_GRADIENT (object);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
2014-06-30 08:40:47 +08:00
|
|
|
case PROP_CONTEXT:
|
|
|
|
if (self->context)
|
|
|
|
g_object_unref (self->context);
|
|
|
|
|
|
|
|
self->context = g_value_dup_object (value);
|
|
|
|
break;
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
case PROP_GRADIENT:
|
|
|
|
{
|
2014-06-30 08:40:47 +08:00
|
|
|
GimpGradient *gradient = g_value_get_object (value);
|
|
|
|
|
2018-04-14 04:33:16 +08:00
|
|
|
g_clear_object (&self->gradient);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2014-06-30 08:40:47 +08:00
|
|
|
if (gradient)
|
|
|
|
{
|
|
|
|
if (gimp_gradient_has_fg_bg_segments (gradient))
|
|
|
|
self->gradient = gimp_gradient_flatten (gradient, self->context);
|
|
|
|
else
|
|
|
|
self->gradient = g_object_ref (gradient);
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_START_X:
|
|
|
|
self->start_x = g_value_get_double (value);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_START_Y:
|
|
|
|
self->start_y = g_value_get_double (value);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_END_X:
|
|
|
|
self->end_x = g_value_get_double (value);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_END_Y:
|
|
|
|
self->end_y = g_value_get_double (value);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_GRADIENT_TYPE:
|
|
|
|
self->gradient_type = g_value_get_enum (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_GRADIENT_REPEAT:
|
|
|
|
self->gradient_repeat = g_value_get_enum (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_OFFSET:
|
|
|
|
self->offset = g_value_get_double (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_GRADIENT_REVERSE:
|
|
|
|
self->gradient_reverse = g_value_get_boolean (value);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
2018-04-14 04:33:16 +08:00
|
|
|
case PROP_GRADIENT_BLEND_COLOR_SPACE:
|
|
|
|
self->gradient_blend_color_space = g_value_get_enum (value);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
gimp_operation_gradient_invalidate_cache (self);
|
2018-04-14 04:33:16 +08:00
|
|
|
break;
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
case PROP_SUPERSAMPLE:
|
|
|
|
self->supersample = g_value_get_boolean (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_SUPERSAMPLE_DEPTH:
|
|
|
|
self->supersample_depth = g_value_get_int (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_SUPERSAMPLE_THRESHOLD:
|
|
|
|
self->supersample_threshold = g_value_get_double (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_DITHER:
|
|
|
|
self->dither = g_value_get_boolean (value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_prepare (GeglOperation *operation)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
|
|
|
gegl_operation_set_format (operation, "output", babl_format ("R'G'B'A float"));
|
|
|
|
}
|
|
|
|
|
|
|
|
static GeglRectangle
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_get_bounding_box (GeglOperation *operation)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
|
|
|
return gegl_rectangle_infinite_plane ();
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
|
|
|
gradient_calc_conical_sym_factor (gdouble dist,
|
|
|
|
gdouble *axis,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
|
|
|
{
|
|
|
|
if (dist == 0.0)
|
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
else if ((x != 0) || (y != 0))
|
|
|
|
{
|
|
|
|
gdouble vec[2];
|
|
|
|
gdouble r;
|
|
|
|
gdouble rat;
|
|
|
|
|
|
|
|
/* Calculate offset from the start in pixels */
|
|
|
|
|
|
|
|
r = sqrt (SQR (x) + SQR (y));
|
|
|
|
|
|
|
|
vec[0] = x / r;
|
|
|
|
vec[1] = y / r;
|
|
|
|
|
|
|
|
rat = axis[0] * vec[0] + axis[1] * vec[1]; /* Dot product */
|
|
|
|
|
|
|
|
if (rat > 1.0)
|
|
|
|
rat = 1.0;
|
|
|
|
else if (rat < -1.0)
|
|
|
|
rat = -1.0;
|
|
|
|
|
|
|
|
/* This cool idea is courtesy Josh MacDonald,
|
|
|
|
* Ali Rahimi --- two more XCF losers. */
|
|
|
|
|
|
|
|
rat = acos (rat) / G_PI;
|
|
|
|
rat = pow (rat, (offset / 10.0) + 1.0);
|
|
|
|
|
|
|
|
return CLAMP (rat, 0.0, 1.0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0.5;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
|
|
|
gradient_calc_conical_asym_factor (gdouble dist,
|
|
|
|
gdouble *axis,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
|
|
|
{
|
|
|
|
if (dist == 0.0)
|
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
else if (x != 0 || y != 0)
|
|
|
|
{
|
|
|
|
gdouble ang0, ang1;
|
|
|
|
gdouble ang;
|
|
|
|
gdouble rat;
|
|
|
|
|
|
|
|
ang0 = atan2 (axis[0], axis[1]) + G_PI;
|
|
|
|
|
|
|
|
ang1 = atan2 (x, y) + G_PI;
|
|
|
|
|
|
|
|
ang = ang1 - ang0;
|
|
|
|
|
|
|
|
if (ang < 0.0)
|
|
|
|
ang += (2.0 * G_PI);
|
|
|
|
|
|
|
|
rat = ang / (2.0 * G_PI);
|
|
|
|
rat = pow (rat, (offset / 10.0) + 1.0);
|
|
|
|
|
|
|
|
return CLAMP (rat, 0.0, 1.0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0.5; /* We are on middle point */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
|
|
|
gradient_calc_square_factor (gdouble dist,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
|
|
|
{
|
|
|
|
if (dist == 0.0)
|
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gdouble r;
|
|
|
|
gdouble rat;
|
|
|
|
|
|
|
|
/* Calculate offset from start as a value in [0, 1] */
|
|
|
|
|
|
|
|
offset = offset / 100.0;
|
|
|
|
|
|
|
|
r = MAX (fabs (x), fabs (y));
|
|
|
|
rat = r / dist;
|
|
|
|
|
|
|
|
if (rat < offset)
|
|
|
|
return 0.0;
|
|
|
|
else if (offset == 1.0)
|
|
|
|
return (rat >= 1.0) ? 1.0 : 0.0;
|
|
|
|
else
|
|
|
|
return (rat - offset) / (1.0 - offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
|
|
|
gradient_calc_radial_factor (gdouble dist,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
|
|
|
{
|
|
|
|
if (dist == 0.0)
|
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gdouble r;
|
|
|
|
gdouble rat;
|
|
|
|
|
|
|
|
/* Calculate radial offset from start as a value in [0, 1] */
|
|
|
|
|
|
|
|
offset = offset / 100.0;
|
|
|
|
|
|
|
|
r = sqrt (SQR (x) + SQR (y));
|
|
|
|
rat = r / dist;
|
|
|
|
|
|
|
|
if (rat < offset)
|
|
|
|
return 0.0;
|
|
|
|
else if (offset == 1.0)
|
|
|
|
return (rat >= 1.0) ? 1.0 : 0.0;
|
|
|
|
else
|
|
|
|
return (rat - offset) / (1.0 - offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
|
|
|
gradient_calc_linear_factor (gdouble dist,
|
|
|
|
gdouble *vec,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
|
|
|
{
|
|
|
|
if (dist == 0.0)
|
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gdouble r;
|
|
|
|
gdouble rat;
|
|
|
|
|
|
|
|
offset = offset / 100.0;
|
|
|
|
|
|
|
|
r = vec[0] * x + vec[1] * y;
|
|
|
|
rat = r / dist;
|
|
|
|
|
|
|
|
if (rat >= 0.0 && rat < offset)
|
|
|
|
return 0.0;
|
|
|
|
else if (offset == 1.0)
|
|
|
|
return (rat >= 1.0) ? 1.0 : 0.0;
|
|
|
|
else if (rat < 0.0)
|
|
|
|
return rat / (1.0 - offset);
|
|
|
|
else
|
|
|
|
return (rat - offset) / (1.0 - offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
|
|
|
gradient_calc_bilinear_factor (gdouble dist,
|
|
|
|
gdouble *vec,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
|
|
|
{
|
|
|
|
if (dist == 0.0)
|
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gdouble r;
|
|
|
|
gdouble rat;
|
|
|
|
|
|
|
|
/* Calculate linear offset from the start line outward */
|
|
|
|
|
|
|
|
offset = offset / 100.0;
|
|
|
|
|
|
|
|
r = vec[0] * x + vec[1] * y;
|
|
|
|
rat = r / dist;
|
|
|
|
|
|
|
|
if (fabs (rat) < offset)
|
|
|
|
return 0.0;
|
|
|
|
else if (offset == 1.0)
|
|
|
|
return (rat == 1.0) ? 1.0 : 0.0;
|
|
|
|
else
|
|
|
|
return (fabs (rat) - offset) / (1.0 - offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
|
|
|
gradient_calc_spiral_factor (gdouble dist,
|
|
|
|
gdouble *axis,
|
|
|
|
gdouble offset,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y,
|
|
|
|
gboolean clockwise)
|
|
|
|
{
|
|
|
|
if (dist == 0.0)
|
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
else if (x != 0.0 || y != 0.0)
|
|
|
|
{
|
|
|
|
gdouble ang0, ang1;
|
|
|
|
gdouble ang;
|
|
|
|
double r;
|
|
|
|
|
|
|
|
ang0 = atan2 (axis[0], axis[1]) + G_PI;
|
|
|
|
ang1 = atan2 (x, y) + G_PI;
|
|
|
|
|
|
|
|
if (clockwise)
|
|
|
|
ang = ang1 - ang0;
|
|
|
|
else
|
|
|
|
ang = ang0 - ang1;
|
|
|
|
|
|
|
|
if (ang < 0.0)
|
|
|
|
ang += (2.0 * G_PI);
|
|
|
|
|
|
|
|
r = sqrt (SQR (x) + SQR (y)) / dist;
|
|
|
|
|
|
|
|
return fmod (ang / (2.0 * G_PI) + r + offset, 1.0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0.5 ; /* We are on the middle point */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gdouble
|
2018-05-26 00:07:38 +08:00
|
|
|
gradient_calc_shapeburst_angular_factor (GeglSampler *dist_sampler,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
|
|
|
gfloat value;
|
|
|
|
|
2018-05-26 00:07:38 +08:00
|
|
|
gegl_sampler_get (dist_sampler, x, y, NULL, &value, GEGL_ABYSS_NONE);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
value = 1.0 - value;
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gdouble
|
2018-05-26 00:07:38 +08:00
|
|
|
gradient_calc_shapeburst_spherical_factor (GeglSampler *dist_sampler,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
|
|
|
gfloat value;
|
|
|
|
|
2018-05-26 00:07:38 +08:00
|
|
|
gegl_sampler_get (dist_sampler, x, y, NULL, &value, GEGL_ABYSS_NONE);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
value = 1.0 - sin (0.5 * G_PI * value);
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gdouble
|
2018-05-26 00:07:38 +08:00
|
|
|
gradient_calc_shapeburst_dimpled_factor (GeglSampler *dist_sampler,
|
|
|
|
gdouble x,
|
|
|
|
gdouble y)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
|
|
|
gfloat value;
|
|
|
|
|
2018-05-26 00:07:38 +08:00
|
|
|
gegl_sampler_get (dist_sampler, x, y, NULL, &value, GEGL_ABYSS_NONE);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
value = cos (0.5 * G_PI * value);
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gradient_render_pixel (gdouble x,
|
|
|
|
gdouble y,
|
|
|
|
GimpRGB *color,
|
|
|
|
gpointer render_data)
|
|
|
|
{
|
|
|
|
RenderBlendData *rbd = render_data;
|
|
|
|
gdouble factor;
|
|
|
|
|
2015-08-27 05:23:16 +08:00
|
|
|
/* we want to calculate the color at the pixel's center */
|
|
|
|
x += 0.5;
|
|
|
|
y += 0.5;
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
/* Calculate blending factor */
|
|
|
|
|
|
|
|
switch (rbd->gradient_type)
|
|
|
|
{
|
|
|
|
case GIMP_GRADIENT_LINEAR:
|
|
|
|
factor = gradient_calc_linear_factor (rbd->dist,
|
|
|
|
rbd->vec, rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_BILINEAR:
|
|
|
|
factor = gradient_calc_bilinear_factor (rbd->dist,
|
|
|
|
rbd->vec, rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_RADIAL:
|
|
|
|
factor = gradient_calc_radial_factor (rbd->dist,
|
|
|
|
rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SQUARE:
|
|
|
|
factor = gradient_calc_square_factor (rbd->dist, rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_CONICAL_SYMMETRIC:
|
|
|
|
factor = gradient_calc_conical_sym_factor (rbd->dist,
|
|
|
|
rbd->vec, rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_CONICAL_ASYMMETRIC:
|
|
|
|
factor = gradient_calc_conical_asym_factor (rbd->dist,
|
|
|
|
rbd->vec, rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SHAPEBURST_ANGULAR:
|
2018-05-26 00:07:38 +08:00
|
|
|
factor = gradient_calc_shapeburst_angular_factor (rbd->dist_sampler,
|
|
|
|
x, y);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SHAPEBURST_SPHERICAL:
|
2018-05-26 00:07:38 +08:00
|
|
|
factor = gradient_calc_shapeburst_spherical_factor (rbd->dist_sampler,
|
|
|
|
x, y);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SHAPEBURST_DIMPLED:
|
2018-05-26 00:07:38 +08:00
|
|
|
factor = gradient_calc_shapeburst_dimpled_factor (rbd->dist_sampler,
|
|
|
|
x, y);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SPIRAL_CLOCKWISE:
|
|
|
|
factor = gradient_calc_spiral_factor (rbd->dist,
|
|
|
|
rbd->vec, rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy, TRUE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SPIRAL_ANTICLOCKWISE:
|
|
|
|
factor = gradient_calc_spiral_factor (rbd->dist,
|
|
|
|
rbd->vec, rbd->offset,
|
|
|
|
x - rbd->sx, y - rbd->sy, FALSE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_return_if_reached ();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Adjust for repeat */
|
|
|
|
|
|
|
|
switch (rbd->repeat)
|
|
|
|
{
|
|
|
|
case GIMP_REPEAT_NONE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_REPEAT_SAWTOOTH:
|
|
|
|
factor = factor - floor (factor);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_REPEAT_TRIANGULAR:
|
|
|
|
{
|
|
|
|
guint ifactor;
|
|
|
|
|
|
|
|
if (factor < 0.0)
|
|
|
|
factor = -factor;
|
|
|
|
|
|
|
|
ifactor = (guint) factor;
|
|
|
|
factor = factor - floor (factor);
|
|
|
|
|
|
|
|
if (ifactor & 1)
|
|
|
|
factor = 1.0 - factor;
|
|
|
|
}
|
|
|
|
break;
|
2018-04-18 07:11:33 +08:00
|
|
|
|
|
|
|
case GIMP_REPEAT_TRUNCATE:
|
|
|
|
if (factor < 0.0 || factor > 1.0)
|
|
|
|
{
|
|
|
|
gimp_rgba_set (color, 0.0, 0.0, 0.0, 0.0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Blend the colors */
|
|
|
|
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
if (rbd->gradient_cache)
|
|
|
|
{
|
|
|
|
factor = CLAMP (factor, 0.0, 1.0);
|
2018-04-18 07:11:33 +08:00
|
|
|
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
*color =
|
|
|
|
rbd->gradient_cache[ROUND (factor * (rbd->gradient_cache_size - 1))];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
rbd->last_seg = gimp_gradient_get_color_at (rbd->gradient, NULL,
|
|
|
|
rbd->last_seg, factor,
|
|
|
|
rbd->reverse,
|
|
|
|
rbd->blend_color_space,
|
|
|
|
color);
|
|
|
|
}
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gradient_put_pixel (gint x,
|
|
|
|
gint y,
|
|
|
|
GimpRGB *color,
|
|
|
|
gpointer put_pixel_data)
|
|
|
|
{
|
|
|
|
PutPixelData *ppd = put_pixel_data;
|
|
|
|
const gint index = x - ppd->roi_x;
|
|
|
|
gfloat *dest = ppd->row_data + 4 * index;
|
|
|
|
|
|
|
|
if (ppd->dither_rand)
|
|
|
|
{
|
2014-09-09 04:49:51 +08:00
|
|
|
gfloat r, g, b, a;
|
|
|
|
gint i = g_rand_int (ppd->dither_rand);
|
|
|
|
|
|
|
|
r = color->r + (gdouble) (i & 0xff) / 256.0 / 256.0; i >>= 8;
|
|
|
|
g = color->g + (gdouble) (i & 0xff) / 256.0 / 256.0; i >>= 8;
|
|
|
|
b = color->b + (gdouble) (i & 0xff) / 256.0 / 256.0; i >>= 8;
|
2016-04-17 08:06:08 +08:00
|
|
|
|
|
|
|
if (color->a > 0.0 && color->a < 1.0)
|
|
|
|
a = color->a + (gdouble) (i & 0xff) / 256.0 / 256.0;
|
|
|
|
else
|
|
|
|
a = color->a;
|
2014-09-09 04:49:51 +08:00
|
|
|
|
|
|
|
*dest++ = MAX (r, 0.0);
|
|
|
|
*dest++ = MAX (g, 0.0);
|
|
|
|
*dest++ = MAX (b, 0.0);
|
|
|
|
*dest++ = MAX (a, 0.0);
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*dest++ = color->r;
|
|
|
|
*dest++ = color->g;
|
|
|
|
*dest++ = color->b;
|
|
|
|
*dest++ = color->a;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Paint whole row if we are on the rightmost pixel */
|
|
|
|
if (index == (ppd->width - 1))
|
|
|
|
gegl_buffer_set (ppd->buffer, GEGL_RECTANGLE (ppd->roi_x, y, ppd->width, 1),
|
|
|
|
0, babl_format ("R'G'B'A float"), ppd->row_data,
|
|
|
|
GEGL_AUTO_ROWSTRIDE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2018-04-14 05:36:16 +08:00
|
|
|
gimp_operation_gradient_process (GeglOperation *operation,
|
|
|
|
GeglBuffer *input,
|
|
|
|
GeglBuffer *output,
|
|
|
|
const GeglRectangle *result,
|
|
|
|
gint level)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
2018-04-14 05:36:16 +08:00
|
|
|
GimpOperationGradient *self = GIMP_OPERATION_GRADIENT (operation);
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
const gdouble sx = self->start_x;
|
|
|
|
const gdouble sy = self->start_y;
|
|
|
|
const gdouble ex = self->end_x;
|
|
|
|
const gdouble ey = self->end_y;
|
|
|
|
|
|
|
|
RenderBlendData rbd = { 0, };
|
|
|
|
|
2018-04-18 07:11:33 +08:00
|
|
|
if (! self->gradient)
|
|
|
|
return TRUE;
|
2014-06-13 08:57:26 +08:00
|
|
|
|
2018-04-26 01:05:29 +08:00
|
|
|
gimp_operation_gradient_validate_cache (self);
|
|
|
|
|
2018-04-18 07:11:33 +08:00
|
|
|
rbd.gradient = self->gradient;
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
rbd.reverse = self->gradient_reverse;
|
|
|
|
rbd.blend_color_space = self->gradient_blend_color_space;
|
2018-04-18 07:11:33 +08:00
|
|
|
rbd.gradient_cache = self->gradient_cache;
|
|
|
|
rbd.gradient_cache_size = self->gradient_cache_size;
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
/* Calculate type-specific parameters */
|
|
|
|
|
|
|
|
switch (self->gradient_type)
|
|
|
|
{
|
|
|
|
case GIMP_GRADIENT_RADIAL:
|
|
|
|
rbd.dist = sqrt (SQR (ex - sx) + SQR (ey - sy));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SQUARE:
|
|
|
|
rbd.dist = MAX (fabs (ex - sx), fabs (ey - sy));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_CONICAL_SYMMETRIC:
|
|
|
|
case GIMP_GRADIENT_CONICAL_ASYMMETRIC:
|
|
|
|
case GIMP_GRADIENT_SPIRAL_CLOCKWISE:
|
|
|
|
case GIMP_GRADIENT_SPIRAL_ANTICLOCKWISE:
|
|
|
|
case GIMP_GRADIENT_LINEAR:
|
|
|
|
case GIMP_GRADIENT_BILINEAR:
|
|
|
|
rbd.dist = sqrt (SQR (ex - sx) + SQR (ey - sy));
|
|
|
|
|
|
|
|
if (rbd.dist > 0.0)
|
|
|
|
{
|
|
|
|
rbd.vec[0] = (ex - sx) / rbd.dist;
|
|
|
|
rbd.vec[1] = (ey - sy) / rbd.dist;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GIMP_GRADIENT_SHAPEBURST_ANGULAR:
|
|
|
|
case GIMP_GRADIENT_SHAPEBURST_SPHERICAL:
|
|
|
|
case GIMP_GRADIENT_SHAPEBURST_DIMPLED:
|
|
|
|
rbd.dist = sqrt (SQR (ex - sx) + SQR (ey - sy));
|
2018-05-26 00:07:38 +08:00
|
|
|
rbd.dist_sampler = gegl_buffer_sampler_new_at_level (
|
|
|
|
input, babl_format ("Y float"), GEGL_SAMPLER_NEAREST, level);
|
2014-06-13 08:57:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_return_val_if_reached (FALSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize render data */
|
|
|
|
|
|
|
|
rbd.offset = self->offset;
|
|
|
|
rbd.sx = self->start_x;
|
|
|
|
rbd.sy = self->start_y;
|
|
|
|
rbd.gradient_type = self->gradient_type;
|
|
|
|
rbd.repeat = self->gradient_repeat;
|
|
|
|
|
|
|
|
/* Render the gradient! */
|
|
|
|
|
|
|
|
if (self->supersample)
|
|
|
|
{
|
2014-09-09 04:49:51 +08:00
|
|
|
PutPixelData ppd = { 0, };
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
ppd.buffer = output;
|
|
|
|
ppd.row_data = g_malloc (sizeof (float) * 4 * result->width);
|
|
|
|
ppd.roi_x = result->x;
|
|
|
|
ppd.width = result->width;
|
|
|
|
if (self->dither)
|
|
|
|
ppd.dither_rand = g_rand_new ();
|
|
|
|
|
|
|
|
gimp_adaptive_supersample_area (result->x, result->y,
|
|
|
|
result->x + result->width - 1,
|
|
|
|
result->y + result->height - 1,
|
|
|
|
self->supersample_depth,
|
|
|
|
self->supersample_threshold,
|
|
|
|
gradient_render_pixel, &rbd,
|
|
|
|
gradient_put_pixel, &ppd,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (self->dither)
|
|
|
|
g_rand_free (ppd.dither_rand);
|
|
|
|
g_free (ppd.row_data);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GeglBufferIterator *iter;
|
|
|
|
GeglRectangle *roi;
|
2018-04-18 07:11:33 +08:00
|
|
|
GRand *seed = NULL;
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
iter = gegl_buffer_iterator_new (output, result, 0,
|
|
|
|
babl_format ("R'G'B'A float"),
|
2014-07-23 02:07:09 +08:00
|
|
|
GEGL_ACCESS_WRITE, GEGL_ABYSS_NONE);
|
2014-06-13 08:57:26 +08:00
|
|
|
roi = &iter->roi[0];
|
|
|
|
|
|
|
|
if (self->dither)
|
2018-04-18 07:11:33 +08:00
|
|
|
seed = g_rand_new ();
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
while (gegl_buffer_iterator_next (iter))
|
|
|
|
{
|
|
|
|
gfloat *dest = iter->data[0];
|
2014-09-09 04:49:51 +08:00
|
|
|
gint endx = roi->x + roi->width;
|
|
|
|
gint endy = roi->y + roi->height;
|
2014-06-13 08:57:26 +08:00
|
|
|
gint x, y;
|
|
|
|
|
2018-04-18 07:11:33 +08:00
|
|
|
if (seed)
|
2014-06-13 08:57:26 +08:00
|
|
|
{
|
2018-04-18 07:11:33 +08:00
|
|
|
GRand *dither_rand = g_rand_new_with_seed (g_rand_int (seed));
|
2014-06-13 08:57:26 +08:00
|
|
|
|
|
|
|
for (y = roi->y; y < endy; y++)
|
|
|
|
for (x = roi->x; x < endx; x++)
|
|
|
|
{
|
|
|
|
GimpRGB color = { 0.0, 0.0, 0.0, 1.0 };
|
2014-09-09 04:49:51 +08:00
|
|
|
gfloat r, g, b, a;
|
2014-06-13 08:57:26 +08:00
|
|
|
gint i = g_rand_int (dither_rand);
|
|
|
|
|
|
|
|
gradient_render_pixel (x, y, &color, &rbd);
|
|
|
|
|
2014-09-09 04:49:51 +08:00
|
|
|
r = color.r + (gdouble) (i & 0xff) / 256.0 / 256.0; i >>= 8;
|
|
|
|
g = color.g + (gdouble) (i & 0xff) / 256.0 / 256.0; i >>= 8;
|
|
|
|
b = color.b + (gdouble) (i & 0xff) / 256.0 / 256.0; i >>= 8;
|
2016-04-17 08:06:08 +08:00
|
|
|
|
|
|
|
if (color.a > 0.0 && color.a < 1.0)
|
|
|
|
a = color.a + (gdouble) (i & 0xff) / 256.0 / 256.0;
|
|
|
|
else
|
|
|
|
a = color.a;
|
2014-09-09 04:49:51 +08:00
|
|
|
|
|
|
|
*dest++ = MAX (r, 0.0);
|
|
|
|
*dest++ = MAX (g, 0.0);
|
|
|
|
*dest++ = MAX (b, 0.0);
|
|
|
|
*dest++ = MAX (a, 0.0);
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
g_rand_free (dither_rand);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (y = roi->y; y < endy; y++)
|
|
|
|
for (x = roi->x; x < endx; x++)
|
|
|
|
{
|
|
|
|
GimpRGB color = { 0.0, 0.0, 0.0, 1.0 };
|
|
|
|
|
|
|
|
gradient_render_pixel (x, y, &color, &rbd);
|
|
|
|
|
|
|
|
*dest++ = color.r;
|
|
|
|
*dest++ = color.g;
|
|
|
|
*dest++ = color.b;
|
|
|
|
*dest++ = color.a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self->dither)
|
2018-04-18 07:11:33 +08:00
|
|
|
g_rand_free (seed);
|
2014-06-13 08:57:26 +08:00
|
|
|
}
|
|
|
|
|
2018-05-26 00:07:38 +08:00
|
|
|
g_clear_object (&rbd.dist_sampler);
|
|
|
|
|
2014-06-13 08:57:26 +08:00
|
|
|
return TRUE;
|
|
|
|
}
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
static void
|
|
|
|
gimp_operation_gradient_invalidate_cache (GimpOperationGradient *self)
|
|
|
|
{
|
|
|
|
g_clear_pointer (&self->gradient_cache, g_free);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gimp_operation_gradient_validate_cache (GimpOperationGradient *self)
|
|
|
|
{
|
|
|
|
GimpGradientSegment *last_seg = NULL;
|
|
|
|
gint cache_size;
|
|
|
|
gint i;
|
|
|
|
|
2018-04-26 01:05:29 +08:00
|
|
|
if (! self->gradient)
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
return;
|
|
|
|
|
2018-04-26 01:05:29 +08:00
|
|
|
g_mutex_lock (&self->gradient_cache_mutex);
|
|
|
|
|
|
|
|
if (self->gradient_cache)
|
|
|
|
{
|
|
|
|
g_mutex_unlock (&self->gradient_cache_mutex);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
cache_size = ceil (hypot (self->start_x - self->end_x,
|
|
|
|
self->start_y - self->end_y)) *
|
|
|
|
GRADIENT_CACHE_N_SUPERSAMPLES;
|
|
|
|
|
|
|
|
/* have at least two values in the cache */
|
|
|
|
cache_size = MAX (cache_size, 2);
|
|
|
|
|
|
|
|
/* don't use a cache if its necessary size is too big */
|
|
|
|
if (cache_size > GRADIENT_CACHE_MAX_SIZE)
|
2018-04-26 01:05:29 +08:00
|
|
|
{
|
|
|
|
g_mutex_unlock (&self->gradient_cache_mutex);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
|
|
|
|
self->gradient_cache = g_new0 (GimpRGB, cache_size);
|
|
|
|
self->gradient_cache_size = cache_size;
|
|
|
|
|
|
|
|
for (i = 0; i < self->gradient_cache_size; i++)
|
|
|
|
{
|
|
|
|
gdouble factor = (gdouble) i / (gdouble) (self->gradient_cache_size - 1);
|
|
|
|
|
|
|
|
last_seg = gimp_gradient_get_color_at (self->gradient, NULL, last_seg,
|
|
|
|
factor,
|
|
|
|
self->gradient_reverse,
|
|
|
|
self->gradient_blend_color_space,
|
|
|
|
self->gradient_cache + i);
|
|
|
|
}
|
2018-04-26 01:05:29 +08:00
|
|
|
|
|
|
|
g_mutex_unlock (&self->gradient_cache_mutex);
|
app: various improvements to GimpOperationGradient
Invalidate the gradient cache while setting relevant properties,
and validate it, reconstructing if necessary, during prepare(),
rather than process(), to avoid the need to use a mutex.
Make sure the cache has at least two elements, corresponding to
the initial and final colors of the gradient, since both colors
might be needed, and to avoid division by zero.
Avoid using a cache if its necessary size is too big, or if the
gradient type is conical, since the necessary cache size for
conical gradients is unrelated to the gradient line length.
Improve index rounding during cache lookup.
Lots of indentation fixes.
2018-04-25 21:41:13 +08:00
|
|
|
}
|