summaryrefslogtreecommitdiffstats
path: root/app/core/gimpdrawable-gradient.c
diff options
context:
space:
mode:
Diffstat (limited to 'app/core/gimpdrawable-gradient.c')
-rw-r--r--app/core/gimpdrawable-gradient.c313
1 files changed, 313 insertions, 0 deletions
diff --git a/app/core/gimpdrawable-gradient.c b/app/core/gimpdrawable-gradient.c
new file mode 100644
index 0000000..2d7cbe8
--- /dev/null
+++ b/app/core/gimpdrawable-gradient.c
@@ -0,0 +1,313 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include <cairo.h>
+#include <gegl.h>
+#include <gdk-pixbuf/gdk-pixbuf.h>
+
+#include "libgimpmath/gimpmath.h"
+
+#include "core-types.h"
+
+#include "gegl/gimp-gegl-apply-operation.h"
+#include "gegl/gimp-gegl-loops.h"
+#include "gegl/gimp-gegl-utils.h"
+
+#include "operations/layer-modes/gimp-layer-modes.h"
+
+#include "gimp.h"
+#include "gimpchannel.h"
+#include "gimpcontext.h"
+#include "gimpdrawable-gradient.h"
+#include "gimpgradient.h"
+#include "gimpimage.h"
+#include "gimpprogress.h"
+
+#include "gimp-intl.h"
+
+
+/* public functions */
+
+void
+gimp_drawable_gradient (GimpDrawable *drawable,
+ GimpContext *context,
+ GimpGradient *gradient,
+ GeglDistanceMetric metric,
+ GimpLayerMode paint_mode,
+ GimpGradientType gradient_type,
+ gdouble opacity,
+ gdouble offset,
+ GimpRepeatMode repeat,
+ gboolean reverse,
+ GimpGradientBlendColorSpace blend_color_space,
+ gboolean supersample,
+ gint max_depth,
+ gdouble threshold,
+ gboolean dither,
+ gdouble startx,
+ gdouble starty,
+ gdouble endx,
+ gdouble endy,
+ GimpProgress *progress)
+{
+ GimpImage *image;
+ GeglBuffer *buffer;
+ GeglBuffer *shapeburst = NULL;
+ GeglNode *render;
+ gint x, y, width, height;
+
+ g_return_if_fail (GIMP_IS_DRAWABLE (drawable));
+ g_return_if_fail (gimp_item_is_attached (GIMP_ITEM (drawable)));
+ g_return_if_fail (GIMP_IS_CONTEXT (context));
+ g_return_if_fail (GIMP_IS_GRADIENT (gradient));
+ g_return_if_fail (progress == NULL || GIMP_IS_PROGRESS (progress));
+
+ image = gimp_item_get_image (GIMP_ITEM (drawable));
+
+ if (! gimp_item_mask_intersect (GIMP_ITEM (drawable), &x, &y, &width, &height))
+ return;
+
+ gimp_set_busy (image->gimp);
+
+ /* Always create an alpha temp buf (for generality) */
+ buffer = gegl_buffer_new (GEGL_RECTANGLE (x, y, width, height),
+ gimp_drawable_get_format_with_alpha (drawable));
+
+ if (gradient_type >= GIMP_GRADIENT_SHAPEBURST_ANGULAR &&
+ gradient_type <= GIMP_GRADIENT_SHAPEBURST_DIMPLED)
+ {
+ shapeburst =
+ gimp_drawable_gradient_shapeburst_distmap (drawable, metric,
+ GEGL_RECTANGLE (x, y, width, height),
+ progress);
+ }
+
+ gimp_drawable_gradient_adjust_coords (drawable,
+ gradient_type,
+ GEGL_RECTANGLE (x, y, width, height),
+ &startx, &starty, &endx, &endy);
+
+ render = gegl_node_new_child (NULL,
+ "operation", "gimp:gradient",
+ "context", context,
+ "gradient", gradient,
+ "start-x", startx,
+ "start-y", starty,
+ "end-x", endx,
+ "end-y", endy,
+ "gradient-type", gradient_type,
+ "gradient-repeat", repeat,
+ "offset", offset,
+ "gradient-reverse", reverse,
+ "gradient-blend-color-space", blend_color_space,
+ "supersample", supersample,
+ "supersample-depth", max_depth,
+ "supersample-threshold", threshold,
+ "dither", dither,
+ NULL);
+
+ gimp_gegl_apply_operation (shapeburst, progress, C_("undo-type", "Gradient"),
+ render,
+ buffer, GEGL_RECTANGLE (x, y, width, height),
+ FALSE);
+
+ g_object_unref (render);
+
+ if (shapeburst)
+ g_object_unref (shapeburst);
+
+ gimp_drawable_apply_buffer (drawable, buffer,
+ GEGL_RECTANGLE (x, y, width, height),
+ TRUE, C_("undo-type", "Gradient"),
+ opacity, paint_mode,
+ GIMP_LAYER_COLOR_SPACE_AUTO,
+ GIMP_LAYER_COLOR_SPACE_AUTO,
+ gimp_layer_mode_get_paint_composite_mode (paint_mode),
+ NULL, x, y);
+
+ gimp_drawable_update (drawable, x, y, width, height);
+
+ g_object_unref (buffer);
+
+ gimp_unset_busy (image->gimp);
+}
+
+GeglBuffer *
+gimp_drawable_gradient_shapeburst_distmap (GimpDrawable *drawable,
+ GeglDistanceMetric metric,
+ const GeglRectangle *region,
+ GimpProgress *progress)
+{
+ GimpChannel *mask;
+ GimpImage *image;
+ GeglBuffer *dist_buffer;
+ GeglBuffer *temp_buffer;
+ GeglNode *shapeburst;
+
+ g_return_val_if_fail (GIMP_IS_DRAWABLE (drawable), NULL);
+ g_return_val_if_fail (gimp_item_is_attached (GIMP_ITEM (drawable)), NULL);
+ g_return_val_if_fail (progress == NULL || GIMP_IS_PROGRESS (progress), NULL);
+
+ image = gimp_item_get_image (GIMP_ITEM (drawable));
+
+ /* allocate the distance map */
+ dist_buffer = gegl_buffer_new (region, babl_format ("Y float"));
+
+ /* allocate the selection mask copy */
+ temp_buffer = gegl_buffer_new (region, babl_format ("Y float"));
+
+ mask = gimp_image_get_mask (image);
+
+ /* If the image mask is not empty, use it as the shape burst source */
+ if (! gimp_channel_is_empty (mask))
+ {
+ gint x, y, width, height;
+ gint off_x, off_y;
+
+ gimp_item_mask_intersect (GIMP_ITEM (drawable), &x, &y, &width, &height);
+ gimp_item_get_offset (GIMP_ITEM (drawable), &off_x, &off_y);
+
+ /* copy the mask to the temp mask */
+ gimp_gegl_buffer_copy (
+ gimp_drawable_get_buffer (GIMP_DRAWABLE (mask)),
+ GEGL_RECTANGLE (x + off_x, y + off_y, width, height),
+ GEGL_ABYSS_NONE, temp_buffer, region);
+ }
+ else
+ {
+ /* If the intended drawable has an alpha channel, use that */
+ if (gimp_drawable_has_alpha (drawable))
+ {
+ const Babl *component_format;
+
+ component_format = babl_format ("A float");
+
+ /* extract the aplha into the temp mask */
+ gegl_buffer_set_format (temp_buffer, component_format);
+ gimp_gegl_buffer_copy (gimp_drawable_get_buffer (drawable), region,
+ GEGL_ABYSS_NONE,
+ temp_buffer, region);
+ gegl_buffer_set_format (temp_buffer, NULL);
+ }
+ else
+ {
+ GeglColor *white = gegl_color_new ("white");
+
+ /* Otherwise, just fill the shapeburst to white */
+ gegl_buffer_set_color (temp_buffer, NULL, white);
+ g_object_unref (white);
+ }
+ }
+
+ shapeburst = gegl_node_new_child (NULL,
+ "operation", "gegl:distance-transform",
+ "normalize", TRUE,
+ "metric", metric,
+ NULL);
+
+ if (progress)
+ gimp_gegl_progress_connect (shapeburst, progress,
+ _("Calculating distance map"));
+
+ gimp_gegl_apply_operation (temp_buffer, NULL, NULL,
+ shapeburst,
+ dist_buffer, region, FALSE);
+
+ g_object_unref (shapeburst);
+
+ g_object_unref (temp_buffer);
+
+ return dist_buffer;
+}
+
+void
+gimp_drawable_gradient_adjust_coords (GimpDrawable *drawable,
+ GimpGradientType gradient_type,
+ const GeglRectangle *region,
+ gdouble *startx,
+ gdouble *starty,
+ gdouble *endx,
+ gdouble *endy)
+{
+ g_return_if_fail (GIMP_IS_DRAWABLE (drawable));
+ g_return_if_fail (region != NULL);
+ g_return_if_fail (startx != NULL);
+ g_return_if_fail (starty != NULL);
+ g_return_if_fail (endx != NULL);
+ g_return_if_fail (endy != NULL);
+
+ /* we potentially adjust the gradient coordinates according to the gradient
+ * type, so that in cases where the gradient span is not related to the
+ * segment length, the gradient cache (in GimpOperationGradient) is big
+ * enough not to produce banding.
+ */
+
+ switch (gradient_type)
+ {
+ /* for conical gradients, use a segment with the original origin and
+ * direction, whose length is the circumference of the largest circle
+ * centered at the origin, passing through one of the regions's vertices.
+ */
+ case GIMP_GRADIENT_CONICAL_SYMMETRIC:
+ case GIMP_GRADIENT_CONICAL_ASYMMETRIC:
+ {
+ gdouble r = 0.0;
+ GimpVector2 v;
+
+ r = MAX (r, hypot (region->x - *startx,
+ region->y - *starty));
+ r = MAX (r, hypot (region->x + region->width - *startx,
+ region->y - *starty));
+ r = MAX (r, hypot (region->x - *startx,
+ region->y + region->height - *starty));
+ r = MAX (r, hypot (region->x + region->width - *startx,
+ region->y + region->height - *starty));
+
+ /* symmetric conical gradients only span half a revolution, and
+ * therefore require only half the cache size.
+ */
+ if (gradient_type == GIMP_GRADIENT_CONICAL_SYMMETRIC)
+ r /= 2.0;
+
+ gimp_vector2_set (&v, *endx - *startx, *endy - *starty);
+ gimp_vector2_normalize (&v);
+ gimp_vector2_mul (&v, 2.0 * G_PI * r);
+
+ *endx = *startx + v.x;
+ *endy = *starty + v.y;
+ }
+ break;
+
+ /* for shaped gradients, only the segment's length matters; use the
+ * regions's diagonal, which is the largest possible distance between two
+ * points in the region.
+ */
+ case GIMP_GRADIENT_SHAPEBURST_ANGULAR:
+ case GIMP_GRADIENT_SHAPEBURST_SPHERICAL:
+ case GIMP_GRADIENT_SHAPEBURST_DIMPLED:
+ *startx = region->x;
+ *starty = region->y;
+ *endx = region->x + region->width;
+ *endy = region->y + region->height;
+ break;
+
+ default:
+ break;
+ }
+}