1 /* GDK - The GIMP Drawing Kit
2  * Copyright (C) 2000 Red Hat, Inc.
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library. If not, see <http://www.gnu.org/licenses/>.
16  */
17 
18 #include "config.h"
19 
20 #include "gdkpango.h"
21 
22 #include "gdkscreen.h"
23 #include "gdkintl.h"
24 
25 #include <math.h>
26 #include <pango/pangocairo.h>
27 
28 
29 /**
30  * SECTION:pango_interaction
31  * @Short_description: Using Pango in GDK
32  * @Title: Pango Interaction
33  *
34  * Pango is the text layout system used by GDK and GTK+. The functions
35  * and types in this section are used to obtain clip regions for
36  * #PangoLayouts, and to get #PangoContexts that can be used with
37  * GDK.
38  *
39  * Creating a #PangoLayout object is the first step in rendering text,
40  * and requires getting a handle to a #PangoContext. For GTK+ programs,
41  * you’ll usually want to use gtk_widget_get_pango_context(), or
42  * gtk_widget_create_pango_layout(), rather than using the lowlevel
43  * gdk_pango_context_get_for_screen(). Once you have a #PangoLayout, you
44  * can set the text and attributes of it with Pango functions like
45  * pango_layout_set_text() and get its size with pango_layout_get_size().
46  * (Note that Pango uses a fixed point system internally, so converting
47  * between Pango units and pixels using [PANGO_SCALE][PANGO-SCALE-CAPS]
48  * or the PANGO_PIXELS() macro.)
49  *
50  * Rendering a Pango layout is done most simply with pango_cairo_show_layout();
51  * you can also draw pieces of the layout with pango_cairo_show_layout_line().
52  *
53  * ## Draw transformed text with Pango and cairo ## {#rotated-example}
54  *
55  * |[<!-- language="C" -->
56  * #define RADIUS 100
57  * #define N_WORDS 10
58  * #define FONT "Sans Bold 18"
59  *
60  * PangoContext *context;
61  * PangoLayout *layout;
62  * PangoFontDescription *desc;
63  *
64  * double radius;
65  * int width, height;
66  * int i;
67  *
68  * // Set up a transformation matrix so that the user space coordinates for
69  * // where we are drawing are [-RADIUS, RADIUS], [-RADIUS, RADIUS]
70  * // We first center, then change the scale
71  *
72  * width = gdk_window_get_width (window);
73  * height = gdk_window_get_height (window);
74  * radius = MIN (width, height) / 2.;
75  *
76  * cairo_translate (cr,
77  *                  radius + (width - 2 * radius) / 2,
78  *                  radius + (height - 2 * radius) / 2);
79  *                  cairo_scale (cr, radius / RADIUS, radius / RADIUS);
80  *
81  * // Create a PangoLayout, set the font and text
82  * context = gdk_pango_context_get_for_screen (screen);
83  * layout = pango_layout_new (context);
84  * pango_layout_set_text (layout, "Text", -1);
85  * desc = pango_font_description_from_string (FONT);
86  * pango_layout_set_font_description (layout, desc);
87  * pango_font_description_free (desc);
88  *
89  * // Draw the layout N_WORDS times in a circle
90  * for (i = 0; i < N_WORDS; i++)
91  *   {
92  *     double red, green, blue;
93  *     double angle = 2 * G_PI * i / n_words;
94  *
95  *     cairo_save (cr);
96  *
97  *     // Gradient from red at angle == 60 to blue at angle == 300
98  *     red = (1 + cos (angle - 60)) / 2;
99  *     green = 0;
100  *     blue = 1 - red;
101  *
102  *     cairo_set_source_rgb (cr, red, green, blue);
103  *     cairo_rotate (cr, angle);
104  *
105  *     // Inform Pango to re-layout the text with the new transformation matrix
106  *     pango_cairo_update_layout (cr, layout);
107  *
108  *     pango_layout_get_size (layout, &width, &height);
109  *
110  *     cairo_move_to (cr, - width / 2 / PANGO_SCALE, - DEFAULT_TEXT_RADIUS);
111  *     pango_cairo_show_layout (cr, layout);
112  *
113  *     cairo_restore (cr);
114  *   }
115  *
116  * g_object_unref (layout);
117  * g_object_unref (context);
118  * ]|
119  *
120  * ## Output of the [example][rotated-example] above.
121  *
122  * ![](rotated-text.png)
123  */
124 
125 /* Get a clip region to draw only part of a layout. index_ranges
126  * contains alternating range starts/stops. The region is the
127  * region which contains the given ranges, i.e. if you draw with the
128  * region as clip, only the given ranges are drawn.
129  */
130 static cairo_region_t*
layout_iter_get_line_clip_region(PangoLayoutIter * iter,gint x_origin,gint y_origin,const gint * index_ranges,gint n_ranges)131 layout_iter_get_line_clip_region (PangoLayoutIter *iter,
132 				  gint             x_origin,
133 				  gint             y_origin,
134 				  const gint      *index_ranges,
135 				  gint             n_ranges)
136 {
137   PangoLayoutLine *line;
138   cairo_region_t *clip_region;
139   PangoRectangle logical_rect;
140   gint baseline;
141   gint i;
142 
143   line = pango_layout_iter_get_line_readonly (iter);
144 
145   clip_region = cairo_region_create ();
146 
147   pango_layout_iter_get_line_extents (iter, NULL, &logical_rect);
148   baseline = pango_layout_iter_get_baseline (iter);
149 
150   i = 0;
151   while (i < n_ranges)
152     {
153       gint *pixel_ranges = NULL;
154       gint n_pixel_ranges = 0;
155       gint j;
156 
157       /* Note that get_x_ranges returns layout coordinates
158        */
159       if (index_ranges[i*2+1] >= line->start_index &&
160 	  index_ranges[i*2] < line->start_index + line->length)
161 	pango_layout_line_get_x_ranges (line,
162 					index_ranges[i*2],
163 					index_ranges[i*2+1],
164 					&pixel_ranges, &n_pixel_ranges);
165 
166       for (j = 0; j < n_pixel_ranges; j++)
167         {
168           GdkRectangle rect;
169 	  int x_off, y_off;
170 
171           x_off = PANGO_PIXELS (pixel_ranges[2*j] - logical_rect.x);
172 	  y_off = PANGO_PIXELS (baseline - logical_rect.y);
173 
174           rect.x = x_origin + x_off;
175           rect.y = y_origin - y_off;
176           rect.width = PANGO_PIXELS (pixel_ranges[2*j + 1] - logical_rect.x) - x_off;
177           rect.height = PANGO_PIXELS (baseline - logical_rect.y + logical_rect.height) - y_off;
178 
179           cairo_region_union_rectangle (clip_region, &rect);
180         }
181 
182       g_free (pixel_ranges);
183       ++i;
184     }
185   return clip_region;
186 }
187 
188 /**
189  * gdk_pango_layout_line_get_clip_region: (skip)
190  * @line: a #PangoLayoutLine
191  * @x_origin: X pixel where you intend to draw the layout line with this clip
192  * @y_origin: baseline pixel where you intend to draw the layout line with this clip
193  * @index_ranges: (array): array of byte indexes into the layout,
194  *     where even members of array are start indexes and odd elements
195  *     are end indexes
196  * @n_ranges: number of ranges in @index_ranges, i.e. half the size of @index_ranges
197  *
198  * Obtains a clip region which contains the areas where the given
199  * ranges of text would be drawn. @x_origin and @y_origin are the top left
200  * position of the layout. @index_ranges
201  * should contain ranges of bytes in the layout’s text. The clip
202  * region will include space to the left or right of the line (to the
203  * layout bounding box) if you have indexes above or below the indexes
204  * contained inside the line. This is to draw the selection all the way
205  * to the side of the layout. However, the clip region is in line coordinates,
206  * not layout coordinates.
207  *
208  * Note that the regions returned correspond to logical extents of the text
209  * ranges, not ink extents. So the drawn line may in fact touch areas out of
210  * the clip region.  The clip region is mainly useful for highlightling parts
211  * of text, such as when text is selected.
212  *
213  * Returns: a clip region containing the given ranges
214  **/
215 cairo_region_t*
gdk_pango_layout_line_get_clip_region(PangoLayoutLine * line,gint x_origin,gint y_origin,const gint * index_ranges,gint n_ranges)216 gdk_pango_layout_line_get_clip_region (PangoLayoutLine *line,
217                                        gint             x_origin,
218                                        gint             y_origin,
219                                        const gint      *index_ranges,
220                                        gint             n_ranges)
221 {
222   cairo_region_t *clip_region;
223   PangoLayoutIter *iter;
224 
225   g_return_val_if_fail (line != NULL, NULL);
226   g_return_val_if_fail (index_ranges != NULL, NULL);
227 
228   iter = pango_layout_get_iter (line->layout);
229   while (pango_layout_iter_get_line_readonly (iter) != line)
230     pango_layout_iter_next_line (iter);
231 
232   clip_region = layout_iter_get_line_clip_region(iter, x_origin, y_origin, index_ranges, n_ranges);
233 
234   pango_layout_iter_free (iter);
235 
236   return clip_region;
237 }
238 
239 /**
240  * gdk_pango_layout_get_clip_region: (skip)
241  * @layout: a #PangoLayout
242  * @x_origin: X pixel where you intend to draw the layout with this clip
243  * @y_origin: Y pixel where you intend to draw the layout with this clip
244  * @index_ranges: array of byte indexes into the layout, where even members of array are start indexes and odd elements are end indexes
245  * @n_ranges: number of ranges in @index_ranges, i.e. half the size of @index_ranges
246  *
247  * Obtains a clip region which contains the areas where the given ranges
248  * of text would be drawn. @x_origin and @y_origin are the top left point
249  * to center the layout. @index_ranges should contain
250  * ranges of bytes in the layout’s text.
251  *
252  * Note that the regions returned correspond to logical extents of the text
253  * ranges, not ink extents. So the drawn layout may in fact touch areas out of
254  * the clip region.  The clip region is mainly useful for highlightling parts
255  * of text, such as when text is selected.
256  *
257  * Returns: a clip region containing the given ranges
258  **/
259 cairo_region_t*
gdk_pango_layout_get_clip_region(PangoLayout * layout,gint x_origin,gint y_origin,const gint * index_ranges,gint n_ranges)260 gdk_pango_layout_get_clip_region (PangoLayout *layout,
261                                   gint         x_origin,
262                                   gint         y_origin,
263                                   const gint  *index_ranges,
264                                   gint         n_ranges)
265 {
266   PangoLayoutIter *iter;
267   cairo_region_t *clip_region;
268 
269   g_return_val_if_fail (PANGO_IS_LAYOUT (layout), NULL);
270   g_return_val_if_fail (index_ranges != NULL, NULL);
271 
272   clip_region = cairo_region_create ();
273 
274   iter = pango_layout_get_iter (layout);
275 
276   do
277     {
278       PangoRectangle logical_rect;
279       cairo_region_t *line_region;
280       gint baseline;
281 
282       pango_layout_iter_get_line_extents (iter, NULL, &logical_rect);
283       baseline = pango_layout_iter_get_baseline (iter);
284 
285       line_region = layout_iter_get_line_clip_region(iter,
286 						     x_origin + PANGO_PIXELS (logical_rect.x),
287 						     y_origin + PANGO_PIXELS (baseline),
288 						     index_ranges,
289 						     n_ranges);
290 
291       cairo_region_union (clip_region, line_region);
292       cairo_region_destroy (line_region);
293     }
294   while (pango_layout_iter_next_line (iter));
295 
296   pango_layout_iter_free (iter);
297 
298   return clip_region;
299 }
300 
301 /**
302  * gdk_pango_context_get:
303  *
304  * Creates a #PangoContext for the default GDK screen.
305  *
306  * The context must be freed when you’re finished with it.
307  *
308  * When using GTK+, normally you should use gtk_widget_get_pango_context()
309  * instead of this function, to get the appropriate context for
310  * the widget you intend to render text onto.
311  *
312  * The newly created context will have the default font options (see
313  * #cairo_font_options_t) for the default screen; if these options
314  * change it will not be updated. Using gtk_widget_get_pango_context()
315  * is more convenient if you want to keep a context around and track
316  * changes to the screen’s font rendering settings.
317  *
318  * Returns: (transfer full): a new #PangoContext for the default display
319  **/
320 PangoContext *
gdk_pango_context_get(void)321 gdk_pango_context_get (void)
322 {
323   return gdk_pango_context_get_for_display (gdk_display_get_default ());
324 }
325 
326 /**
327  * gdk_pango_context_get_for_screen:
328  * @screen: the #GdkScreen for which the context is to be created.
329  *
330  * Creates a #PangoContext for @screen.
331  *
332  * The context must be freed when you’re finished with it.
333  *
334  * When using GTK+, normally you should use gtk_widget_get_pango_context()
335  * instead of this function, to get the appropriate context for
336  * the widget you intend to render text onto.
337  *
338  * The newly created context will have the default font options
339  * (see #cairo_font_options_t) for the screen; if these options
340  * change it will not be updated. Using gtk_widget_get_pango_context()
341  * is more convenient if you want to keep a context around and track
342  * changes to the screen’s font rendering settings.
343  *
344  * Returns: (transfer full): a new #PangoContext for @screen
345  *
346  * Since: 2.2
347  **/
348 PangoContext *
gdk_pango_context_get_for_screen(GdkScreen * screen)349 gdk_pango_context_get_for_screen (GdkScreen *screen)
350 {
351   PangoFontMap *fontmap;
352   PangoContext *context;
353   const cairo_font_options_t *options;
354   double dpi;
355 
356   g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
357 
358   fontmap = pango_cairo_font_map_get_default ();
359   context = pango_font_map_create_context (fontmap);
360 
361   options = gdk_screen_get_font_options (screen);
362   pango_cairo_context_set_font_options (context, options);
363 
364   dpi = gdk_screen_get_resolution (screen);
365   pango_cairo_context_set_resolution (context, dpi);
366 
367   return context;
368 }
369 
370 /**
371  * gdk_pango_context_get_for_display:
372  * @display: the #GdkDisplay for which the context is to be created
373  *
374  * Creates a #PangoContext for @display.
375  *
376  * The context must be freed when you’re finished with it.
377  *
378  * When using GTK+, normally you should use gtk_widget_get_pango_context()
379  * instead of this function, to get the appropriate context for
380  * the widget you intend to render text onto.
381  *
382  * The newly created context will have the default font options
383  * (see #cairo_font_options_t) for the display; if these options
384  * change it will not be updated. Using gtk_widget_get_pango_context()
385  * is more convenient if you want to keep a context around and track
386  * changes to the font rendering settings.
387  *
388  * Returns: (transfer full): a new #PangoContext for @display
389  *
390  * Since: 3.22
391  */
392 PangoContext *
gdk_pango_context_get_for_display(GdkDisplay * display)393 gdk_pango_context_get_for_display (GdkDisplay *display)
394 {
395   g_return_val_if_fail (GDK_IS_DISPLAY (display), NULL);
396 
397   return gdk_pango_context_get_for_screen (gdk_display_get_default_screen (display));
398 }
399