mirror of
https://gitlab.gnome.org/GNOME/gimp.git
synced 2025-10-06 01:12:40 +02:00
In layer expansion if user is trying to draw outside the layer boundary with expand option turned on but the "Lock position and size" enabled, the lock square will be blinked. If user is painting on layer mask, the corresponding layer's lock will be blinked. It will be blinked only once per stroke.
574 lines
16 KiB
C
574 lines
16 KiB
C
/* 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 <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpmath/gimpmath.h"
|
|
|
|
#include "tools-types.h"
|
|
|
|
#include "core/gimpdrawable.h"
|
|
#include "core/gimpimage.h"
|
|
#include "core/gimplayer.h"
|
|
#include "core/gimplayermask.h"
|
|
#include "core/gimpprojection.h"
|
|
|
|
#include "paint/gimppaintcore.h"
|
|
#include "paint/gimppaintoptions.h"
|
|
|
|
#include "display/gimpdisplay.h"
|
|
#include "display/gimpdisplayshell.h"
|
|
#include "display/gimpdisplayshell-utils.h"
|
|
|
|
#include "gimppainttool.h"
|
|
#include "gimppainttool-paint.h"
|
|
#include "gimptools-utils.h"
|
|
|
|
|
|
#define DISPLAY_UPDATE_INTERVAL 10000 /* microseconds */
|
|
|
|
|
|
#define PAINT_FINISH NULL
|
|
|
|
|
|
typedef struct
|
|
{
|
|
GimpPaintTool *paint_tool;
|
|
GimpPaintToolPaintFunc func;
|
|
union
|
|
{
|
|
gpointer data;
|
|
gboolean *finished;
|
|
};
|
|
} PaintItem;
|
|
|
|
typedef struct
|
|
{
|
|
GList *drawables;
|
|
GimpCoords coords;
|
|
guint32 time;
|
|
} InterpolateData;
|
|
|
|
|
|
/* local function prototypes */
|
|
|
|
static gboolean gimp_paint_tool_paint_use_thread (GimpPaintTool *paint_tool);
|
|
static gpointer gimp_paint_tool_paint_thread (gpointer data);
|
|
|
|
static gboolean gimp_paint_tool_paint_timeout (GimpPaintTool *paint_tool);
|
|
|
|
static void gimp_paint_tool_paint_interpolate (GimpPaintTool *paint_tool,
|
|
InterpolateData *data);
|
|
|
|
|
|
/* static variables */
|
|
|
|
static GThread *paint_thread;
|
|
|
|
static GMutex paint_mutex;
|
|
static GCond paint_cond;
|
|
|
|
static GQueue paint_queue = G_QUEUE_INIT;
|
|
static GMutex paint_queue_mutex;
|
|
static GCond paint_queue_cond;
|
|
|
|
static guint paint_timeout_id;
|
|
static volatile gboolean paint_timeout_pending;
|
|
|
|
|
|
/* private functions */
|
|
|
|
|
|
static gboolean
|
|
gimp_paint_tool_paint_use_thread (GimpPaintTool *paint_tool)
|
|
{
|
|
if (! paint_tool->draw_line)
|
|
{
|
|
if (! paint_thread)
|
|
{
|
|
static gint use_paint_thread = -1;
|
|
|
|
if (use_paint_thread < 0)
|
|
use_paint_thread = g_getenv ("GIMP_NO_PAINT_THREAD") == NULL;
|
|
|
|
if (use_paint_thread)
|
|
{
|
|
paint_thread = g_thread_new ("paint",
|
|
gimp_paint_tool_paint_thread, NULL);
|
|
}
|
|
}
|
|
|
|
return paint_thread != NULL;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gpointer
|
|
gimp_paint_tool_paint_thread (gpointer data)
|
|
{
|
|
g_mutex_lock (&paint_queue_mutex);
|
|
|
|
while (TRUE)
|
|
{
|
|
PaintItem *item;
|
|
|
|
while (! (item = g_queue_pop_head (&paint_queue)))
|
|
g_cond_wait (&paint_queue_cond, &paint_queue_mutex);
|
|
|
|
if (item->func == PAINT_FINISH)
|
|
{
|
|
*item->finished = TRUE;
|
|
g_cond_signal (&paint_queue_cond);
|
|
}
|
|
else
|
|
{
|
|
g_mutex_unlock (&paint_queue_mutex);
|
|
g_mutex_lock (&paint_mutex);
|
|
|
|
while (paint_timeout_pending)
|
|
g_cond_wait (&paint_cond, &paint_mutex);
|
|
|
|
item->func (item->paint_tool, item->data);
|
|
|
|
g_mutex_unlock (&paint_mutex);
|
|
g_mutex_lock (&paint_queue_mutex);
|
|
}
|
|
|
|
g_slice_free (PaintItem, item);
|
|
}
|
|
|
|
g_mutex_unlock (&paint_queue_mutex);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static gboolean
|
|
gimp_paint_tool_paint_timeout (GimpPaintTool *paint_tool)
|
|
{
|
|
GimpPaintCore *core = paint_tool->core;
|
|
gboolean update = FALSE;
|
|
|
|
paint_timeout_pending = TRUE;
|
|
|
|
g_mutex_lock (&paint_mutex);
|
|
|
|
paint_tool->paint_x = core->last_paint.x;
|
|
paint_tool->paint_y = core->last_paint.y;
|
|
|
|
for (GList *iter = paint_tool->drawables; iter; iter = iter->next)
|
|
{
|
|
update |= gimp_drawable_flush_paint (iter->data);
|
|
if (update)
|
|
break;
|
|
}
|
|
|
|
if (update && GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_flush)
|
|
GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_flush (paint_tool);
|
|
|
|
paint_timeout_pending = FALSE;
|
|
g_cond_signal (&paint_cond);
|
|
|
|
g_mutex_unlock (&paint_mutex);
|
|
|
|
if (update)
|
|
{
|
|
GimpDrawTool *draw_tool = GIMP_DRAW_TOOL (paint_tool);
|
|
GimpDisplay *display = paint_tool->display;
|
|
GimpImage *image = gimp_display_get_image (display);
|
|
|
|
if (paint_tool->snap_brush)
|
|
gimp_draw_tool_pause (draw_tool);
|
|
|
|
gimp_projection_flush_now (gimp_image_get_projection (image), TRUE);
|
|
gimp_display_flush_now (display);
|
|
|
|
if (paint_tool->snap_brush)
|
|
gimp_draw_tool_resume (draw_tool);
|
|
}
|
|
|
|
return G_SOURCE_CONTINUE;
|
|
}
|
|
|
|
static void
|
|
gimp_paint_tool_paint_interpolate (GimpPaintTool *paint_tool,
|
|
InterpolateData *data)
|
|
{
|
|
GimpPaintOptions *paint_options = GIMP_PAINT_TOOL_GET_OPTIONS (paint_tool);
|
|
GimpPaintCore *core = paint_tool->core;
|
|
|
|
gimp_paint_core_interpolate (core, data->drawables, paint_options,
|
|
&data->coords, data->time);
|
|
|
|
/* Blink the lock box if required */
|
|
if (core->lock_blink_state == GIMP_PAINT_LOCK_BLINK_PENDING)
|
|
{
|
|
GList *iter;
|
|
GimpLayer *layer;
|
|
|
|
/* Blink the lock only once per stroke */
|
|
core->lock_blink_state = GIMP_PAINT_LOCK_BLINKED;
|
|
|
|
for (iter = data->drawables; iter; iter = g_list_next (iter))
|
|
{
|
|
layer = GIMP_IS_LAYER_MASK (iter->data) ?
|
|
GIMP_LAYER_MASK (iter->data)->layer :
|
|
GIMP_LAYER (iter->data);
|
|
|
|
if (gimp_item_get_lock_position (GIMP_ITEM (layer)))
|
|
gimp_tools_blink_lock_box (GIMP_CONTEXT (paint_options)->gimp,
|
|
GIMP_ITEM (layer));
|
|
}
|
|
}
|
|
|
|
g_list_free (data->drawables);
|
|
g_slice_free (InterpolateData, data);
|
|
}
|
|
|
|
|
|
/* public functions */
|
|
|
|
|
|
gboolean
|
|
gimp_paint_tool_paint_start (GimpPaintTool *paint_tool,
|
|
GimpDisplay *display,
|
|
const GimpCoords *coords,
|
|
guint32 time,
|
|
gboolean constrain,
|
|
GError **error)
|
|
{
|
|
GimpTool *tool;
|
|
GimpPaintOptions *paint_options;
|
|
GimpPaintCore *core;
|
|
GimpDisplayShell *shell;
|
|
GimpImage *image;
|
|
GList *drawables;
|
|
GList *iter;
|
|
GimpCoords curr_coords;
|
|
|
|
g_return_val_if_fail (GIMP_IS_PAINT_TOOL (paint_tool), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_DISPLAY (display), FALSE);
|
|
g_return_val_if_fail (coords != NULL, FALSE);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
g_return_val_if_fail (paint_tool->display == NULL, FALSE);
|
|
|
|
tool = GIMP_TOOL (paint_tool);
|
|
paint_tool = GIMP_PAINT_TOOL (paint_tool);
|
|
paint_options = GIMP_PAINT_TOOL_GET_OPTIONS (paint_tool);
|
|
core = paint_tool->core;
|
|
shell = gimp_display_get_shell (display);
|
|
image = gimp_display_get_image (display);
|
|
drawables = gimp_image_get_selected_drawables (image);
|
|
|
|
g_return_val_if_fail (g_list_length (drawables) == 1 ||
|
|
(g_list_length (drawables) > 1 && paint_tool->can_multi_paint),
|
|
FALSE);
|
|
|
|
curr_coords = *coords;
|
|
|
|
paint_tool->paint_x = curr_coords.x;
|
|
paint_tool->paint_y = curr_coords.y;
|
|
|
|
/* If we use a separate paint thread, enter paint mode before starting the
|
|
* paint core
|
|
*/
|
|
if (gimp_paint_tool_paint_use_thread (paint_tool))
|
|
for (iter = drawables; iter; iter = iter->next)
|
|
gimp_drawable_start_paint (iter->data);
|
|
|
|
/* Prepare to start the paint core */
|
|
if (GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_prepare)
|
|
GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_prepare (paint_tool, display);
|
|
|
|
/* Start the paint core */
|
|
if (! gimp_paint_core_start (core,
|
|
drawables, paint_options, &curr_coords,
|
|
error))
|
|
{
|
|
for (iter = drawables; iter; iter = iter->next)
|
|
gimp_drawable_end_paint (iter->data);
|
|
g_list_free (drawables);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
paint_tool->display = display;
|
|
g_list_free (paint_tool->drawables);
|
|
paint_tool->drawables = drawables;
|
|
|
|
if ((display != tool->display) || ! paint_tool->draw_line)
|
|
{
|
|
/* If this is a new display, reset the "last stroke's endpoint"
|
|
* because there is none
|
|
*/
|
|
if (display != tool->display)
|
|
core->start_coords = core->cur_coords;
|
|
|
|
core->last_coords = core->cur_coords;
|
|
|
|
core->distance = 0.0;
|
|
core->pixel_dist = 0.0;
|
|
}
|
|
else if (paint_tool->draw_line)
|
|
{
|
|
gdouble offset_angle;
|
|
gdouble xres, yres;
|
|
|
|
gimp_display_shell_get_constrained_line_params (shell,
|
|
&offset_angle,
|
|
&xres, &yres);
|
|
|
|
/* If shift is down and this is not the first paint
|
|
* stroke, then draw a line from the last coords to the pointer
|
|
*/
|
|
gimp_paint_core_round_line (core, paint_options,
|
|
constrain, offset_angle, xres, yres);
|
|
}
|
|
|
|
/* Notify subclasses */
|
|
if (gimp_paint_tool_paint_use_thread (paint_tool) &&
|
|
GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_start)
|
|
{
|
|
GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_start (paint_tool);
|
|
}
|
|
|
|
/* Let the specific painting function initialize itself */
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_INIT, time);
|
|
|
|
/* Paint to the image */
|
|
if (paint_tool->draw_line)
|
|
{
|
|
gimp_paint_core_interpolate (core, drawables, paint_options,
|
|
&core->cur_coords, time);
|
|
}
|
|
else
|
|
{
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_MOTION, time);
|
|
}
|
|
|
|
gimp_projection_flush_now (gimp_image_get_projection (image), TRUE);
|
|
gimp_display_flush_now (display);
|
|
|
|
/* Start the display update timeout */
|
|
if (gimp_paint_tool_paint_use_thread (paint_tool))
|
|
{
|
|
paint_timeout_id = g_timeout_add_full (
|
|
G_PRIORITY_HIGH_IDLE,
|
|
DISPLAY_UPDATE_INTERVAL / 1000,
|
|
(GSourceFunc) gimp_paint_tool_paint_timeout,
|
|
paint_tool, NULL);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_paint_tool_paint_end (GimpPaintTool *paint_tool,
|
|
guint32 time,
|
|
gboolean cancel)
|
|
{
|
|
GimpPaintOptions *paint_options;
|
|
GimpPaintCore *core;
|
|
GList *drawables;
|
|
|
|
g_return_if_fail (GIMP_IS_PAINT_TOOL (paint_tool));
|
|
g_return_if_fail (paint_tool->display != NULL);
|
|
|
|
paint_options = GIMP_PAINT_TOOL_GET_OPTIONS (paint_tool);
|
|
core = paint_tool->core;
|
|
drawables = paint_tool->drawables;
|
|
|
|
/* Process remaining paint items */
|
|
if (gimp_paint_tool_paint_use_thread (paint_tool))
|
|
{
|
|
PaintItem *item;
|
|
gboolean finished = FALSE;
|
|
guint64 end_time;
|
|
|
|
g_return_if_fail (gimp_paint_tool_paint_is_active (paint_tool));
|
|
|
|
g_source_remove (paint_timeout_id);
|
|
paint_timeout_id = 0;
|
|
|
|
item = g_slice_new (PaintItem);
|
|
|
|
item->paint_tool = paint_tool;
|
|
item->func = PAINT_FINISH;
|
|
item->finished = &finished;
|
|
|
|
g_mutex_lock (&paint_queue_mutex);
|
|
|
|
g_queue_push_tail (&paint_queue, item);
|
|
g_cond_signal (&paint_queue_cond);
|
|
|
|
end_time = g_get_monotonic_time () + DISPLAY_UPDATE_INTERVAL;
|
|
|
|
while (! finished)
|
|
{
|
|
if (! g_cond_wait_until (&paint_queue_cond, &paint_queue_mutex,
|
|
end_time))
|
|
{
|
|
g_mutex_unlock (&paint_queue_mutex);
|
|
|
|
gimp_paint_tool_paint_timeout (paint_tool);
|
|
|
|
g_mutex_lock (&paint_queue_mutex);
|
|
|
|
end_time = g_get_monotonic_time () + DISPLAY_UPDATE_INTERVAL;
|
|
}
|
|
}
|
|
|
|
g_mutex_unlock (&paint_queue_mutex);
|
|
}
|
|
|
|
/* Let the specific painting function finish up */
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_FINISH, time);
|
|
|
|
if (cancel)
|
|
gimp_paint_core_cancel (core, drawables);
|
|
else
|
|
gimp_paint_core_finish (core, drawables, TRUE);
|
|
|
|
/* Notify subclasses */
|
|
if (gimp_paint_tool_paint_use_thread (paint_tool) &&
|
|
GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_end)
|
|
{
|
|
GIMP_PAINT_TOOL_GET_CLASS (paint_tool)->paint_end (paint_tool);
|
|
}
|
|
|
|
/* Exit paint mode */
|
|
if (gimp_paint_tool_paint_use_thread (paint_tool))
|
|
for (GList *iter = drawables; iter; iter = iter->next)
|
|
gimp_drawable_end_paint (iter->data);
|
|
|
|
paint_tool->display = NULL;
|
|
g_clear_pointer (&paint_tool->drawables, g_list_free);
|
|
}
|
|
|
|
gboolean
|
|
gimp_paint_tool_paint_is_active (GimpPaintTool *paint_tool)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_PAINT_TOOL (paint_tool), FALSE);
|
|
|
|
for (GList *iter = paint_tool->drawables; iter; iter = iter->next)
|
|
{
|
|
if (gimp_drawable_is_painting (iter->data))
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
gimp_paint_tool_paint_push (GimpPaintTool *paint_tool,
|
|
GimpPaintToolPaintFunc func,
|
|
gpointer data)
|
|
{
|
|
g_return_if_fail (GIMP_IS_PAINT_TOOL (paint_tool));
|
|
g_return_if_fail (func != NULL);
|
|
|
|
if (gimp_paint_tool_paint_use_thread (paint_tool))
|
|
{
|
|
PaintItem *item;
|
|
|
|
g_return_if_fail (gimp_paint_tool_paint_is_active (paint_tool));
|
|
|
|
/* Push an item to the queue, to be processed by the paint thread */
|
|
|
|
item = g_slice_new (PaintItem);
|
|
|
|
item->paint_tool = paint_tool;
|
|
item->func = func;
|
|
item->data = data;
|
|
|
|
g_mutex_lock (&paint_queue_mutex);
|
|
|
|
g_queue_push_tail (&paint_queue, item);
|
|
g_cond_signal (&paint_queue_cond);
|
|
|
|
g_mutex_unlock (&paint_queue_mutex);
|
|
}
|
|
else
|
|
{
|
|
GimpDrawTool *draw_tool = GIMP_DRAW_TOOL (paint_tool);
|
|
GimpDisplay *display = paint_tool->display;
|
|
GimpImage *image = gimp_display_get_image (display);
|
|
|
|
/* Paint directly */
|
|
|
|
gimp_draw_tool_pause (draw_tool);
|
|
|
|
func (paint_tool, data);
|
|
|
|
gimp_projection_flush_now (gimp_image_get_projection (image), TRUE);
|
|
gimp_display_flush_now (display);
|
|
|
|
gimp_draw_tool_resume (draw_tool);
|
|
}
|
|
}
|
|
|
|
void
|
|
gimp_paint_tool_paint_motion (GimpPaintTool *paint_tool,
|
|
const GimpCoords *coords,
|
|
guint32 time)
|
|
{
|
|
GimpPaintOptions *paint_options;
|
|
GimpPaintCore *core;
|
|
GList *drawables;
|
|
InterpolateData *data;
|
|
|
|
g_return_if_fail (GIMP_IS_PAINT_TOOL (paint_tool));
|
|
g_return_if_fail (coords != NULL);
|
|
g_return_if_fail (paint_tool->display != NULL);
|
|
|
|
paint_options = GIMP_PAINT_TOOL_GET_OPTIONS (paint_tool);
|
|
core = paint_tool->core;
|
|
drawables = paint_tool->drawables;
|
|
|
|
data = g_slice_new (InterpolateData);
|
|
|
|
data->drawables = g_list_copy (drawables);
|
|
data->coords = *coords;
|
|
data->time = time;
|
|
|
|
paint_tool->cursor_x = data->coords.x;
|
|
paint_tool->cursor_y = data->coords.y;
|
|
|
|
gimp_paint_core_smooth_coords (core, paint_options, &data->coords);
|
|
|
|
/* Don't paint while the Shift key is pressed for line drawing */
|
|
if (paint_tool->draw_line)
|
|
{
|
|
gimp_paint_core_set_current_coords (core, &data->coords);
|
|
g_list_free (data->drawables);
|
|
|
|
g_slice_free (InterpolateData, data);
|
|
|
|
return;
|
|
}
|
|
|
|
gimp_paint_tool_paint_push (paint_tool,
|
|
(GimpPaintToolPaintFunc) gimp_paint_tool_paint_interpolate,
|
|
data);
|
|
}
|