/* 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 "libpikamath/pikamath.h" #include "paint-types.h" #include "gegl/pika-gegl-loops.h" #include "gegl/pika-gegl-utils.h" #include "core/pika-palettes.h" #include "core/pikabrush.h" #include "core/pikabrush-header.h" #include "core/pikadrawable.h" #include "core/pikadynamics.h" #include "core/pikaimage.h" #include "core/pikapickable.h" #include "core/pikasymmetry.h" #include "core/pikatempbuf.h" #include "pikasmudge.h" #include "pikasmudgeoptions.h" #include "pika-intl.h" static void pika_smudge_finalize (GObject *object); static void pika_smudge_paint (PikaPaintCore *paint_core, GList *drawables, PikaPaintOptions *paint_options, PikaSymmetry *sym, PikaPaintState paint_state, guint32 time); static gboolean pika_smudge_start (PikaPaintCore *paint_core, PikaDrawable *drawable, PikaPaintOptions *paint_options, PikaSymmetry *sym); static void pika_smudge_motion (PikaPaintCore *paint_core, PikaDrawable *drawable, PikaPaintOptions *paint_options, PikaSymmetry *sym); static void pika_smudge_accumulator_coords (PikaPaintCore *paint_core, const PikaCoords *coords, gint stroke, gint *x, gint *y); static void pika_smudge_accumulator_size (PikaPaintOptions *paint_options, const PikaCoords *coords, gint *accumulator_size); G_DEFINE_TYPE (PikaSmudge, pika_smudge, PIKA_TYPE_BRUSH_CORE) #define parent_class pika_smudge_parent_class void pika_smudge_register (Pika *pika, PikaPaintRegisterCallback callback) { (* callback) (pika, PIKA_TYPE_SMUDGE, PIKA_TYPE_SMUDGE_OPTIONS, "pika-smudge", _("Smudge"), "pika-tool-smudge"); } static void pika_smudge_class_init (PikaSmudgeClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); PikaPaintCoreClass *paint_core_class = PIKA_PAINT_CORE_CLASS (klass); PikaBrushCoreClass *brush_core_class = PIKA_BRUSH_CORE_CLASS (klass); object_class->finalize = pika_smudge_finalize; paint_core_class->paint = pika_smudge_paint; brush_core_class->handles_changing_brush = TRUE; brush_core_class->handles_transforming_brush = TRUE; brush_core_class->handles_dynamic_transforming_brush = TRUE; } static void pika_smudge_init (PikaSmudge *smudge) { } static void pika_smudge_finalize (GObject *object) { PikaSmudge *smudge = PIKA_SMUDGE (object); if (smudge->accum_buffers) { GList *iter; for (iter = smudge->accum_buffers; iter; iter = g_list_next (iter)) { if (iter->data) g_object_unref (iter->data); } g_list_free (smudge->accum_buffers); smudge->accum_buffers = NULL; } G_OBJECT_CLASS (parent_class)->finalize (object); } static void pika_smudge_paint (PikaPaintCore *paint_core, GList *drawables, PikaPaintOptions *paint_options, PikaSymmetry *sym, PikaPaintState paint_state, guint32 time) { PikaSmudge *smudge = PIKA_SMUDGE (paint_core); g_return_if_fail (g_list_length (drawables) == 1); switch (paint_state) { case PIKA_PAINT_STATE_INIT: { PikaContext *context = PIKA_CONTEXT (paint_options); PikaSmudgeOptions *options = PIKA_SMUDGE_OPTIONS (paint_options); PikaBrushCore *brush_core = PIKA_BRUSH_CORE (paint_core); PikaDynamics *dynamics = pika_context_get_dynamics (context); /* Don't add to color history when * 1. pure smudging (flow=0) * 2. color is from gradient or pixmap brushes */ if (options->flow > 0.0 && ! pika_dynamics_is_output_enabled (dynamics, PIKA_DYNAMICS_OUTPUT_COLOR) && ! (brush_core->brush && pika_brush_get_pixmap (brush_core->brush))) { PikaRGB foreground; pika_context_get_foreground (context, &foreground); pika_palettes_add_color_history (context->pika, &foreground); } } break; case PIKA_PAINT_STATE_MOTION: /* initialization fails if the user starts outside the drawable */ if (! smudge->initialized) smudge->initialized = pika_smudge_start (paint_core, drawables->data, paint_options, sym); if (smudge->initialized) pika_smudge_motion (paint_core, drawables->data, paint_options, sym); break; case PIKA_PAINT_STATE_FINISH: if (smudge->accum_buffers) { GList *iter; for (iter = smudge->accum_buffers; iter; iter = g_list_next (iter)) { if (iter->data) g_object_unref (iter->data); } g_list_free (smudge->accum_buffers); smudge->accum_buffers = NULL; } smudge->initialized = FALSE; break; default: break; } } static gboolean pika_smudge_start (PikaPaintCore *paint_core, PikaDrawable *drawable, PikaPaintOptions *paint_options, PikaSymmetry *sym) { PikaSmudge *smudge = PIKA_SMUDGE (paint_core); PikaBrushCore *brush_core = PIKA_BRUSH_CORE (paint_core); PikaSmudgeOptions *options = PIKA_SMUDGE_OPTIONS (paint_options); PikaPickable *dest_pickable; GeglBuffer *pickable_buffer; GeglBuffer *paint_buffer; PikaCoords coords; gint dest_pickable_off_x; gint dest_pickable_off_y; gint paint_buffer_x; gint paint_buffer_y; gint accum_size; gint n_strokes; gint i; gint x, y; gint off_x, off_y; coords = *(pika_symmetry_get_origin (sym)); pika_item_get_offset (PIKA_ITEM (drawable), &off_x, &off_y); coords.x -= off_x; coords.y -= off_y; pika_brush_core_eval_transform_dynamics (brush_core, pika_item_get_image (PIKA_ITEM (drawable)), paint_options, &coords); if (options->sample_merged) { dest_pickable = pika_paint_core_get_image_pickable (paint_core); pika_item_get_offset (PIKA_ITEM (drawable), &dest_pickable_off_x, &dest_pickable_off_y); } else { dest_pickable = PIKA_PICKABLE (drawable); dest_pickable_off_x = 0; dest_pickable_off_y = 0; } pickable_buffer = pika_pickable_get_buffer (dest_pickable); n_strokes = pika_symmetry_get_size (sym); for (i = 0; i < n_strokes; i++) { GeglBuffer *accum_buffer; coords = *(pika_symmetry_get_coords (sym, i)); coords.x -= off_x; coords.y -= off_y; pika_smudge_accumulator_size (paint_options, &coords, &accum_size); /* Allocate the accumulation buffer */ accum_buffer = gegl_buffer_new (GEGL_RECTANGLE (0, 0, accum_size, accum_size), babl_format ("RGBA float")); smudge->accum_buffers = g_list_prepend (smudge->accum_buffers, accum_buffer); /* adjust the x and y coordinates to the upper left corner of the * accumulator */ paint_buffer = pika_paint_core_get_paint_buffer (paint_core, drawable, paint_options, PIKA_LAYER_MODE_NORMAL, &coords, &paint_buffer_x, &paint_buffer_y, NULL, NULL); if (! paint_buffer) continue; pika_smudge_accumulator_coords (paint_core, &coords, 0, &x, &y); /* If clipped, prefill the smudge buffer with the color at the * brush position. */ if (x != paint_buffer_x || y != paint_buffer_y || accum_size != gegl_buffer_get_width (paint_buffer) || accum_size != gegl_buffer_get_height (paint_buffer)) { gfloat pixel[4]; GeglColor *color; gint pick_x; gint pick_y; pick_x = CLAMP ((gint) coords.x + dest_pickable_off_x, 0, gegl_buffer_get_width (pickable_buffer) - 1); pick_y = CLAMP ((gint) coords.y + dest_pickable_off_y, 0, gegl_buffer_get_height (pickable_buffer) - 1); pika_pickable_get_pixel_at (dest_pickable, pick_x, pick_y, babl_format ("RGBA float"), pixel); color = gegl_color_new (NULL); gegl_color_set_pixel (color, babl_format ("RGBA float"), pixel); gegl_buffer_set_color (accum_buffer, NULL, color); g_object_unref (color); } /* copy the region under the original painthit. */ pika_gegl_buffer_copy (pickable_buffer, GEGL_RECTANGLE (paint_buffer_x + dest_pickable_off_x, paint_buffer_y + dest_pickable_off_y, gegl_buffer_get_width (paint_buffer), gegl_buffer_get_height (paint_buffer)), GEGL_ABYSS_NONE, accum_buffer, GEGL_RECTANGLE (paint_buffer_x - x, paint_buffer_y - y, 0, 0)); } smudge->accum_buffers = g_list_reverse (smudge->accum_buffers); return TRUE; } static void pika_smudge_motion (PikaPaintCore *paint_core, PikaDrawable *drawable, PikaPaintOptions *paint_options, PikaSymmetry *sym) { PikaSmudge *smudge = PIKA_SMUDGE (paint_core); PikaBrushCore *brush_core = PIKA_BRUSH_CORE (paint_core); PikaSmudgeOptions *options = PIKA_SMUDGE_OPTIONS (paint_options); PikaContext *context = PIKA_CONTEXT (paint_options); PikaDynamics *dynamics = PIKA_BRUSH_CORE (paint_core)->dynamics; PikaImage *image = pika_item_get_image (PIKA_ITEM (drawable)); PikaPickable *dest_pickable; GeglBuffer *paint_buffer; gint dest_pickable_off_x; gint dest_pickable_off_y; gint paint_buffer_x; gint paint_buffer_y; gint paint_buffer_width; gint paint_buffer_height; /* brush dynamics */ gdouble fade_point; gdouble opacity; gdouble rate; gdouble flow; gdouble grad_point; /* brush color */ PikaRGB brush_color; PikaRGB *brush_color_ptr; /* whether use single color or pixmap */ /* accum buffer */ gint x, y; GeglBuffer *accum_buffer; /* other variables */ gdouble force; PikaCoords coords; gint off_x, off_y; gint paint_width, paint_height; gint n_strokes; gint i; if (options->sample_merged) { dest_pickable = pika_paint_core_get_image_pickable (paint_core); pika_item_get_offset (PIKA_ITEM (drawable), &dest_pickable_off_x, &dest_pickable_off_y); } else { dest_pickable = PIKA_PICKABLE (drawable); dest_pickable_off_x = 0; dest_pickable_off_y = 0; } fade_point = pika_paint_options_get_fade (paint_options, image, paint_core->pixel_dist); pika_item_get_offset (PIKA_ITEM (drawable), &off_x, &off_y); coords = *(pika_symmetry_get_origin (sym)); coords.x -= off_x; coords.y -= off_y; pika_symmetry_set_origin (sym, drawable, &coords); paint_core->sym = sym; opacity = pika_dynamics_get_linear_value (dynamics, PIKA_DYNAMICS_OUTPUT_OPACITY, &coords, paint_options, fade_point); if (opacity == 0.0) return; pika_brush_core_eval_transform_dynamics (brush_core, image, paint_options, &coords); /* Get brush dynamic values other than opacity */ rate = ((options->rate / 100.0) * pika_dynamics_get_linear_value (dynamics, PIKA_DYNAMICS_OUTPUT_RATE, &coords, paint_options, fade_point)); flow = ((options->flow / 100.0) * pika_dynamics_get_linear_value (dynamics, PIKA_DYNAMICS_OUTPUT_FLOW, &coords, paint_options, fade_point)); grad_point = pika_dynamics_get_linear_value (dynamics, PIKA_DYNAMICS_OUTPUT_COLOR, &coords, paint_options, fade_point); /* Get current gradient color, brush pixmap, or foreground color */ brush_color_ptr = &brush_color; if (pika_paint_options_get_gradient_color (paint_options, image, grad_point, paint_core->pixel_dist, &brush_color)) { /* No more processing needed */ } else if (brush_core->brush && pika_brush_get_pixmap (brush_core->brush)) { brush_color_ptr = NULL; } else { pika_context_get_foreground (context, &brush_color); } /* Convert to linear RGBA */ if (brush_color_ptr) pika_pickable_rgb_to_pixel (dest_pickable, &brush_color, babl_format ("RGBA double"), &brush_color); n_strokes = pika_symmetry_get_size (sym); for (i = 0; i < n_strokes; i++) { coords = *(pika_symmetry_get_coords (sym, i)); pika_brush_core_eval_transform_symmetry (brush_core, sym, i); paint_buffer = pika_paint_core_get_paint_buffer (paint_core, drawable, paint_options, PIKA_LAYER_MODE_NORMAL, &coords, &paint_buffer_x, &paint_buffer_y, &paint_width, &paint_height); if (! paint_buffer) continue; paint_buffer_width = gegl_buffer_get_width (paint_buffer); paint_buffer_height = gegl_buffer_get_height (paint_buffer); /* Get the unclipped accumulator coordinates */ pika_smudge_accumulator_coords (paint_core, &coords, i, &x, &y); accum_buffer = g_list_nth_data (smudge->accum_buffers, i); /* Old smudge tool: * Smudge uses the buffer Accum. * For each successive painthit Accum is built like this * Accum = rate*Accum + (1-rate)*I. * where I is the pixels under the current painthit. * Then the paint area (paint_area) is built as * (Accum,1) (if no alpha), */ /* 2017/4/22: New smudge painting tool: * Accum=rate*Accum + (1-rate)*I * if brush_color_ptr!=NULL * Paint=(1-flow)*Accum + flow*BrushColor * else, draw brush pixmap on the paint_buffer and * Paint=(1-flow)*Accum + flow*Paint * * For non-pixmap brushes, calculate blending in * pika_gegl_smudge_with_paint() instead of calling * gegl_buffer_set_color() to reduce gegl's internal processing. */ if (! brush_color_ptr && flow > 0.0) { pika_brush_core_color_area_with_pixmap (brush_core, drawable, &coords, paint_buffer, paint_buffer_x, paint_buffer_y, TRUE); } pika_gegl_smudge_with_paint (accum_buffer, GEGL_RECTANGLE (paint_buffer_x - x, paint_buffer_y - y, paint_buffer_width, paint_buffer_height), pika_pickable_get_buffer (dest_pickable), GEGL_RECTANGLE (paint_buffer_x + dest_pickable_off_x, paint_buffer_y + dest_pickable_off_y, paint_buffer_width, paint_buffer_height), brush_color_ptr, paint_buffer, options->no_erasing, flow, rate); if (pika_dynamics_is_output_enabled (dynamics, PIKA_DYNAMICS_OUTPUT_FORCE)) force = pika_dynamics_get_linear_value (dynamics, PIKA_DYNAMICS_OUTPUT_FORCE, &coords, paint_options, fade_point); else force = paint_options->brush_force; pika_brush_core_replace_canvas (PIKA_BRUSH_CORE (paint_core), drawable, &coords, MIN (opacity, PIKA_OPACITY_OPAQUE), pika_context_get_opacity (context), pika_paint_options_get_brush_mode (paint_options), force, PIKA_PAINT_INCREMENTAL); } } static void pika_smudge_accumulator_coords (PikaPaintCore *paint_core, const PikaCoords *coords, gint stroke, gint *x, gint *y) { PikaSmudge *smudge = PIKA_SMUDGE (paint_core); GeglBuffer *accum_buffer; accum_buffer = g_list_nth_data (smudge->accum_buffers, stroke); *x = (gint) coords->x - gegl_buffer_get_width (accum_buffer) / 2; *y = (gint) coords->y - gegl_buffer_get_height (accum_buffer) / 2; } static void pika_smudge_accumulator_size (PikaPaintOptions *paint_options, const PikaCoords *coords, gint *accumulator_size) { gdouble max_view_scale = 1.0; gdouble max_brush_size; if (paint_options->brush_lock_to_view) max_view_scale = MAX (coords->xscale, coords->yscale); max_brush_size = MIN (paint_options->brush_size / max_view_scale, PIKA_BRUSH_MAX_SIZE); /* Note: the max brush mask size plus a border of 1 pixel and a * little headroom */ *accumulator_size = ceil (sqrt (2 * SQR (max_brush_size + 1)) + 2); }