mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-01 16:30:15 +00:00
412 lines
9.9 KiB
C
412 lines
9.9 KiB
C
/* GDK - The GIMP Drawing Kit
|
||
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
||
*
|
||
* 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 2 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
|
||
* Lesser 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 <http://www.gnu.org/licenses/>.
|
||
*/
|
||
|
||
/*
|
||
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
||
* file for a list of people on the GTK+ Team. See the ChangeLog
|
||
* files for a list of changes. These files are distributed with
|
||
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
||
*/
|
||
|
||
#include "config.h"
|
||
#include "gdkrgba.h"
|
||
#include <string.h>
|
||
#include <errno.h>
|
||
#include <math.h>
|
||
|
||
#include "fallback-c89.c"
|
||
|
||
/**
|
||
* SECTION:rgba_colors
|
||
* @Short_description: RGBA colors
|
||
* @Title: RGBA Colors
|
||
*
|
||
* #GdkRGBA is a convenient way to pass rgba colors around.
|
||
* It’s based on cairo’s way to deal with colors and mirrors its behavior.
|
||
* All values are in the range from 0.0 to 1.0 inclusive. So the color
|
||
* (0.0, 0.0, 0.0, 0.0) represents transparent black and
|
||
* (1.0, 1.0, 1.0, 1.0) is opaque white. Other values will be clamped
|
||
* to this range when drawing.
|
||
*/
|
||
|
||
G_DEFINE_BOXED_TYPE (GdkRGBA, gdk_rgba,
|
||
gdk_rgba_copy, gdk_rgba_free)
|
||
|
||
/**
|
||
* GdkRGBA:
|
||
* @red: The intensity of the red channel from 0.0 to 1.0 inclusive
|
||
* @green: The intensity of the green channel from 0.0 to 1.0 inclusive
|
||
* @blue: The intensity of the blue channel from 0.0 to 1.0 inclusive
|
||
* @alpha: The opacity of the color from 0.0 for completely translucent to
|
||
* 1.0 for opaque
|
||
*
|
||
* A #GdkRGBA is used to represent a (possibly translucent)
|
||
* color, in a way that is compatible with cairos notion of color.
|
||
*/
|
||
|
||
/**
|
||
* gdk_rgba_copy:
|
||
* @rgba: a #GdkRGBA
|
||
*
|
||
* Makes a copy of a #GdkRGBA.
|
||
*
|
||
* The result must be freed through gdk_rgba_free().
|
||
*
|
||
* Returns: A newly allocated #GdkRGBA, with the same contents as @rgba
|
||
*
|
||
* Since: 3.0
|
||
*/
|
||
GdkRGBA *
|
||
gdk_rgba_copy (const GdkRGBA *rgba)
|
||
{
|
||
return g_slice_dup (GdkRGBA, rgba);
|
||
}
|
||
|
||
/**
|
||
* gdk_rgba_free:
|
||
* @rgba: a #GdkRGBA
|
||
*
|
||
* Frees a #GdkRGBA created with gdk_rgba_copy()
|
||
*
|
||
* Since: 3.0
|
||
*/
|
||
void
|
||
gdk_rgba_free (GdkRGBA *rgba)
|
||
{
|
||
g_slice_free (GdkRGBA, rgba);
|
||
}
|
||
|
||
/**
|
||
* gdk_rgba_is_clear:
|
||
* @rgba: a #GdkRGBA
|
||
*
|
||
* Checks if an @rgba value is transparent. That is, drawing with the value
|
||
* would not produce any change.
|
||
*
|
||
* Returns: %TRUE if the @rgba is clear
|
||
*
|
||
* Since: 3.90
|
||
*/
|
||
gboolean
|
||
gdk_rgba_is_clear (const GdkRGBA *rgba)
|
||
{
|
||
return rgba->alpha < ((double) 0x00ff / (double) 0xffff);
|
||
}
|
||
|
||
/**
|
||
* gdk_rgba_is_opaque:
|
||
* @rgba: a #GdkRGBA
|
||
*
|
||
* Checks if an @rgba value is opaque. That is, drawing with the value
|
||
* will not retain any results from previous contents.
|
||
*
|
||
* Returns: %TRUE if the @rgba is opaque
|
||
*
|
||
* Since: 3.90
|
||
*/
|
||
gboolean
|
||
gdk_rgba_is_opaque (const GdkRGBA *rgba)
|
||
{
|
||
return rgba->alpha > ((double)0xff00 / (double)0xffff);
|
||
}
|
||
|
||
#define SKIP_WHITESPACES(s) while (*(s) == ' ') (s)++;
|
||
|
||
/* Parses a single color component from a rgb() or rgba() specification
|
||
* according to CSS3 rules. Compared to exact CSS3 parsing we are liberal
|
||
* in what we accept as follows:
|
||
*
|
||
* - For non-percentage values, we accept floats in the range 0-255
|
||
* not just [0-9]+ integers
|
||
* - For percentage values we accept any float, not just
|
||
* [ 0-9]+ | [0-9]* “.” [0-9]+
|
||
* - We accept mixed percentages and non-percentages in a single
|
||
* rgb() or rgba() specification.
|
||
*/
|
||
static gboolean
|
||
parse_rgb_value (const gchar *str,
|
||
gchar **endp,
|
||
gdouble *number)
|
||
{
|
||
const char *p;
|
||
|
||
*number = g_ascii_strtod (str, endp);
|
||
if (errno == ERANGE || *endp == str ||
|
||
isinf (*number) || isnan (*number))
|
||
return FALSE;
|
||
|
||
p = *endp;
|
||
|
||
SKIP_WHITESPACES (p);
|
||
|
||
if (*p == '%')
|
||
{
|
||
*endp = (char *)(p + 1);
|
||
*number = CLAMP(*number / 100., 0., 1.);
|
||
}
|
||
else
|
||
{
|
||
*number = CLAMP(*number / 255., 0., 1.);
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
/**
|
||
* gdk_rgba_parse:
|
||
* @rgba: the #GdkRGBA to fill in
|
||
* @spec: the string specifying the color
|
||
*
|
||
* Parses a textual representation of a color, filling in
|
||
* the @red, @green, @blue and @alpha fields of the @rgba #GdkRGBA.
|
||
*
|
||
* The string can be either one of:
|
||
* - A standard name (Taken from the X11 rgb.txt file).
|
||
* - A hexadecimal value in the form “\#rgb”, “\#rrggbb”,
|
||
* “\#rrrgggbbb” or ”\#rrrrggggbbbb”
|
||
* - A RGB color in the form “rgb(r,g,b)” (In this case the color will
|
||
* have full opacity)
|
||
* - A RGBA color in the form “rgba(r,g,b,a)”
|
||
*
|
||
* Where “r”, “g”, “b” and “a” are respectively the red, green, blue and
|
||
* alpha color values. In the last two cases, r g and b are either integers
|
||
* in the range 0 to 255 or percentage values in the range 0% to 100%, and
|
||
* a is a floating point value in the range 0 to 1.
|
||
*
|
||
* Returns: %TRUE if the parsing succeeded
|
||
*
|
||
* Since: 3.0
|
||
*/
|
||
gboolean
|
||
gdk_rgba_parse (GdkRGBA *rgba,
|
||
const gchar *spec)
|
||
{
|
||
gboolean has_alpha;
|
||
gdouble r, g, b, a;
|
||
gchar *str = (gchar *) spec;
|
||
gchar *p;
|
||
|
||
g_return_val_if_fail (spec != NULL, FALSE);
|
||
|
||
|
||
if (strncmp (str, "rgba", 4) == 0)
|
||
{
|
||
has_alpha = TRUE;
|
||
str += 4;
|
||
}
|
||
else if (strncmp (str, "rgb", 3) == 0)
|
||
{
|
||
has_alpha = FALSE;
|
||
a = 1;
|
||
str += 3;
|
||
}
|
||
else
|
||
{
|
||
PangoColor pango_color;
|
||
|
||
/* Resort on PangoColor for rgb.txt color
|
||
* map and '#' prefixed colors
|
||
*/
|
||
if (pango_color_parse (&pango_color, str))
|
||
{
|
||
if (rgba)
|
||
{
|
||
rgba->red = pango_color.red / 65535.;
|
||
rgba->green = pango_color.green / 65535.;
|
||
rgba->blue = pango_color.blue / 65535.;
|
||
rgba->alpha = 1;
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
else
|
||
return FALSE;
|
||
}
|
||
|
||
SKIP_WHITESPACES (str);
|
||
|
||
if (*str != '(')
|
||
return FALSE;
|
||
|
||
str++;
|
||
|
||
/* Parse red */
|
||
SKIP_WHITESPACES (str);
|
||
if (!parse_rgb_value (str, &str, &r))
|
||
return FALSE;
|
||
SKIP_WHITESPACES (str);
|
||
|
||
if (*str != ',')
|
||
return FALSE;
|
||
|
||
str++;
|
||
|
||
/* Parse green */
|
||
SKIP_WHITESPACES (str);
|
||
if (!parse_rgb_value (str, &str, &g))
|
||
return FALSE;
|
||
SKIP_WHITESPACES (str);
|
||
|
||
if (*str != ',')
|
||
return FALSE;
|
||
|
||
str++;
|
||
|
||
/* Parse blue */
|
||
SKIP_WHITESPACES (str);
|
||
if (!parse_rgb_value (str, &str, &b))
|
||
return FALSE;
|
||
SKIP_WHITESPACES (str);
|
||
|
||
if (has_alpha)
|
||
{
|
||
if (*str != ',')
|
||
return FALSE;
|
||
|
||
str++;
|
||
|
||
SKIP_WHITESPACES (str);
|
||
a = g_ascii_strtod (str, &p);
|
||
if (errno == ERANGE || p == str ||
|
||
isinf (a) || isnan (a))
|
||
return FALSE;
|
||
str = p;
|
||
SKIP_WHITESPACES (str);
|
||
}
|
||
|
||
if (*str != ')')
|
||
return FALSE;
|
||
|
||
str++;
|
||
|
||
SKIP_WHITESPACES (str);
|
||
|
||
if (*str != '\0')
|
||
return FALSE;
|
||
|
||
if (rgba)
|
||
{
|
||
rgba->red = CLAMP (r, 0, 1);
|
||
rgba->green = CLAMP (g, 0, 1);
|
||
rgba->blue = CLAMP (b, 0, 1);
|
||
rgba->alpha = CLAMP (a, 0, 1);
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
#undef SKIP_WHITESPACES
|
||
|
||
/**
|
||
* gdk_rgba_hash:
|
||
* @p: (type GdkRGBA): a #GdkRGBA pointer
|
||
*
|
||
* A hash function suitable for using for a hash
|
||
* table that stores #GdkRGBAs.
|
||
*
|
||
* Returns: The hash value for @p
|
||
*
|
||
* Since: 3.0
|
||
*/
|
||
guint
|
||
gdk_rgba_hash (gconstpointer p)
|
||
{
|
||
const GdkRGBA *rgba = p;
|
||
|
||
return ((guint) (rgba->red * 65535) +
|
||
((guint) (rgba->green * 65535) << 11) +
|
||
((guint) (rgba->blue * 65535) << 22) +
|
||
((guint) (rgba->alpha * 65535) >> 6));
|
||
}
|
||
|
||
/**
|
||
* gdk_rgba_equal:
|
||
* @p1: (type GdkRGBA): a #GdkRGBA pointer
|
||
* @p2: (type GdkRGBA): another #GdkRGBA pointer
|
||
*
|
||
* Compares two RGBA colors.
|
||
*
|
||
* Returns: %TRUE if the two colors compare equal
|
||
*
|
||
* Since: 3.0
|
||
*/
|
||
gboolean
|
||
gdk_rgba_equal (gconstpointer p1,
|
||
gconstpointer p2)
|
||
{
|
||
const GdkRGBA *rgba1, *rgba2;
|
||
|
||
rgba1 = p1;
|
||
rgba2 = p2;
|
||
|
||
if (rgba1->red == rgba2->red &&
|
||
rgba1->green == rgba2->green &&
|
||
rgba1->blue == rgba2->blue &&
|
||
rgba1->alpha == rgba2->alpha)
|
||
return TRUE;
|
||
|
||
return FALSE;
|
||
}
|
||
|
||
/**
|
||
* gdk_rgba_to_string:
|
||
* @rgba: a #GdkRGBA
|
||
*
|
||
* Returns a textual specification of @rgba in the form
|
||
* `rgb (r, g, b)` or
|
||
* `rgba (r, g, b, a)`,
|
||
* where “r”, “g”, “b” and “a” represent the red, green,
|
||
* blue and alpha values respectively. r, g, and b are
|
||
* represented as integers in the range 0 to 255, and a
|
||
* is represented as floating point value in the range 0 to 1.
|
||
*
|
||
* These string forms are string forms those supported by
|
||
* the CSS3 colors module, and can be parsed by gdk_rgba_parse().
|
||
*
|
||
* Note that this string representation may lose some
|
||
* precision, since r, g and b are represented as 8-bit
|
||
* integers. If this is a concern, you should use a
|
||
* different representation.
|
||
*
|
||
* Returns: A newly allocated text string
|
||
*
|
||
* Since: 3.0
|
||
*/
|
||
gchar *
|
||
gdk_rgba_to_string (const GdkRGBA *rgba)
|
||
{
|
||
if (rgba->alpha > 0.999)
|
||
{
|
||
return g_strdup_printf ("rgb(%d,%d,%d)",
|
||
(int)(0.5 + CLAMP (rgba->red, 0., 1.) * 255.),
|
||
(int)(0.5 + CLAMP (rgba->green, 0., 1.) * 255.),
|
||
(int)(0.5 + CLAMP (rgba->blue, 0., 1.) * 255.));
|
||
}
|
||
else
|
||
{
|
||
gchar alpha[G_ASCII_DTOSTR_BUF_SIZE];
|
||
|
||
g_ascii_formatd (alpha, G_ASCII_DTOSTR_BUF_SIZE, "%g", CLAMP (rgba->alpha, 0, 1));
|
||
|
||
return g_strdup_printf ("rgba(%d,%d,%d,%s)",
|
||
(int)(0.5 + CLAMP (rgba->red, 0., 1.) * 255.),
|
||
(int)(0.5 + CLAMP (rgba->green, 0., 1.) * 255.),
|
||
(int)(0.5 + CLAMP (rgba->blue, 0., 1.) * 255.),
|
||
alpha);
|
||
}
|
||
}
|