2011-04-28 21:50:39 +08:00
|
|
|
/* LIBGIMP - The GIMP Library
|
|
|
|
* Copyright (C) 1995-1997 Peter Mattis and Spencer Kimball
|
|
|
|
*
|
2012-05-03 09:56:10 +08:00
|
|
|
* gimpcairo.h
|
|
|
|
* Copyright (C) 2007 Sven Neumann <sven@gimp.org>
|
|
|
|
* 2010-2012 Michael Natterer <mitch@gimp.org>
|
2011-04-28 21:50:39 +08:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* 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
|
2018-07-12 05:27:07 +08:00
|
|
|
* <https://www.gnu.org/licenses/>.
|
2011-04-28 21:50:39 +08:00
|
|
|
*/
|
|
|
|
|
2012-05-03 09:56:10 +08:00
|
|
|
#ifndef __GIMP_CAIRO_H__
|
|
|
|
#define __GIMP_CAIRO_H__
|
2011-04-28 21:50:39 +08:00
|
|
|
|
|
|
|
|
2012-05-03 10:07:16 +08:00
|
|
|
cairo_pattern_t * gimp_cairo_checkerboard_create (cairo_t *cr,
|
|
|
|
gint size,
|
2024-03-31 09:09:45 +08:00
|
|
|
const GeglColor *light,
|
|
|
|
const GeglColor *dark);
|
2012-05-03 10:07:16 +08:00
|
|
|
|
|
|
|
const Babl * gimp_cairo_surface_get_format (cairo_surface_t *surface);
|
app, libgimp, pdb, plug-ins: GimpText* using GeglColor.
One of the big improvement in this commit is that text layers are now much
better at space accuracy. They were already space-aware, yet rendered as sRGB u8
only before being converted to the image's space. It means that text layers had
the following limitations:
* Any color out of sRGB gamut were trimmed.
* Precision was always 8-bit (even if the image was high-bit depth).
Now GimpTextLayout keeps track of its source space (for RGB and CMYK only, this
won't be as easy when we will support more backend, since Cairo has only RGB
support for image data) and the image TRC (in case it bypasses the color space's
TRB) and it draws within this gamut and space.
It means first that we are not limited to sRGB colors; we will draw text main
color in the full image gamut, with still 2 remaining limitations:
* Unbounded colors are impossible because Pango format (to color text) uses
hexadecimal (so even with half/float images, you can't draw out-of-gamut text
unfortunately).
* Main color precision is still 8-bit, yet a tiny bit better than before as we
at least follow TRC (so we avoid some of the precision loss when converting,
even though the bit-depth is still the biggest loss).
The outline color on the other hand is drawn through Cairo API entirely, in
float. This means that the outline color will now be without any precision loss.
Note that this depends on CAIRO_FORMAT_RGBA128F which is only available since
Cairo 1.17.2 which is not in Debian bookworm (our current baseline for GIMP
3.0). It means that the old precision will still happen with older Cairo
version, as determined by #if code at compilation.
2023-11-18 05:36:31 +08:00
|
|
|
GeglBuffer * gimp_cairo_surface_create_buffer (cairo_surface_t *surface,
|
|
|
|
const Babl *format);
|
2011-04-28 21:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
|
/* some useful macros for writing directly to a Cairo surface */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GIMP_CAIRO_RGB24_SET_PIXEL:
|
|
|
|
* @d: pointer to the destination buffer
|
|
|
|
* @r: red component
|
|
|
|
* @g: green component
|
|
|
|
* @b: blue component
|
|
|
|
*
|
|
|
|
* Sets a single pixel in an Cairo image surface in %CAIRO_FORMAT_RGB24.
|
|
|
|
*
|
2015-06-01 03:18:09 +08:00
|
|
|
* Since: 2.6
|
2011-04-28 21:50:39 +08:00
|
|
|
**/
|
|
|
|
#if G_BYTE_ORDER == G_LITTLE_ENDIAN
|
|
|
|
#define GIMP_CAIRO_RGB24_SET_PIXEL(d, r, g, b) \
|
|
|
|
G_STMT_START { d[0] = (b); d[1] = (g); d[2] = (r); } G_STMT_END
|
|
|
|
#else
|
|
|
|
#define GIMP_CAIRO_RGB24_SET_PIXEL(d, r, g, b) \
|
|
|
|
G_STMT_START { d[1] = (r); d[2] = (g); d[3] = (b); } G_STMT_END
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GIMP_CAIRO_RGB24_GET_PIXEL:
|
|
|
|
* @s: pointer to the source buffer
|
|
|
|
* @r: red component
|
|
|
|
* @g: green component
|
|
|
|
* @b: blue component
|
|
|
|
*
|
|
|
|
* Gets a single pixel from a Cairo image surface in %CAIRO_FORMAT_RGB24.
|
|
|
|
*
|
2015-06-01 03:18:09 +08:00
|
|
|
* Since: 2.8
|
2011-04-28 21:50:39 +08:00
|
|
|
**/
|
|
|
|
#if G_BYTE_ORDER == G_LITTLE_ENDIAN
|
|
|
|
#define GIMP_CAIRO_RGB24_GET_PIXEL(s, r, g, b) \
|
|
|
|
G_STMT_START { (b) = s[0]; (g) = s[1]; (r) = s[2]; } G_STMT_END
|
|
|
|
#else
|
|
|
|
#define GIMP_CAIRO_RGB24_GET_PIXEL(s, r, g, b) \
|
|
|
|
G_STMT_START { (r) = s[1]; (g) = s[2]; (b) = s[3]; } G_STMT_END
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GIMP_CAIRO_ARGB32_SET_PIXEL:
|
|
|
|
* @d: pointer to the destination buffer
|
|
|
|
* @r: red component, not pre-multiplied
|
|
|
|
* @g: green component, not pre-multiplied
|
|
|
|
* @b: blue component, not pre-multiplied
|
|
|
|
* @a: alpha component
|
|
|
|
*
|
|
|
|
* Sets a single pixel in an Cairo image surface in %CAIRO_FORMAT_ARGB32.
|
|
|
|
*
|
2015-06-01 03:18:09 +08:00
|
|
|
* Since: 2.6
|
2011-04-28 21:50:39 +08:00
|
|
|
**/
|
|
|
|
#if G_BYTE_ORDER == G_LITTLE_ENDIAN
|
|
|
|
#define GIMP_CAIRO_ARGB32_SET_PIXEL(d, r, g, b, a) \
|
|
|
|
G_STMT_START { \
|
|
|
|
const guint tr = (a) * (r) + 0x80; \
|
|
|
|
const guint tg = (a) * (g) + 0x80; \
|
|
|
|
const guint tb = (a) * (b) + 0x80; \
|
|
|
|
(d)[0] = (((tb) >> 8) + (tb)) >> 8; \
|
|
|
|
(d)[1] = (((tg) >> 8) + (tg)) >> 8; \
|
|
|
|
(d)[2] = (((tr) >> 8) + (tr)) >> 8; \
|
|
|
|
(d)[3] = (a); \
|
|
|
|
} G_STMT_END
|
|
|
|
#else
|
|
|
|
#define GIMP_CAIRO_ARGB32_SET_PIXEL(d, r, g, b, a) \
|
|
|
|
G_STMT_START { \
|
|
|
|
const guint tr = (a) * (r) + 0x80; \
|
|
|
|
const guint tg = (a) * (g) + 0x80; \
|
|
|
|
const guint tb = (a) * (b) + 0x80; \
|
|
|
|
(d)[0] = (a); \
|
|
|
|
(d)[1] = (((tr) >> 8) + (tr)) >> 8; \
|
|
|
|
(d)[2] = (((tg) >> 8) + (tg)) >> 8; \
|
|
|
|
(d)[3] = (((tb) >> 8) + (tb)) >> 8; \
|
|
|
|
} G_STMT_END
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GIMP_CAIRO_ARGB32_GET_PIXEL:
|
|
|
|
* @s: pointer to the source buffer
|
|
|
|
* @r: red component, not pre-multiplied
|
|
|
|
* @g: green component, not pre-multiplied
|
|
|
|
* @b: blue component, not pre-multiplied
|
|
|
|
* @a: alpha component
|
|
|
|
*
|
|
|
|
* Gets a single pixel from a Cairo image surface in %CAIRO_FORMAT_ARGB32.
|
|
|
|
*
|
2015-06-01 03:18:09 +08:00
|
|
|
* Since: 2.8
|
2011-04-28 21:50:39 +08:00
|
|
|
**/
|
|
|
|
#if G_BYTE_ORDER == G_LITTLE_ENDIAN
|
|
|
|
#define GIMP_CAIRO_ARGB32_GET_PIXEL(s, r, g, b, a) \
|
|
|
|
G_STMT_START { \
|
|
|
|
const guint tb = (s)[0]; \
|
|
|
|
const guint tg = (s)[1]; \
|
|
|
|
const guint tr = (s)[2]; \
|
|
|
|
const guint ta = (s)[3]; \
|
|
|
|
(r) = (tr << 8) / (ta + 1); \
|
|
|
|
(g) = (tg << 8) / (ta + 1); \
|
|
|
|
(b) = (tb << 8) / (ta + 1); \
|
|
|
|
(a) = ta; \
|
|
|
|
} G_STMT_END
|
|
|
|
#else
|
|
|
|
#define GIMP_CAIRO_ARGB32_GET_PIXEL(s, r, g, b, a) \
|
|
|
|
G_STMT_START { \
|
|
|
|
const guint ta = (s)[0]; \
|
|
|
|
const guint tr = (s)[1]; \
|
|
|
|
const guint tg = (s)[2]; \
|
|
|
|
const guint tb = (s)[3]; \
|
|
|
|
(r) = (tr << 8) / (ta + 1); \
|
|
|
|
(g) = (tg << 8) / (ta + 1); \
|
|
|
|
(b) = (tb << 8) / (ta + 1); \
|
|
|
|
(a) = ta; \
|
|
|
|
} G_STMT_END
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2012-05-03 09:56:10 +08:00
|
|
|
#endif /* __GIMP_CAIRO_H__ */
|