Graphviz 13.0.0~dev.20250402.0402
Loading...
Searching...
No Matches
gvtextlayout_pango.c
Go to the documentation of this file.
1/*************************************************************************
2 * Copyright (c) 2011 AT&T Intellectual Property
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Eclipse Public License v1.0
5 * which accompanies this distribution, and is available at
6 * https://www.eclipse.org/legal/epl-v10.html
7 *
8 * Contributors: Details at https://graphviz.org
9 *************************************************************************/
10
11#include "config.h"
12
13#include <assert.h>
14#include <limits.h>
15#include <stdbool.h>
16#include <stdlib.h>
17#include <string.h>
18#include <gvc/gvplugin_render.h>
19#include <common/utils.h>
21#include <util/agxbuf.h>
22#include <util/alloc.h>
23#include <util/gv_math.h>
24
25#include <pango/pangocairo.h>
26#include "gvgetfontlist.h"
27#ifdef HAVE_PANGO_FC_FONT_LOCK_FACE
28#include <pango/pangofc-font.h>
29#endif
30
31static void pango_free_layout (void *layout)
32{
33 g_object_unref(layout);
34}
35
37{
38 agxbuf buf = {0};
39 agxbprint(&buf, "%s,", pa->family);
40 if (pa->weight) {
41 agxbprint(&buf, " %s", pa->weight);
42 }
43 if (pa->stretch) {
44 agxbprint(&buf, " %s", pa->stretch);
45 }
46 if (pa->style) {
47 agxbprint(&buf, " %s", pa->style);
48 }
49 return agxbdisown(&buf);
50}
51
52#define FONT_DPI 96.
53
54#define ENABLE_PANGO_MARKUP
55
56// wrapper to handle difference in calling conventions between `agxbput` and
57// `xml_escape`’s `cb`
58static int agxbput_int(void *buffer, const char *s) {
59 size_t len = agxbput(buffer, s);
60 assert(len <= INT_MAX);
61 return (int)len;
62}
63
64static bool pango_textlayout(textspan_t * span, char **fontpath)
65{
66 static agxbuf buf; // returned in fontpath, only good until next call
67 static PangoFontMap *fontmap;
68 static PangoContext *context;
69 static PangoFontDescription *desc;
70 static char *fontname;
71 static double fontsize;
72 static gv_font_map* gv_fmap;
73 char *fnt, *psfnt = NULL;
74 PangoFont *font;
75#ifdef ENABLE_PANGO_MARKUP
76 PangoAttrList *attrs;
77 GError *error = NULL;
78 int flags;
79#endif
80 bool text_needs_free = false;
81 char *text;
82
83 if (!context) {
84 fontmap = pango_cairo_font_map_new();
85 gv_fmap = get_font_mapping(fontmap);
86 context = pango_font_map_create_context (fontmap);
87 cairo_font_options_t* options = cairo_font_options_create();
88 cairo_font_options_set_antialias(options,CAIRO_ANTIALIAS_GRAY);
89 cairo_font_options_set_hint_style(options,CAIRO_HINT_STYLE_FULL);
90 cairo_font_options_set_hint_metrics(options,CAIRO_HINT_METRICS_ON);
91 cairo_font_options_set_subpixel_order(options,CAIRO_SUBPIXEL_ORDER_BGR);
92 pango_cairo_context_set_font_options(context, options);
93 pango_cairo_context_set_resolution(context, FONT_DPI);
94 cairo_font_options_destroy(options);
95 g_object_unref(fontmap);
96 }
97
98 if (!fontname || strcmp(fontname, span->font->name) != 0 ||
99 !is_exactly_equal(fontsize, span->font->size)) {
100
101 /* check if the conversion to Pango units below will overflow */
102 if (INT_MAX / PANGO_SCALE < span->font->size) {
103 return false;
104 }
105
106 free(fontname);
107 fontname = gv_strdup(span->font->name);
108 fontsize = span->font->size;
109 pango_font_description_free (desc);
110
112 bool psfnt_needs_free = false;
113 if (pA) {
114 psfnt = fnt = gv_fmap[pA->xfig_code].gv_font;
115 if(!psfnt) {
116 psfnt = fnt = pango_psfontResolve (pA);
117 psfnt_needs_free = true;
118 }
119 }
120 else
121 fnt = fontname;
122
123 desc = pango_font_description_from_string(fnt);
124 /* all text layout is done at a scale of FONT_DPI (nominaly 96.) */
125 pango_font_description_set_size (desc, (int)(fontsize * PANGO_SCALE));
126
127 if (fontpath && (font = pango_font_map_load_font(fontmap, context, desc))) { /* -v support */
128 const char *fontclass = G_OBJECT_CLASS_NAME(G_OBJECT_GET_CLASS(font));
129
130 agxbclear(&buf);
131 if (psfnt) {
132 agxbprint(&buf, "(ps:pango %s) ", psfnt);
133 }
134 agxbprint(&buf, "(%s) ", fontclass);
135#ifdef HAVE_PANGO_FC_FONT_LOCK_FACE
136 if (strcmp(fontclass, "PangoCairoFcFont") == 0) {
137 PangoFcFont *fcfont = PANGO_FC_FONT(font);
138// `pango_fc_font_lock_face` arrived in Pango 1.4 and then was deprecated in
139// 1.44. Its replacement is `pango_font_get_hb_font`. However this replacement
140// does not seem to provide access to the underlying FreeType font information
141// we want. Since this code path is only enabled when `pango_fc_font_lock_face`
142// is available, suppress deprecation warnings and we will have to deal with the
143// loss of this informational output altogether when Pango finally removes it.
144#if __GNUC__
145#pragma GCC diagnostic push
146#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
147#endif
148 FT_Face face = pango_fc_font_lock_face(fcfont);
149#ifdef __GNUC__
150#pragma GCC diagnostic pop
151#endif
152 if (face) {
153 agxbprint(&buf, "\"%s, %s\" ", face->family_name, face->style_name);
154
155 FT_Stream stream = face->stream;
156 if (stream) {
157 FT_StreamDesc streamdesc = stream->pathname;
158 if (streamdesc.pointer)
159 agxbput(&buf, streamdesc.pointer);
160 else
161 agxbput(&buf, "*no pathname available*");
162 }
163 else
164 agxbput(&buf, "*no stream available*");
165 }
166#if __GNUC__
167#pragma GCC diagnostic push
168#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
169#endif
170 pango_fc_font_unlock_face(fcfont);
171#ifdef __GNUC__
172#pragma GCC diagnostic pop
173#endif
174 }
175 else
176#endif
177 {
178 PangoFontDescription *tdesc = pango_font_describe(font);
179 char *tfont = pango_font_description_to_string(tdesc);
180 agxbprint(&buf, "\"%s\" ", tfont);
181 g_free(tfont);
182 }
183 *fontpath = agxbuse(&buf);
184 }
185 if (psfnt_needs_free) {
186 free(psfnt);
187 }
188 }
189
190#ifdef ENABLE_PANGO_MARKUP
191 if (span->font && (flags = span->font->flags)) {
192 agxbuf xb = {0};
193
194 agxbput(&xb,"<span");
195
196 if (flags & HTML_BF)
197 agxbput(&xb," weight=\"bold\"");
198 if (flags & HTML_IF)
199 agxbput(&xb," style=\"italic\"");
200 if (flags & HTML_UL)
201 agxbput(&xb," underline=\"single\"");
202 if (flags & HTML_S)
203 agxbput(&xb," strikethrough=\"true\"");
204 agxbput (&xb,">");
205
206 if (flags & HTML_SUP)
207 agxbput(&xb,"<sup>");
208 if (flags & HTML_SUB)
209 agxbput(&xb,"<sub>");
210
211 const xml_flags_t xml_flags = {.raw = 1, .dash = 1, .nbsp = 1};
212 xml_escape(span->str, xml_flags, agxbput_int, &xb);
213
214 if (flags & HTML_SUB)
215 agxbput(&xb,"</sub>");
216 if (flags & HTML_SUP)
217 agxbput(&xb,"</sup>");
218
219 agxbput (&xb,"</span>");
220 if (!pango_parse_markup (agxbuse(&xb), -1, 0, &attrs, &text, NULL, &error)) {
221 fprintf (stderr, "Error - pango_parse_markup: %s\n", error->message);
222 text = span->str;
223 attrs = NULL;
224 } else {
225 text_needs_free = true;
226 }
227 agxbfree (&xb);
228 }
229 else {
230 text = span->str;
231 attrs = NULL;
232 }
233#else
234 text = span->str;
235#endif
236
237 PangoLayout *layout = pango_layout_new (context);
238 span->layout = layout; /* layout free with textspan - see labels.c */
239 span->free_layout = pango_free_layout; /* function for freeing pango layout */
240
241 pango_layout_set_text (layout, text, -1);
242 pango_layout_set_font_description (layout, desc);
243#ifdef ENABLE_PANGO_MARKUP
244 if (attrs)
245 pango_layout_set_attributes (layout, attrs);
246#endif
247
248 PangoRectangle logical_rect;
249 pango_layout_get_extents (layout, NULL, &logical_rect);
250
251 /* if pango doesn't like the font then it sets width=0 but height = garbage */
252 if (logical_rect.width == 0)
253 logical_rect.height = 0;
254
255 const double textlayout_scale = POINTS_PER_INCH / (FONT_DPI * PANGO_SCALE);
256 span->size.x = logical_rect.width * textlayout_scale;
257 span->size.y = logical_rect.height * textlayout_scale;
258
259 /* The y offset from baseline to 0,0 of the bitmap representation */
260 span->yoffset_layout = pango_layout_get_baseline (layout) * textlayout_scale;
261
262 /* The distance below midline for y centering of text strings */
263 span->yoffset_centerline = 0.05 * span->font->size;
264
265 const bool rc = logical_rect.width != 0 || strcmp(text, "") == 0;
266 if (text_needs_free) {
267 g_free(text);
268 }
269 return rc;
270}
271
275
277 {0, "textlayout", 10, &pango_textlayout_engine, NULL},
278 {0, NULL, 0, NULL, NULL}
279};
static void agxbfree(agxbuf *xb)
free any malloced resources
Definition agxbuf.h:78
static int agxbprint(agxbuf *xb, const char *fmt,...)
Printf-style output to an agxbuf.
Definition agxbuf.h:234
static void agxbclear(agxbuf *xb)
resets pointer to data
Definition agxbuf.h:294
static WUR char * agxbuse(agxbuf *xb)
Definition agxbuf.h:307
static char * agxbdisown(agxbuf *xb)
Definition agxbuf.h:327
Memory allocation wrappers that exit on failure.
static char * gv_strdup(const char *original)
Definition alloc.h:101
static int flags
Definition gc.c:61
#define POINTS_PER_INCH
Definition geom.h:58
static double len(glCompPoint p)
Definition glutils.c:150
void free(void *)
node NULL
Definition grammar.y:163
Arithmetic helper functions.
static bool is_exactly_equal(double a, double b)
are two values precisely the same?
Definition gv_math.h:44
agxbput(xb, staging)
gv_font_map * get_font_mapping(PangoFontMap *pfm)
static gvtextlayout_engine_t pango_textlayout_engine
static int agxbput_int(void *buffer, const char *s)
gvplugin_installed_t gvtextlayout_pango_types[]
static char * pango_psfontResolve(PostscriptAlias *pa)
#define FONT_DPI
static bool pango_textlayout(textspan_t *span, char **fontpath)
static void pango_free_layout(void *layout)
$2 font
Definition htmlparse.y:300
table Syntax error
Definition htmlparse.y:294
static int layout(graph_t *g, layout_info *infop)
Definition layout.c:814
ingroup plugin_api
Definition gvplugin.h:35
Definition gvpr.c:60
double x
Definition geom.h:29
double y
Definition geom.h:29
char * name
Definition textspan.h:54
PostscriptAlias * postscript_alias
Definition textspan.h:56
unsigned int flags
Definition textspan.h:58
double size
Definition textspan.h:57
double yoffset_layout
Definition textspan.h:69
char * str
Definition textspan.h:65
void * layout
Definition textspan.h:67
pointf size
Definition textspan.h:70
textfont_t * font
Definition textspan.h:66
double yoffset_centerline
Definition textspan.h:69
void(* free_layout)(void *layout)
Definition textspan.h:68
unsigned raw
Definition utils.h:40
#define HTML_IF
Definition textspan.h:30
#define HTML_UL
Definition textspan.h:31
#define HTML_BF
Definition textspan.h:29
#define HTML_SUP
Definition textspan.h:32
#define HTML_S
Definition textspan.h:34
#define HTML_SUB
Definition textspan.h:33
Definition grammar.c:93
int xml_escape(const char *s, xml_flags_t flags, int(*cb)(void *state, const char *s), void *state)
Definition xml.c:179