/* PIKA - Photo and Image Kooker Application * a rebranding of The GNU Image Manipulation Program (created with heckimp) * A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio * * Original copyright, applying to most contents (license remains unchanged): * 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 . */ #include "config.h" #include #include #include #include "libpikamath/pikamath.h" #include "core-types.h" #include "gegl/pika-gegl-apply-operation.h" #include "gegl/pika-gegl-loops.h" #include "gegl/pika-gegl-utils.h" #include "operations/layer-modes/pika-layer-modes.h" #include "pika.h" #include "pikachannel.h" #include "pikacontext.h" #include "pikadrawable-gradient.h" #include "pikagradient.h" #include "pikaimage.h" #include "pikaprogress.h" #include "pika-intl.h" /* public functions */ void pika_drawable_gradient (PikaDrawable *drawable, PikaContext *context, PikaGradient *gradient, GeglDistanceMetric metric, PikaLayerMode paint_mode, PikaGradientType gradient_type, gdouble opacity, gdouble offset, PikaRepeatMode repeat, gboolean reverse, PikaGradientBlendColorSpace blend_color_space, gboolean supersample, gint max_depth, gdouble threshold, gboolean dither, gdouble startx, gdouble starty, gdouble endx, gdouble endy, PikaProgress *progress) { PikaImage *image; GeglBuffer *buffer; GeglBuffer *shapeburst = NULL; GeglNode *render; gint x, y, width, height; g_return_if_fail (PIKA_IS_DRAWABLE (drawable)); g_return_if_fail (pika_item_is_attached (PIKA_ITEM (drawable))); g_return_if_fail (PIKA_IS_CONTEXT (context)); g_return_if_fail (PIKA_IS_GRADIENT (gradient)); g_return_if_fail (progress == NULL || PIKA_IS_PROGRESS (progress)); image = pika_item_get_image (PIKA_ITEM (drawable)); if (! pika_item_mask_intersect (PIKA_ITEM (drawable), &x, &y, &width, &height)) return; pika_set_busy (image->pika); /* Always create an alpha temp buf (for generality) */ buffer = gegl_buffer_new (GEGL_RECTANGLE (x, y, width, height), pika_drawable_get_format_with_alpha (drawable)); if (gradient_type >= PIKA_GRADIENT_SHAPEBURST_ANGULAR && gradient_type <= PIKA_GRADIENT_SHAPEBURST_DIMPLED) { shapeburst = pika_drawable_gradient_shapeburst_distmap (drawable, metric, GEGL_RECTANGLE (x, y, width, height), progress); } pika_drawable_gradient_adjust_coords (drawable, gradient_type, GEGL_RECTANGLE (x, y, width, height), &startx, &starty, &endx, &endy); render = gegl_node_new_child (NULL, "operation", "pika: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); pika_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); pika_drawable_apply_buffer (drawable, buffer, GEGL_RECTANGLE (x, y, width, height), TRUE, C_("undo-type", "Gradient"), opacity, paint_mode, PIKA_LAYER_COLOR_SPACE_AUTO, PIKA_LAYER_COLOR_SPACE_AUTO, pika_layer_mode_get_paint_composite_mode (paint_mode), NULL, x, y); pika_drawable_update (drawable, x, y, width, height); g_object_unref (buffer); pika_unset_busy (image->pika); } GeglBuffer * pika_drawable_gradient_shapeburst_distmap (PikaDrawable *drawable, GeglDistanceMetric metric, const GeglRectangle *region, PikaProgress *progress) { PikaChannel *mask; PikaImage *image; GeglBuffer *dist_buffer; GeglBuffer *temp_buffer; GeglNode *shapeburst; g_return_val_if_fail (PIKA_IS_DRAWABLE (drawable), NULL); g_return_val_if_fail (pika_item_is_attached (PIKA_ITEM (drawable)), NULL); g_return_val_if_fail (progress == NULL || PIKA_IS_PROGRESS (progress), NULL); image = pika_item_get_image (PIKA_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 = pika_image_get_mask (image); /* If the image mask is not empty, use it as the shape burst source */ if (! pika_channel_is_empty (mask)) { gint x, y, width, height; gint off_x, off_y; pika_item_mask_intersect (PIKA_ITEM (drawable), &x, &y, &width, &height); pika_item_get_offset (PIKA_ITEM (drawable), &off_x, &off_y); /* copy the mask to the temp mask */ pika_gegl_buffer_copy ( pika_drawable_get_buffer (PIKA_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 (pika_drawable_has_alpha (drawable)) { const Babl *component_format; component_format = babl_format ("A float"); /* extract the alpha into the temp mask */ gegl_buffer_set_format (temp_buffer, component_format); pika_gegl_buffer_copy (pika_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) pika_gegl_progress_connect (shapeburst, progress, _("Calculating distance map")); pika_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 pika_drawable_gradient_adjust_coords (PikaDrawable *drawable, PikaGradientType gradient_type, const GeglRectangle *region, gdouble *startx, gdouble *starty, gdouble *endx, gdouble *endy) { g_return_if_fail (PIKA_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 PikaOperationGradient) 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 PIKA_GRADIENT_CONICAL_SYMMETRIC: case PIKA_GRADIENT_CONICAL_ASYMMETRIC: { gdouble r = 0.0; PikaVector2 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 == PIKA_GRADIENT_CONICAL_SYMMETRIC) r /= 2.0; pika_vector2_set (&v, *endx - *startx, *endy - *starty); pika_vector2_normalize (&v); pika_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 PIKA_GRADIENT_SHAPEBURST_ANGULAR: case PIKA_GRADIENT_SHAPEBURST_SPHERICAL: case PIKA_GRADIENT_SHAPEBURST_DIMPLED: *startx = region->x; *starty = region->y; *endx = region->x + region->width; *endy = region->y + region->height; break; default: break; } }