I tried to have a not too overwhelming API, so we just ask for the label and initial color at construction. We keep sane defaults for the rest and let people tweak the result by getting the color area widgets themselves (if they need to force-showing flat colors or change the drag buttons in particular). Another thing I wondered about was the initial size of the color area. Without a size request or being in some container expanding its children (which may also be ugly), it ends up too small. I can imagine such widget being used especially when you want to display several color rectangles next to each other with a label each. So I just set it this way. Anyone is free to request a resize after constructing the object. Last but not least, the position of the label was especially of interest here. For my idea of a list of colors, I could definitely imagine color blocks aligned with vertically-oriented labels above or below. It might be worth adding an API for this later on.
1040 lines
32 KiB
C
1040 lines
32 KiB
C
/* LIBGIMP - The GIMP Library
|
|
* Copyright (C) 1995-1997 Peter Mattis and Spencer Kimball
|
|
*
|
|
* gimpcolorarea.c
|
|
* Copyright (C) 2001-2002 Sven Neumann <sven@gimp.org>
|
|
*
|
|
* This library is free software: you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 3 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see
|
|
* <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpbase/gimpbase.h"
|
|
#include "libgimpcolor/gimpcolor.h"
|
|
#include "libgimpconfig/gimpconfig.h"
|
|
|
|
#include "gimpwidgetstypes.h"
|
|
|
|
#include "gimpcairo-utils.h"
|
|
#include "gimpcolorarea.h"
|
|
#include "gimpwidgetsutils.h"
|
|
#include "gimpwidgets-private.h"
|
|
|
|
|
|
/**
|
|
* SECTION: gimpcolorarea
|
|
* @title: GimpColorArea
|
|
* @short_description: Displays a #GimpRGB color, optionally with
|
|
* alpha-channel.
|
|
*
|
|
* Displays a #GimpRGB color, optionally with alpha-channel.
|
|
**/
|
|
|
|
|
|
#define DRAG_PREVIEW_SIZE 32
|
|
#define DRAG_ICON_OFFSET -8
|
|
|
|
|
|
enum
|
|
{
|
|
COLOR_CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_COLOR,
|
|
PROP_TYPE,
|
|
PROP_DRAG_MASK,
|
|
PROP_DRAW_BORDER
|
|
};
|
|
|
|
|
|
struct _GimpColorAreaPrivate
|
|
{
|
|
GimpColorConfig *config;
|
|
GimpColorTransform *transform;
|
|
|
|
guchar *buf;
|
|
guint width;
|
|
guint height;
|
|
guint rowstride;
|
|
|
|
GimpColorAreaType type;
|
|
GimpRGB color;
|
|
guint draw_border : 1;
|
|
guint needs_render : 1;
|
|
|
|
gboolean out_of_gamut;
|
|
};
|
|
|
|
#define GET_PRIVATE(obj) (((GimpColorArea *) (obj))->priv)
|
|
|
|
|
|
static void gimp_color_area_dispose (GObject *object);
|
|
static void gimp_color_area_finalize (GObject *object);
|
|
static void gimp_color_area_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
static void gimp_color_area_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
|
|
static void gimp_color_area_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation);
|
|
static gboolean gimp_color_area_draw (GtkWidget *widget,
|
|
cairo_t *cr);
|
|
static void gimp_color_area_render_buf (GtkWidget *widget,
|
|
GimpColorAreaType type,
|
|
guchar *buf,
|
|
guint width,
|
|
guint height,
|
|
guint rowstride,
|
|
GimpRGB *color);
|
|
static void gimp_color_area_render (GimpColorArea *area);
|
|
|
|
static void gimp_color_area_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context);
|
|
static void gimp_color_area_drag_end (GtkWidget *widget,
|
|
GdkDragContext *context);
|
|
static void gimp_color_area_drag_data_received (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
GtkSelectionData *selection_data,
|
|
guint info,
|
|
guint time);
|
|
static void gimp_color_area_drag_data_get (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
GtkSelectionData *selection_data,
|
|
guint info,
|
|
guint time);
|
|
|
|
static void gimp_color_area_create_transform (GimpColorArea *area);
|
|
static void gimp_color_area_destroy_transform (GimpColorArea *area);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (GimpColorArea, gimp_color_area,
|
|
GTK_TYPE_DRAWING_AREA)
|
|
|
|
#define parent_class gimp_color_area_parent_class
|
|
|
|
static guint gimp_color_area_signals[LAST_SIGNAL] = { 0 };
|
|
|
|
static const GtkTargetEntry target = { "application/x-color", 0 };
|
|
|
|
|
|
static void
|
|
gimp_color_area_class_init (GimpColorAreaClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
GimpRGB color;
|
|
|
|
gimp_color_area_signals[COLOR_CHANGED] =
|
|
g_signal_new ("color-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (GimpColorAreaClass, color_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
object_class->dispose = gimp_color_area_dispose;
|
|
object_class->finalize = gimp_color_area_finalize;
|
|
object_class->get_property = gimp_color_area_get_property;
|
|
object_class->set_property = gimp_color_area_set_property;
|
|
|
|
widget_class->size_allocate = gimp_color_area_size_allocate;
|
|
widget_class->draw = gimp_color_area_draw;
|
|
|
|
widget_class->drag_begin = gimp_color_area_drag_begin;
|
|
widget_class->drag_end = gimp_color_area_drag_end;
|
|
widget_class->drag_data_received = gimp_color_area_drag_data_received;
|
|
widget_class->drag_data_get = gimp_color_area_drag_data_get;
|
|
|
|
klass->color_changed = NULL;
|
|
|
|
gimp_rgba_set (&color, 0.0, 0.0, 0.0, 1.0);
|
|
|
|
/**
|
|
* GimpColorArea:color:
|
|
*
|
|
* The color displayed in the color area.
|
|
*
|
|
* Since: 2.4
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_COLOR,
|
|
gimp_param_spec_rgb ("color",
|
|
"Color",
|
|
"The displayed color",
|
|
TRUE, &color,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT));
|
|
/**
|
|
* GimpColorArea:type:
|
|
*
|
|
* The type of the color area.
|
|
*
|
|
* Since: 2.4
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_TYPE,
|
|
g_param_spec_enum ("type",
|
|
"Type",
|
|
"The type of the color area",
|
|
GIMP_TYPE_COLOR_AREA_TYPE,
|
|
GIMP_COLOR_AREA_FLAT,
|
|
GIMP_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT));
|
|
/**
|
|
* GimpColorArea:drag-type:
|
|
*
|
|
* The modifier mask that should trigger drags.
|
|
*
|
|
* Since: 2.4
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_DRAG_MASK,
|
|
g_param_spec_flags ("drag-mask",
|
|
"Drag Mask",
|
|
"The modifier mask that triggers dragging the color",
|
|
GDK_TYPE_MODIFIER_TYPE,
|
|
0,
|
|
GIMP_PARAM_WRITABLE));
|
|
/**
|
|
* GimpColorArea:draw-border:
|
|
*
|
|
* Whether to draw a thin border in the foreground color around the area.
|
|
*
|
|
* Since: 2.4
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_DRAW_BORDER,
|
|
g_param_spec_boolean ("draw-border",
|
|
"Draw Border",
|
|
"Whether to draw a thin border in the foreground color around the area",
|
|
FALSE,
|
|
GIMP_PARAM_READWRITE));
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_init (GimpColorArea *area)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
area->priv = gimp_color_area_get_instance_private (area);
|
|
|
|
priv = GET_PRIVATE (area);
|
|
|
|
priv->buf = NULL;
|
|
priv->width = 0;
|
|
priv->height = 0;
|
|
priv->rowstride = 0;
|
|
priv->draw_border = FALSE;
|
|
|
|
gtk_drag_dest_set (GTK_WIDGET (area),
|
|
GTK_DEST_DEFAULT_HIGHLIGHT |
|
|
GTK_DEST_DEFAULT_MOTION |
|
|
GTK_DEST_DEFAULT_DROP,
|
|
&target, 1,
|
|
GDK_ACTION_COPY);
|
|
|
|
gimp_widget_track_monitor (GTK_WIDGET (area),
|
|
G_CALLBACK (gimp_color_area_destroy_transform),
|
|
NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_dispose (GObject *object)
|
|
{
|
|
GimpColorArea *area = GIMP_COLOR_AREA (object);
|
|
|
|
gimp_color_area_set_color_config (area, NULL);
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_finalize (GObject *object)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (object);
|
|
|
|
g_clear_pointer (&priv->buf, g_free);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_COLOR:
|
|
g_value_set_boxed (value, &priv->color);
|
|
break;
|
|
|
|
case PROP_TYPE:
|
|
g_value_set_enum (value, priv->type);
|
|
break;
|
|
|
|
case PROP_DRAW_BORDER:
|
|
g_value_set_boolean (value, priv->draw_border);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GimpColorArea *area = GIMP_COLOR_AREA (object);
|
|
GdkModifierType drag_mask;
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_COLOR:
|
|
gimp_color_area_set_color (area, g_value_get_boxed (value));
|
|
break;
|
|
|
|
case PROP_TYPE:
|
|
gimp_color_area_set_type (area, g_value_get_enum (value));
|
|
break;
|
|
|
|
case PROP_DRAG_MASK:
|
|
drag_mask = g_value_get_flags (value) & (GDK_BUTTON1_MASK |
|
|
GDK_BUTTON2_MASK |
|
|
GDK_BUTTON3_MASK);
|
|
if (drag_mask)
|
|
gtk_drag_source_set (GTK_WIDGET (area),
|
|
drag_mask,
|
|
&target, 1,
|
|
GDK_ACTION_COPY | GDK_ACTION_MOVE);
|
|
else
|
|
gtk_drag_source_unset (GTK_WIDGET (area));
|
|
break;
|
|
|
|
case PROP_DRAW_BORDER:
|
|
gimp_color_area_set_draw_border (area, g_value_get_boolean (value));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (widget);
|
|
|
|
GTK_WIDGET_CLASS (parent_class)->size_allocate (widget, allocation);
|
|
|
|
if (allocation->width != priv->width ||
|
|
allocation->height != priv->height)
|
|
{
|
|
priv->width = allocation->width;
|
|
priv->height = allocation->height;
|
|
|
|
priv->rowstride = priv->width * 4 + 4;
|
|
|
|
g_free (priv->buf);
|
|
priv->buf = g_new (guchar, priv->rowstride * priv->height);
|
|
|
|
priv->needs_render = TRUE;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gimp_color_area_draw (GtkWidget *widget,
|
|
cairo_t *cr)
|
|
{
|
|
GimpColorArea *area = GIMP_COLOR_AREA (widget);
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (area);
|
|
GtkStyleContext *context = gtk_widget_get_style_context (widget);
|
|
cairo_surface_t *buffer;
|
|
|
|
if (! priv->buf)
|
|
return FALSE;
|
|
|
|
if (priv->needs_render)
|
|
gimp_color_area_render (area);
|
|
|
|
if (! priv->transform)
|
|
gimp_color_area_create_transform (area);
|
|
|
|
if (priv->transform)
|
|
{
|
|
const Babl *format = babl_format ("cairo-RGB24");
|
|
guchar *buf = g_new (guchar, priv->rowstride * priv->height);
|
|
guchar *src = priv->buf;
|
|
guchar *dest = buf;
|
|
guint i;
|
|
|
|
for (i = 0; i < priv->height; i++)
|
|
{
|
|
gimp_color_transform_process_pixels (priv->transform,
|
|
format, src,
|
|
format, dest,
|
|
priv->width);
|
|
|
|
src += priv->rowstride;
|
|
dest += priv->rowstride;
|
|
}
|
|
|
|
buffer = cairo_image_surface_create_for_data (buf,
|
|
CAIRO_FORMAT_RGB24,
|
|
priv->width,
|
|
priv->height,
|
|
priv->rowstride);
|
|
cairo_surface_set_user_data (buffer, NULL,
|
|
buf, (cairo_destroy_func_t) g_free);
|
|
}
|
|
else
|
|
{
|
|
buffer = cairo_image_surface_create_for_data (priv->buf,
|
|
CAIRO_FORMAT_RGB24,
|
|
priv->width,
|
|
priv->height,
|
|
priv->rowstride);
|
|
}
|
|
|
|
cairo_set_source_surface (cr, buffer, 0.0, 0.0);
|
|
cairo_surface_destroy (buffer);
|
|
|
|
if (! gtk_widget_is_sensitive (widget))
|
|
{
|
|
static cairo_pattern_t *pattern = NULL;
|
|
|
|
if (! pattern)
|
|
{
|
|
static const guchar stipple[] = { 0, 255, 0, 0,
|
|
255, 0, 0, 0 };
|
|
cairo_surface_t *surface;
|
|
gint stride;
|
|
|
|
stride = cairo_format_stride_for_width (CAIRO_FORMAT_A8, 2);
|
|
|
|
surface = cairo_image_surface_create_for_data ((guchar *) stipple,
|
|
CAIRO_FORMAT_A8,
|
|
2, 2, stride);
|
|
pattern = cairo_pattern_create_for_surface (surface);
|
|
cairo_surface_destroy (surface);
|
|
|
|
cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
|
|
}
|
|
|
|
cairo_mask (cr, pattern);
|
|
}
|
|
else
|
|
{
|
|
cairo_paint (cr);
|
|
}
|
|
|
|
if (priv->config &&
|
|
((priv->color.r < 0.0 || priv->color.r > 1.0 ||
|
|
priv->color.g < 0.0 || priv->color.g > 1.0 ||
|
|
priv->color.b < 0.0 || priv->color.b > 1.0) ||
|
|
priv->out_of_gamut))
|
|
{
|
|
GimpRGB color;
|
|
gint side = MIN (priv->width, priv->height) * 2 / 3;
|
|
|
|
cairo_move_to (cr, priv->width, 0);
|
|
cairo_line_to (cr, priv->width - side, 0);
|
|
cairo_line_to (cr, priv->width, side);
|
|
cairo_line_to (cr, priv->width, 0);
|
|
|
|
gimp_color_config_get_out_of_gamut_color (priv->config, &color);
|
|
gimp_cairo_set_source_rgb (cr, &color);
|
|
|
|
cairo_fill (cr);
|
|
}
|
|
|
|
if (priv->draw_border)
|
|
{
|
|
GdkRGBA color;
|
|
|
|
cairo_set_line_width (cr, 1.0);
|
|
|
|
gtk_style_context_get_color (context, gtk_widget_get_state_flags (widget),
|
|
&color);
|
|
gdk_cairo_set_source_rgba (cr, &color);
|
|
|
|
cairo_rectangle (cr, 0.5, 0.5, priv->width - 1, priv->height - 1);
|
|
|
|
cairo_stroke (cr);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_new:
|
|
* @color: A pointer to a #GimpRGB struct.
|
|
* @type: The type of color area to create.
|
|
* @drag_mask: The event_mask that should trigger drags.
|
|
*
|
|
* Creates a new #GimpColorArea widget.
|
|
*
|
|
* This returns a preview area showing the color. It handles color
|
|
* DND. If the color changes, the "color_changed" signal is emitted.
|
|
*
|
|
* Returns: Pointer to the new #GimpColorArea widget.
|
|
**/
|
|
GtkWidget *
|
|
gimp_color_area_new (const GimpRGB *color,
|
|
GimpColorAreaType type,
|
|
GdkModifierType drag_mask)
|
|
{
|
|
return g_object_new (GIMP_TYPE_COLOR_AREA,
|
|
"color", color,
|
|
"type", type,
|
|
"drag-mask", drag_mask,
|
|
NULL);
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_set_color:
|
|
* @area: Pointer to a #GimpColorArea.
|
|
* @color: Pointer to a #GimpRGB struct that defines the new color.
|
|
*
|
|
* Sets @area to a different @color.
|
|
**/
|
|
void
|
|
gimp_color_area_set_color (GimpColorArea *area,
|
|
const GimpRGB *color)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
g_return_if_fail (GIMP_IS_COLOR_AREA (area));
|
|
g_return_if_fail (color != NULL);
|
|
|
|
priv = GET_PRIVATE (area);
|
|
|
|
if (gimp_rgba_distance (&priv->color, color) < GIMP_RGBA_EPSILON)
|
|
return;
|
|
|
|
priv->color = *color;
|
|
|
|
priv->needs_render = TRUE;
|
|
gtk_widget_queue_draw (GTK_WIDGET (area));
|
|
|
|
g_object_notify (G_OBJECT (area), "color");
|
|
|
|
g_signal_emit (area, gimp_color_area_signals[COLOR_CHANGED], 0);
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_get_color:
|
|
* @area: Pointer to a #GimpColorArea.
|
|
* @color: (out caller-allocates): Pointer to a #GimpRGB struct
|
|
* that is used to return the color.
|
|
*
|
|
* Retrieves the current color of the @area.
|
|
**/
|
|
void
|
|
gimp_color_area_get_color (GimpColorArea *area,
|
|
GimpRGB *color)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
g_return_if_fail (GIMP_IS_COLOR_AREA (area));
|
|
g_return_if_fail (color != NULL);
|
|
|
|
priv = GET_PRIVATE (area);
|
|
|
|
*color = priv->color;
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_has_alpha:
|
|
* @area: Pointer to a #GimpColorArea.
|
|
*
|
|
* Checks whether the @area shows transparency information. This is determined
|
|
* via the @area's #GimpColorAreaType.
|
|
*
|
|
* Returns: %TRUE if @area shows transparency information, %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
gimp_color_area_has_alpha (GimpColorArea *area)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
g_return_val_if_fail (GIMP_IS_COLOR_AREA (area), FALSE);
|
|
|
|
priv = GET_PRIVATE (area);
|
|
|
|
return priv->type != GIMP_COLOR_AREA_FLAT;
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_set_type:
|
|
* @area: Pointer to a #GimpColorArea.
|
|
* @type: A #GimpColorAreaType.
|
|
*
|
|
* Changes the type of @area. The #GimpColorAreaType determines
|
|
* whether the widget shows transparency information and chooses the
|
|
* size of the checkerboard used to do that.
|
|
**/
|
|
void
|
|
gimp_color_area_set_type (GimpColorArea *area,
|
|
GimpColorAreaType type)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
g_return_if_fail (GIMP_IS_COLOR_AREA (area));
|
|
|
|
priv = GET_PRIVATE (area);
|
|
|
|
if (priv->type != type)
|
|
{
|
|
priv->type = type;
|
|
|
|
priv->needs_render = TRUE;
|
|
gtk_widget_queue_draw (GTK_WIDGET (area));
|
|
|
|
g_object_notify (G_OBJECT (area), "type");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_enable_drag:
|
|
* @area: A #GimpColorArea.
|
|
* @drag_mask: The bitmask of buttons that can start the drag.
|
|
*
|
|
* Allows dragging the color displayed with buttons identified by
|
|
* @drag_mask. The drag supports targets of type "application/x-color".
|
|
*
|
|
* Note that setting a @drag_mask of 0 disables the drag ability.
|
|
**/
|
|
void
|
|
gimp_color_area_enable_drag (GimpColorArea *area,
|
|
GdkModifierType drag_mask)
|
|
{
|
|
g_return_if_fail (GIMP_IS_COLOR_AREA (area));
|
|
|
|
g_object_set (area,
|
|
"drag-mask", drag_mask,
|
|
NULL);
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_set_draw_border:
|
|
* @area: Pointer to a #GimpColorArea.
|
|
* @draw_border: whether to draw a border or not
|
|
*
|
|
* The @area can draw a thin border in the foreground color around
|
|
* itself. This function toggles this behaviour on and off. The
|
|
* default is not draw a border.
|
|
**/
|
|
void
|
|
gimp_color_area_set_draw_border (GimpColorArea *area,
|
|
gboolean draw_border)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
g_return_if_fail (GIMP_IS_COLOR_AREA (area));
|
|
|
|
priv = GET_PRIVATE (area);
|
|
|
|
draw_border = draw_border ? TRUE : FALSE;
|
|
|
|
if (priv->draw_border != draw_border)
|
|
{
|
|
priv->draw_border = draw_border;
|
|
|
|
gtk_widget_queue_draw (GTK_WIDGET (area));
|
|
|
|
g_object_notify (G_OBJECT (area), "draw-border");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_set_out_of_gamut:
|
|
* @area: a #GimpColorArea widget.
|
|
* @out_of_gamut: whether to show an out-of-gamut indicator
|
|
*
|
|
* Sets the color area to render as an out-of-gamut color, i.e. with a
|
|
* small triangle on a corner using the color management out of gamut
|
|
* color (as per gimp_color_area_set_color_config()).
|
|
*
|
|
* By default, @area will render as out-of-gamut for any RGB color with
|
|
* a channel out of the [0; 1] range. This function allows to consider
|
|
* more colors out of gamut (for instance non-gray colors on a grayscale
|
|
* image, or colors absent of palettes in indexed images, etc.)
|
|
*
|
|
* Since: 2.10.10
|
|
*/
|
|
void
|
|
gimp_color_area_set_out_of_gamut (GimpColorArea *area,
|
|
gboolean out_of_gamut)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
g_return_if_fail (GIMP_IS_COLOR_AREA (area));
|
|
|
|
priv = GET_PRIVATE (area);
|
|
if (priv->out_of_gamut != out_of_gamut)
|
|
{
|
|
priv->out_of_gamut = out_of_gamut;
|
|
gtk_widget_queue_draw (GTK_WIDGET (area));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gimp_color_area_set_color_config:
|
|
* @area: a #GimpColorArea widget.
|
|
* @config: a #GimpColorConfig object.
|
|
*
|
|
* Sets the color management configuration to use with this color area.
|
|
*
|
|
* Since: 2.10
|
|
*/
|
|
void
|
|
gimp_color_area_set_color_config (GimpColorArea *area,
|
|
GimpColorConfig *config)
|
|
{
|
|
GimpColorAreaPrivate *priv;
|
|
|
|
g_return_if_fail (GIMP_IS_COLOR_AREA (area));
|
|
g_return_if_fail (config == NULL || GIMP_IS_COLOR_CONFIG (config));
|
|
|
|
priv = GET_PRIVATE (area);
|
|
|
|
if (config != priv->config)
|
|
{
|
|
if (priv->config)
|
|
{
|
|
g_signal_handlers_disconnect_by_func (priv->config,
|
|
gimp_color_area_destroy_transform,
|
|
area);
|
|
|
|
gimp_color_area_destroy_transform (area);
|
|
}
|
|
|
|
g_set_object (&priv->config, config);
|
|
|
|
if (priv->config)
|
|
{
|
|
g_signal_connect_swapped (priv->config, "notify",
|
|
G_CALLBACK (gimp_color_area_destroy_transform),
|
|
area);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* private functions */
|
|
|
|
static void
|
|
gimp_color_area_render_buf (GtkWidget *widget,
|
|
GimpColorAreaType type,
|
|
guchar *buf,
|
|
guint width,
|
|
guint height,
|
|
guint rowstride,
|
|
GimpRGB *color)
|
|
{
|
|
guint x, y;
|
|
guint check_size = 0;
|
|
guchar light[3];
|
|
guchar dark[3];
|
|
guchar opaque[3];
|
|
guchar *p;
|
|
gdouble frac;
|
|
|
|
switch (type)
|
|
{
|
|
case GIMP_COLOR_AREA_FLAT:
|
|
check_size = 0;
|
|
break;
|
|
|
|
case GIMP_COLOR_AREA_SMALL_CHECKS:
|
|
check_size = GIMP_CHECK_SIZE_SM;
|
|
break;
|
|
|
|
case GIMP_COLOR_AREA_LARGE_CHECKS:
|
|
check_size = GIMP_CHECK_SIZE;
|
|
break;
|
|
}
|
|
|
|
gimp_rgb_get_uchar (color, opaque, opaque + 1, opaque + 2);
|
|
|
|
if (check_size == 0 || color->a == 1.0)
|
|
{
|
|
for (y = 0; y < height; y++)
|
|
{
|
|
p = buf + y * rowstride;
|
|
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
GIMP_CAIRO_RGB24_SET_PIXEL (p,
|
|
opaque[0],
|
|
opaque[1],
|
|
opaque[2]);
|
|
|
|
p += 4;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
light[0] = (GIMP_CHECK_LIGHT +
|
|
(color->r - GIMP_CHECK_LIGHT) * color->a) * 255.999;
|
|
light[1] = (GIMP_CHECK_LIGHT +
|
|
(color->g - GIMP_CHECK_LIGHT) * color->a) * 255.999;
|
|
light[2] = (GIMP_CHECK_LIGHT +
|
|
(color->b - GIMP_CHECK_LIGHT) * color->a) * 255.999;
|
|
|
|
dark[0] = (GIMP_CHECK_DARK +
|
|
(color->r - GIMP_CHECK_DARK) * color->a) * 255.999;
|
|
dark[1] = (GIMP_CHECK_DARK +
|
|
(color->g - GIMP_CHECK_DARK) * color->a) * 255.999;
|
|
dark[2] = (GIMP_CHECK_DARK +
|
|
(color->b - GIMP_CHECK_DARK) * color->a) * 255.999;
|
|
|
|
for (y = 0; y < height; y++)
|
|
{
|
|
p = buf + y * rowstride;
|
|
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
if ((width - x) * height > y * width)
|
|
{
|
|
GIMP_CAIRO_RGB24_SET_PIXEL (p,
|
|
opaque[0],
|
|
opaque[1],
|
|
opaque[2]);
|
|
p += 4;
|
|
|
|
continue;
|
|
}
|
|
|
|
frac = y - (gdouble) ((width - x) * height) / (gdouble) width;
|
|
|
|
if (((x / check_size) ^ (y / check_size)) & 1)
|
|
{
|
|
if ((gint) frac)
|
|
{
|
|
GIMP_CAIRO_RGB24_SET_PIXEL (p,
|
|
light[0],
|
|
light[1],
|
|
light[2]);
|
|
}
|
|
else
|
|
{
|
|
GIMP_CAIRO_RGB24_SET_PIXEL (p,
|
|
((gdouble) light[0] * frac +
|
|
(gdouble) opaque[0] * (1.0 - frac)),
|
|
((gdouble) light[1] * frac +
|
|
(gdouble) opaque[1] * (1.0 - frac)),
|
|
((gdouble) light[2] * frac +
|
|
(gdouble) opaque[2] * (1.0 - frac)));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((gint) frac)
|
|
{
|
|
GIMP_CAIRO_RGB24_SET_PIXEL (p,
|
|
dark[0],
|
|
dark[1],
|
|
dark[2]);
|
|
}
|
|
else
|
|
{
|
|
GIMP_CAIRO_RGB24_SET_PIXEL (p,
|
|
((gdouble) dark[0] * frac +
|
|
(gdouble) opaque[0] * (1.0 - frac)),
|
|
((gdouble) dark[1] * frac +
|
|
(gdouble) opaque[1] * (1.0 - frac)),
|
|
((gdouble) dark[2] * frac +
|
|
(gdouble) opaque[2] * (1.0 - frac)));
|
|
}
|
|
}
|
|
|
|
p += 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_render (GimpColorArea *area)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (area);
|
|
|
|
if (! priv->buf)
|
|
return;
|
|
|
|
gimp_color_area_render_buf (GTK_WIDGET (area),
|
|
priv->type,
|
|
priv->buf,
|
|
priv->width, priv->height, priv->rowstride,
|
|
&priv->color);
|
|
|
|
priv->needs_render = FALSE;
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (widget);
|
|
GimpRGB color;
|
|
GtkWidget *window;
|
|
GtkWidget *frame;
|
|
GtkWidget *color_area;
|
|
|
|
window = gtk_window_new (GTK_WINDOW_POPUP);
|
|
gtk_window_set_type_hint (GTK_WINDOW (window), GDK_WINDOW_TYPE_HINT_DND);
|
|
gtk_window_set_screen (GTK_WINDOW (window), gtk_widget_get_screen (widget));
|
|
|
|
gtk_widget_realize (window);
|
|
|
|
frame = gtk_frame_new (NULL);
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_OUT);
|
|
gtk_container_add (GTK_CONTAINER (window), frame);
|
|
|
|
gimp_color_area_get_color (GIMP_COLOR_AREA (widget), &color);
|
|
|
|
color_area = gimp_color_area_new (&color, priv->type, 0);
|
|
|
|
gtk_widget_set_size_request (color_area,
|
|
DRAG_PREVIEW_SIZE, DRAG_PREVIEW_SIZE);
|
|
gtk_container_add (GTK_CONTAINER (frame), color_area);
|
|
gtk_widget_show (color_area);
|
|
gtk_widget_show (frame);
|
|
|
|
g_object_set_data_full (G_OBJECT (widget),
|
|
"gimp-color-area-drag-window",
|
|
window,
|
|
(GDestroyNotify) gtk_widget_destroy);
|
|
|
|
gtk_drag_set_icon_widget (context, window,
|
|
DRAG_ICON_OFFSET, DRAG_ICON_OFFSET);
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_drag_end (GtkWidget *widget,
|
|
GdkDragContext *context)
|
|
{
|
|
g_object_set_data (G_OBJECT (widget),
|
|
"gimp-color-area-drag-window", NULL);
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_drag_data_received (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
GtkSelectionData *selection_data,
|
|
guint info,
|
|
guint time)
|
|
{
|
|
GimpColorArea *area = GIMP_COLOR_AREA (widget);
|
|
const guint16 *vals;
|
|
GimpRGB color;
|
|
|
|
if (gtk_selection_data_get_length (selection_data) != 8 ||
|
|
gtk_selection_data_get_format (selection_data) != 16)
|
|
{
|
|
g_warning ("%s: received invalid color data", G_STRFUNC);
|
|
return;
|
|
}
|
|
|
|
vals = (const guint16 *) gtk_selection_data_get_data (selection_data);
|
|
|
|
gimp_rgba_set (&color,
|
|
(gdouble) vals[0] / 0xffff,
|
|
(gdouble) vals[1] / 0xffff,
|
|
(gdouble) vals[2] / 0xffff,
|
|
(gdouble) vals[3] / 0xffff);
|
|
|
|
gimp_color_area_set_color (area, &color);
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_drag_data_get (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
GtkSelectionData *selection_data,
|
|
guint info,
|
|
guint time)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (widget);
|
|
guint16 vals[4];
|
|
|
|
vals[0] = priv->color.r * 0xffff;
|
|
vals[1] = priv->color.g * 0xffff;
|
|
vals[2] = priv->color.b * 0xffff;
|
|
|
|
if (priv->type == GIMP_COLOR_AREA_FLAT)
|
|
vals[3] = 0xffff;
|
|
else
|
|
vals[3] = priv->color.a * 0xffff;
|
|
|
|
gtk_selection_data_set (selection_data,
|
|
gdk_atom_intern ("application/x-color", FALSE),
|
|
16, (guchar *) vals, 8);
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_create_transform (GimpColorArea *area)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (area);
|
|
|
|
if (priv->config)
|
|
{
|
|
static GimpColorProfile *profile = NULL;
|
|
|
|
const Babl *format = babl_format ("cairo-RGB24");
|
|
|
|
if (G_UNLIKELY (! profile))
|
|
profile = gimp_color_profile_new_rgb_srgb ();
|
|
|
|
priv->transform = gimp_widget_get_color_transform (GTK_WIDGET (area),
|
|
priv->config,
|
|
profile,
|
|
format,
|
|
format);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_color_area_destroy_transform (GimpColorArea *area)
|
|
{
|
|
GimpColorAreaPrivate *priv = GET_PRIVATE (area);
|
|
|
|
g_clear_object (&priv->transform);
|
|
|
|
gtk_widget_queue_draw (GTK_WIDGET (area));
|
|
}
|