switch font rendering to pangocairo
to make colour font rendering work
This commit is contained in:
parent
8412b8233c
commit
f53959b824
@ -22,6 +22,7 @@
|
|||||||
- add vips_image_[set|get]_array_double()
|
- add vips_image_[set|get]_array_double()
|
||||||
- add GIF load with libnsgif
|
- add GIF load with libnsgif
|
||||||
- add JPEG2000 load and save
|
- add JPEG2000 load and save
|
||||||
|
- add "rgba" flag to vips_text() to enable full colour text rendering
|
||||||
|
|
||||||
22/12/20 start 8.10.6
|
22/12/20 start 8.10.6
|
||||||
- don't seek on bad file descriptors [kleisauke]
|
- don't seek on bad file descriptors [kleisauke]
|
||||||
|
30
configure.ac
30
configure.ac
@ -1083,25 +1083,25 @@ fi
|
|||||||
VIPS_CFLAGS="$VIPS_CFLAGS $LIBWEBP_CFLAGS"
|
VIPS_CFLAGS="$VIPS_CFLAGS $LIBWEBP_CFLAGS"
|
||||||
VIPS_LIBS="$VIPS_LIBS $LIBWEBP_LIBS"
|
VIPS_LIBS="$VIPS_LIBS $LIBWEBP_LIBS"
|
||||||
|
|
||||||
# pangoft2
|
# pangocairo for text rendering
|
||||||
AC_ARG_WITH([pangoft2],
|
AC_ARG_WITH([pangocairo],
|
||||||
AS_HELP_STRING([--without-pangoft2],
|
AS_HELP_STRING([--without-pangocairo],
|
||||||
[build without pangoft2 (default: test)]))
|
[build without pangocairo (default: test)]))
|
||||||
|
|
||||||
if test x"$with_pangoft2" != x"no"; then
|
if test x"$with_pangocairo" != x"no"; then
|
||||||
PKG_CHECK_MODULES(PANGOFT2, pangoft2,
|
PKG_CHECK_MODULES(PANGOCAIRO, pangocairo,
|
||||||
[AC_DEFINE(HAVE_PANGOFT2,1,[define if you have pangoft2 installed.])
|
[AC_DEFINE(HAVE_PANGOCAIRO,1,[define if you have pangocairo installed.])
|
||||||
with_pangoft2=yes
|
with_pangocairo=yes
|
||||||
PACKAGES_USED="$PACKAGES_USED pangoft2"
|
PACKAGES_USED="$PACKAGES_USED pangocairo"
|
||||||
],
|
],
|
||||||
[AC_MSG_WARN([pangoft2 not found; disabling pangoft2 support])
|
[AC_MSG_WARN([pangocairo not found; disabling pangocairo support])
|
||||||
with_pangoft2=no
|
with_pangocairo=no
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
fi
|
fi
|
||||||
|
|
||||||
VIPS_CFLAGS="$VIPS_CFLAGS $PANGOFT2_CFLAGS"
|
VIPS_CFLAGS="$VIPS_CFLAGS $PANGOCAIRO_CFLAGS"
|
||||||
VIPS_LIBS="$VIPS_LIBS $PANGOFT2_LIBS"
|
VIPS_LIBS="$VIPS_LIBS $PANGOCAIRO_LIBS"
|
||||||
|
|
||||||
# look for TIFF with pkg-config ... fall back to our tester
|
# look for TIFF with pkg-config ... fall back to our tester
|
||||||
# pkgconfig support for libtiff starts with libtiff-4
|
# pkgconfig support for libtiff starts with libtiff-4
|
||||||
@ -1325,7 +1325,7 @@ use fftw3 for FFT: $with_fftw, \
|
|||||||
accelerate loops with orc: $with_orc, \
|
accelerate loops with orc: $with_orc, \
|
||||||
ICC profile support with lcms: $with_lcms, \
|
ICC profile support with lcms: $with_lcms, \
|
||||||
zlib: $with_zlib, \
|
zlib: $with_zlib, \
|
||||||
text rendering with pangoft2: $with_pangoft2, \
|
text rendering with pangocairo: $with_pangocairo, \
|
||||||
EXIF metadata support with libexif: $with_libexif, \
|
EXIF metadata support with libexif: $with_libexif, \
|
||||||
JPEG load/save with libjpeg: $with_jpeg, \
|
JPEG load/save with libjpeg: $with_jpeg, \
|
||||||
PNG load with libspng: $with_libspng, \
|
PNG load with libspng: $with_libspng, \
|
||||||
@ -1427,7 +1427,7 @@ accelerate loops with orc: $with_orc
|
|||||||
(requires orc-0.4.11 or later)
|
(requires orc-0.4.11 or later)
|
||||||
ICC profile support with lcms: $with_lcms
|
ICC profile support with lcms: $with_lcms
|
||||||
zlib: $with_zlib
|
zlib: $with_zlib
|
||||||
text rendering with pangoft2: $with_pangoft2
|
text rendering with pangocairo: $with_pangocairo
|
||||||
EXIF metadata support with libexif: $with_libexif
|
EXIF metadata support with libexif: $with_libexif
|
||||||
|
|
||||||
## File format support
|
## File format support
|
||||||
|
@ -116,9 +116,9 @@ vips_create_operation_init( void )
|
|||||||
extern GType vips_gaussmat_get_type( void );
|
extern GType vips_gaussmat_get_type( void );
|
||||||
extern GType vips_logmat_get_type( void );
|
extern GType vips_logmat_get_type( void );
|
||||||
extern GType vips_gaussnoise_get_type( void );
|
extern GType vips_gaussnoise_get_type( void );
|
||||||
#ifdef HAVE_PANGOFT2
|
#ifdef HAVE_PANGOCAIRO
|
||||||
extern GType vips_text_get_type( void );
|
extern GType vips_text_get_type( void );
|
||||||
#endif /*HAVE_PANGOFT2*/
|
#endif /*HAVE_PANGOCAIRO*/
|
||||||
extern GType vips_xyz_get_type( void );
|
extern GType vips_xyz_get_type( void );
|
||||||
extern GType vips_eye_get_type( void );
|
extern GType vips_eye_get_type( void );
|
||||||
extern GType vips_grey_get_type( void );
|
extern GType vips_grey_get_type( void );
|
||||||
@ -146,9 +146,9 @@ vips_create_operation_init( void )
|
|||||||
vips_gaussmat_get_type();
|
vips_gaussmat_get_type();
|
||||||
vips_logmat_get_type();
|
vips_logmat_get_type();
|
||||||
vips_gaussnoise_get_type();
|
vips_gaussnoise_get_type();
|
||||||
#ifdef HAVE_PANGOFT2
|
#ifdef HAVE_PANGOCAIRO
|
||||||
vips_text_get_type();
|
vips_text_get_type();
|
||||||
#endif /*HAVE_PANGOFT2*/
|
#endif /*HAVE_PANGOCAIRO*/
|
||||||
vips_xyz_get_type();
|
vips_xyz_get_type();
|
||||||
vips_eye_get_type();
|
vips_eye_get_type();
|
||||||
vips_grey_get_type();
|
vips_grey_get_type();
|
||||||
|
@ -32,6 +32,9 @@
|
|||||||
* - fitting could occasionally terminate early [levmorozov]
|
* - fitting could occasionally terminate early [levmorozov]
|
||||||
* 16/5/20 [keiviv]
|
* 16/5/20 [keiviv]
|
||||||
* - don't add fontfiles repeatedly
|
* - don't add fontfiles repeatedly
|
||||||
|
* 12/4/21
|
||||||
|
* - switch to cairo for text rendering
|
||||||
|
* - add rgba flag
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -74,11 +77,13 @@
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include <vips/vips.h>
|
#include <vips/vips.h>
|
||||||
|
#include <vips/internal.h>
|
||||||
|
|
||||||
#ifdef HAVE_PANGOFT2
|
#ifdef HAVE_PANGOCAIRO
|
||||||
|
|
||||||
|
#include <cairo.h>
|
||||||
#include <pango/pango.h>
|
#include <pango/pango.h>
|
||||||
#include <pango/pangoft2.h>
|
#include <pango/pangocairo.h>
|
||||||
|
|
||||||
#include "pcreate.h"
|
#include "pcreate.h"
|
||||||
|
|
||||||
@ -96,7 +101,6 @@ typedef struct _VipsText {
|
|||||||
char *fontfile;
|
char *fontfile;
|
||||||
gboolean rgba;
|
gboolean rgba;
|
||||||
|
|
||||||
FT_Bitmap bitmap;
|
|
||||||
PangoContext *context;
|
PangoContext *context;
|
||||||
PangoLayout *layout;
|
PangoLayout *layout;
|
||||||
|
|
||||||
@ -130,7 +134,6 @@ vips_text_dispose( GObject *gobject )
|
|||||||
|
|
||||||
VIPS_UNREF( text->layout );
|
VIPS_UNREF( text->layout );
|
||||||
VIPS_UNREF( text->context );
|
VIPS_UNREF( text->context );
|
||||||
VIPS_FREE( text->bitmap.buffer );
|
|
||||||
|
|
||||||
G_OBJECT_CLASS( vips_text_parent_class )->dispose( gobject );
|
G_OBJECT_CLASS( vips_text_parent_class )->dispose( gobject );
|
||||||
}
|
}
|
||||||
@ -187,8 +190,8 @@ vips_text_get_extents( VipsText *text, VipsRect *extents )
|
|||||||
PangoRectangle ink_rect;
|
PangoRectangle ink_rect;
|
||||||
PangoRectangle logical_rect;
|
PangoRectangle logical_rect;
|
||||||
|
|
||||||
pango_ft2_font_map_set_resolution(
|
pango_cairo_font_map_set_resolution(
|
||||||
PANGO_FT2_FONT_MAP( vips_text_fontmap ), text->dpi, text->dpi );
|
PANGO_CAIRO_FONT_MAP( vips_text_fontmap ), text->dpi );
|
||||||
|
|
||||||
VIPS_UNREF( text->layout );
|
VIPS_UNREF( text->layout );
|
||||||
if( !(text->layout = text_layout_new( text->context,
|
if( !(text->layout = text_layout_new( text->context,
|
||||||
@ -345,8 +348,8 @@ vips_text_build( VipsObject *object )
|
|||||||
VipsRect extents;
|
VipsRect extents;
|
||||||
int bands;
|
int bands;
|
||||||
VipsInterpretation interpretation;
|
VipsInterpretation interpretation;
|
||||||
FT_Pixel_Mode pixel_mode;
|
cairo_surface_t *surface;
|
||||||
int y;
|
cairo_t *cr;
|
||||||
|
|
||||||
if( VIPS_OBJECT_CLASS( vips_text_parent_class )->build( object ) )
|
if( VIPS_OBJECT_CLASS( vips_text_parent_class )->build( object ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -360,7 +363,7 @@ vips_text_build( VipsObject *object )
|
|||||||
g_mutex_lock( vips_text_lock );
|
g_mutex_lock( vips_text_lock );
|
||||||
|
|
||||||
if( !vips_text_fontmap )
|
if( !vips_text_fontmap )
|
||||||
vips_text_fontmap = pango_ft2_font_map_new();
|
vips_text_fontmap = pango_cairo_font_map_new();
|
||||||
if( !vips_text_fontfiles )
|
if( !vips_text_fontfiles )
|
||||||
vips_text_fontfiles =
|
vips_text_fontfiles =
|
||||||
g_hash_table_new( g_str_hash, g_str_equal );
|
g_hash_table_new( g_str_hash, g_str_equal );
|
||||||
@ -368,6 +371,7 @@ vips_text_build( VipsObject *object )
|
|||||||
text->context = pango_font_map_create_context(
|
text->context = pango_font_map_create_context(
|
||||||
PANGO_FONT_MAP( vips_text_fontmap ) );
|
PANGO_FONT_MAP( vips_text_fontmap ) );
|
||||||
|
|
||||||
|
/*
|
||||||
if( text->fontfile &&
|
if( text->fontfile &&
|
||||||
!g_hash_table_lookup( vips_text_fontfiles, text->fontfile ) ) {
|
!g_hash_table_lookup( vips_text_fontfiles, text->fontfile ) ) {
|
||||||
if( !FcConfigAppFontAddFile( NULL,
|
if( !FcConfigAppFontAddFile( NULL,
|
||||||
@ -382,6 +386,7 @@ vips_text_build( VipsObject *object )
|
|||||||
text->fontfile,
|
text->fontfile,
|
||||||
g_strdup( text->fontfile ) );
|
g_strdup( text->fontfile ) );
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
/* If our caller set height and not dpi, we adjust dpi until
|
/* If our caller set height and not dpi, we adjust dpi until
|
||||||
* we get a fit.
|
* we get a fit.
|
||||||
@ -406,54 +411,56 @@ vips_text_build( VipsObject *object )
|
|||||||
if( text->rgba ) {
|
if( text->rgba ) {
|
||||||
interpretation = VIPS_INTERPRETATION_sRGB;
|
interpretation = VIPS_INTERPRETATION_sRGB;
|
||||||
bands = 4;
|
bands = 4;
|
||||||
pixel_mode = FT_PIXEL_MODE_BGRA;
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
interpretation = VIPS_INTERPRETATION_MULTIBAND;
|
interpretation = VIPS_INTERPRETATION_MULTIBAND;
|
||||||
bands = 1;
|
bands = 1;
|
||||||
pixel_mode = FT_PIXEL_MODE_GRAY;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
text->bitmap.width = extents.width;
|
|
||||||
text->bitmap.pitch = (bands * text->bitmap.width + 3) & ~3;
|
|
||||||
text->bitmap.rows = extents.height;
|
|
||||||
if( !(text->bitmap.buffer =
|
|
||||||
VIPS_ARRAY( NULL,
|
|
||||||
text->bitmap.pitch * text->bitmap.rows, VipsPel )) ) {
|
|
||||||
g_mutex_unlock( vips_text_lock );
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
text->bitmap.num_grays = 256;
|
|
||||||
text->bitmap.pixel_mode = pixel_mode;
|
|
||||||
memset( text->bitmap.buffer, 0x00,
|
|
||||||
text->bitmap.pitch * text->bitmap.rows );
|
|
||||||
|
|
||||||
pango_ft2_render_layout( &text->bitmap, text->layout,
|
|
||||||
-extents.left, -extents.top );
|
|
||||||
|
|
||||||
/* Set DPI as pixels/mm.
|
/* Set DPI as pixels/mm.
|
||||||
*/
|
*/
|
||||||
vips_image_init_fields( create->out,
|
vips_image_init_fields( create->out,
|
||||||
text->bitmap.width, text->bitmap.rows, bands,
|
extents.width, extents.height, bands,
|
||||||
VIPS_FORMAT_UCHAR, VIPS_CODING_NONE,
|
VIPS_FORMAT_UCHAR, VIPS_CODING_NONE,
|
||||||
interpretation, text->dpi / 25.4, text->dpi / 25.4 );
|
interpretation, text->dpi / 25.4, text->dpi / 25.4 );
|
||||||
|
|
||||||
g_mutex_unlock( vips_text_lock );
|
|
||||||
|
|
||||||
vips_image_pipelinev( create->out, VIPS_DEMAND_STYLE_ANY, NULL );
|
vips_image_pipelinev( create->out, VIPS_DEMAND_STYLE_ANY, NULL );
|
||||||
create->out->Xoffset = extents.left;
|
create->out->Xoffset = extents.left;
|
||||||
create->out->Yoffset = extents.top;
|
create->out->Yoffset = extents.top;
|
||||||
|
|
||||||
if( text->rgba ) {
|
if( vips_image_write_prepare( create->out ) ) {
|
||||||
/* Convert premultiplied BGRA to RGBA.
|
g_mutex_unlock( vips_text_lock );
|
||||||
*/
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
for( y = 0; y < text->bitmap.rows; y++ )
|
surface = cairo_image_surface_create_for_data(
|
||||||
if( vips_image_write_line( create->out, y,
|
VIPS_IMAGE_ADDR( create->out, 0, 0 ),
|
||||||
(VipsPel *) text->bitmap.buffer +
|
text->rgba ? CAIRO_FORMAT_ARGB32 : CAIRO_FORMAT_A8,
|
||||||
y * text->bitmap.pitch ) )
|
create->out->Xsize, create->out->Ysize,
|
||||||
return( -1 );
|
VIPS_IMAGE_SIZEOF_LINE( create->out ) );
|
||||||
|
cr = cairo_create( surface );
|
||||||
|
cairo_surface_destroy( surface );
|
||||||
|
|
||||||
|
cairo_translate( cr, -extents.left, -extents.top );
|
||||||
|
|
||||||
|
pango_cairo_show_layout( cr, text->layout );
|
||||||
|
|
||||||
|
cairo_destroy( cr );
|
||||||
|
|
||||||
|
g_mutex_unlock( vips_text_lock );
|
||||||
|
|
||||||
|
if( text->rgba ) {
|
||||||
|
int y;
|
||||||
|
|
||||||
|
/* Cairo makes pre-multipled BRGA -- we must byteswap and
|
||||||
|
* unpremultiply.
|
||||||
|
*/
|
||||||
|
for( y = 0; y < create->out->Ysize; y++ )
|
||||||
|
vips__premultiplied_bgra2rgba(
|
||||||
|
(guint32 *)
|
||||||
|
VIPS_IMAGE_ADDR( create->out, 0, y ),
|
||||||
|
create->out->Xsize );
|
||||||
|
}
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
@ -569,11 +576,10 @@ vips_text_init( VipsText *text )
|
|||||||
{
|
{
|
||||||
text->align = VIPS_ALIGN_LOW;
|
text->align = VIPS_ALIGN_LOW;
|
||||||
text->dpi = 72;
|
text->dpi = 72;
|
||||||
text->bitmap.buffer = NULL;
|
|
||||||
VIPS_SETSTR( text->font, "sans 12" );
|
VIPS_SETSTR( text->font, "sans 12" );
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /*HAVE_PANGOFT2*/
|
#endif /*HAVE_PANGOCAIRO*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* vips_text:
|
* vips_text:
|
||||||
|
Loading…
Reference in New Issue
Block a user