mirror of
https://gitlab.gnome.org/GNOME/gimp.git
synced 2025-07-03 09:23:24 +00:00

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.
225 lines
6.6 KiB
C
225 lines
6.6 KiB
C
/* LIBGIMP - The GIMP Library
|
|
* Copyright (C) 1995-1997 Peter Mattis and Spencer Kimball
|
|
*
|
|
* gimpcairo.c
|
|
* Copyright (C) 2007 Sven Neumann <sven@gimp.org>
|
|
* 2010-2012 Michael Natterer <mitch@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
|
|
* 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
|
|
* <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <cairo.h>
|
|
#include <gio/gio.h>
|
|
#include <gegl.h>
|
|
|
|
#include "libgimpbase/gimpbase.h"
|
|
|
|
#include "gimpcolortypes.h"
|
|
|
|
#include "gimpcairo.h"
|
|
|
|
|
|
/**
|
|
* SECTION: gimpcairo
|
|
* @title: GimpCairo
|
|
* @short_description: Color utility functions for cairo
|
|
*
|
|
* Utility functions that make cairo easier to use with GIMP color
|
|
* data types.
|
|
**/
|
|
|
|
|
|
/**
|
|
* gimp_cairo_set_source_rgb:
|
|
* @cr: Cairo context
|
|
* @color: GimpRGB color
|
|
*
|
|
* Sets the source pattern within @cr to the solid opaque color
|
|
* described by @color.
|
|
*
|
|
* This function calls cairo_set_source_rgb() for you.
|
|
*
|
|
* Since: 2.6
|
|
**/
|
|
void
|
|
gimp_cairo_set_source_rgb (cairo_t *cr,
|
|
const GimpRGB *color)
|
|
{
|
|
cairo_set_source_rgb (cr, color->r, color->g, color->b);
|
|
}
|
|
|
|
/**
|
|
* gimp_cairo_set_source_rgba:
|
|
* @cr: Cairo context
|
|
* @color: GimpRGB color
|
|
*
|
|
* Sets the source pattern within @cr to the solid translucent color
|
|
* described by @color.
|
|
*
|
|
* This function calls cairo_set_source_rgba() for you.
|
|
*
|
|
* Since: 2.6
|
|
**/
|
|
void
|
|
gimp_cairo_set_source_rgba (cairo_t *cr,
|
|
const GimpRGB *color)
|
|
{
|
|
cairo_set_source_rgba (cr, color->r, color->g, color->b, color->a);
|
|
}
|
|
|
|
/**
|
|
* gimp_cairo_checkerboard_create:
|
|
* @cr: Cairo context
|
|
* @size: check size
|
|
* @light: light check color or %NULL to use the default light gray
|
|
* @dark: dark check color or %NULL to use the default dark gray
|
|
*
|
|
* Create a repeating checkerboard pattern.
|
|
*
|
|
* Returns: a new Cairo pattern that can be used as a source on @cr.
|
|
*
|
|
* Since: 2.6
|
|
**/
|
|
cairo_pattern_t *
|
|
gimp_cairo_checkerboard_create (cairo_t *cr,
|
|
gint size,
|
|
const GimpRGB *light,
|
|
const GimpRGB *dark)
|
|
{
|
|
cairo_t *context;
|
|
cairo_surface_t *surface;
|
|
cairo_pattern_t *pattern;
|
|
|
|
g_return_val_if_fail (cr != NULL, NULL);
|
|
g_return_val_if_fail (size > 0, NULL);
|
|
|
|
surface = cairo_surface_create_similar (cairo_get_target (cr),
|
|
CAIRO_CONTENT_COLOR,
|
|
2 * size, 2 * size);
|
|
context = cairo_create (surface);
|
|
|
|
if (light)
|
|
gimp_cairo_set_source_rgb (context, light);
|
|
else
|
|
cairo_set_source_rgb (context,
|
|
GIMP_CHECK_LIGHT, GIMP_CHECK_LIGHT, GIMP_CHECK_LIGHT);
|
|
|
|
cairo_rectangle (context, 0, 0, size, size);
|
|
cairo_rectangle (context, size, size, size, size);
|
|
cairo_fill (context);
|
|
|
|
if (dark)
|
|
gimp_cairo_set_source_rgb (context, dark);
|
|
else
|
|
cairo_set_source_rgb (context,
|
|
GIMP_CHECK_DARK, GIMP_CHECK_DARK, GIMP_CHECK_DARK);
|
|
|
|
cairo_rectangle (context, 0, size, size, size);
|
|
cairo_rectangle (context, size, 0, size, size);
|
|
cairo_fill (context);
|
|
|
|
cairo_destroy (context);
|
|
|
|
pattern = cairo_pattern_create_for_surface (surface);
|
|
cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
|
|
|
|
cairo_surface_destroy (surface);
|
|
|
|
return pattern;
|
|
}
|
|
|
|
/**
|
|
* gimp_cairo_surface_get_format:
|
|
* @surface: a Cairo surface
|
|
*
|
|
* This function returns a #Babl format that corresponds to @surface's
|
|
* pixel format.
|
|
*
|
|
* Returns: the #Babl format of @surface.
|
|
*
|
|
* Since: 2.10
|
|
**/
|
|
const Babl *
|
|
gimp_cairo_surface_get_format (cairo_surface_t *surface)
|
|
{
|
|
g_return_val_if_fail (surface != NULL, NULL);
|
|
g_return_val_if_fail (cairo_surface_get_type (surface) ==
|
|
CAIRO_SURFACE_TYPE_IMAGE, NULL);
|
|
|
|
switch (cairo_image_surface_get_format (surface))
|
|
{
|
|
case CAIRO_FORMAT_RGB24: return babl_format ("cairo-RGB24");
|
|
case CAIRO_FORMAT_ARGB32: return babl_format ("cairo-ARGB32");
|
|
case CAIRO_FORMAT_A8: return babl_format ("cairo-A8");
|
|
#if CAIRO_VERSION >= CAIRO_VERSION_ENCODE(1, 17, 2)
|
|
/* Since Cairo 1.17.2 */
|
|
case CAIRO_FORMAT_RGB96F: return babl_format ("R'B'B' float");
|
|
case CAIRO_FORMAT_RGBA128F: return babl_format ("R'G'B'A float");
|
|
#endif
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
g_return_val_if_reached (NULL);
|
|
}
|
|
|
|
/**
|
|
* gimp_cairo_surface_create_buffer:
|
|
* @surface: a Cairo surface
|
|
* @format: a Babl format.
|
|
*
|
|
* This function returns a #GeglBuffer which wraps @surface's pixels.
|
|
* It must only be called on image surfaces, calling it on other surface
|
|
* types is an error.
|
|
*
|
|
* If @format is set, the returned [class@Gegl.Buffer] will use it. It has to
|
|
* map with @surface Cairo format. If unset, the buffer format will be
|
|
* determined from @surface. The main difference is that automatically
|
|
* determined format has sRGB space and TRC by default.
|
|
*
|
|
* Returns: (transfer full): a #GeglBuffer
|
|
*
|
|
* Since: 2.10
|
|
**/
|
|
GeglBuffer *
|
|
gimp_cairo_surface_create_buffer (cairo_surface_t *surface,
|
|
const Babl *format)
|
|
{
|
|
gint width;
|
|
gint height;
|
|
|
|
g_return_val_if_fail (surface != NULL, NULL);
|
|
g_return_val_if_fail (cairo_surface_get_type (surface) ==
|
|
CAIRO_SURFACE_TYPE_IMAGE, NULL);
|
|
g_return_val_if_fail (format == NULL ||
|
|
babl_format_get_bytes_per_pixel (format) == babl_format_get_bytes_per_pixel (gimp_cairo_surface_get_format (surface)),
|
|
NULL);
|
|
|
|
if (format == NULL)
|
|
format = gimp_cairo_surface_get_format (surface);
|
|
width = cairo_image_surface_get_width (surface);
|
|
height = cairo_image_surface_get_height (surface);
|
|
|
|
return
|
|
gegl_buffer_linear_new_from_data (cairo_image_surface_get_data (surface),
|
|
format,
|
|
GEGL_RECTANGLE (0, 0, width, height),
|
|
cairo_image_surface_get_stride (surface),
|
|
(GDestroyNotify) cairo_surface_destroy,
|
|
cairo_surface_reference (surface));
|
|
}
|