/* Pango
* pangocairo-render.c: Rendering routines to Cairo surfaces
*
* Copyright (C) 2004 Red Hat, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
/**
* SECTION:pangocairo
* @short_description:Font handling and rendering with Cairo
* @title:Cairo Fonts and Rendering
*
* The Cairo library is a
* vector graphics library with a powerful rendering model. It has such
* features as anti-aliased primitives, alpha-compositing, and
* gradients. Multiple backends for Cairo are available, to allow
* rendering to images, to PDF files, and to the screen on X and on other
* windowing systems. The functions in this section allow using Pango
* to render to Cairo surfaces.
*
* Using Pango with Cairo is straightforward. A #PangoContext created
* with pango_cairo_font_map_create_context() can be used on any
* Cairo context (cairo_t), but needs to be updated to match the
* current transformation matrix and target surface of the Cairo context
* using pango_cairo_update_context(). The convenience functions
* pango_cairo_create_layout() and pango_cairo_update_layout() handle
* the common case where the program doesn't need to manipulate the
* properties of the #PangoContext.
*
* When you get the metrics of a layout or of a piece of a layout using
* functions such as pango_layout_get_extents(), the reported metrics
* are in user-space coordinates. If a piece of text is 10 units long,
* and you call cairo_scale (cr, 2.0), it still is more-or-less 10
* units long. However, the results will be affected by hinting
* (that is, the process of adjusting the text to look good on the
* pixel grid), so you shouldn't assume they are completely independent
* of the current transformation matrix. Note that the basic metrics
* functions in Pango report results in integer Pango units. To get
* to the floating point units used in Cairo divide by %PANGO_SCALE.
*
* ## Using Pango with Cairo ## {#rotated-example}
*
* |[
* #include
* #include
*
* static void
* draw_text (cairo_t *cr)
* {
* #define RADIUS 150
* #define N_WORDS 10
* #define FONT "Sans Bold 27"
*
* PangoLayout *layout;
* PangoFontDescription *desc;
* int i;
*
* /* Center coordinates on the middle of the region we are drawing
* */
* cairo_translate (cr, RADIUS, RADIUS);
*
* /* Create a PangoLayout, set the font and text */
* layout = pango_cairo_create_layout (cr);
*
* pango_layout_set_text (layout, "Text", -1);
* desc = pango_font_description_from_string (FONT);
* pango_layout_set_font_description (layout, desc);
* pango_font_description_free (desc);
*
* /* Draw the layout N_WORDS times in a circle */
* for (i = 0; i < N_WORDS; i++)
* {
* int width, height;
* double angle = (360. * i) / N_WORDS;
* double red;
*
* cairo_save (cr);
*
* /* Gradient from red at angle == 60 to blue at angle == 240 */
* red = (1 + cos ((angle - 60) * G_PI / 180.)) / 2;
* cairo_set_source_rgb (cr, red, 0, 1.0 - red);
*
* cairo_rotate (cr, angle * G_PI / 180.);
*
* /* Inform Pango to re-layout the text with the new transformation */
* pango_cairo_update_layout (cr, layout);
*
* pango_layout_get_size (layout, &width, &height);
* cairo_move_to (cr, - ((double)width / PANGO_SCALE) / 2, - RADIUS);
* pango_cairo_show_layout (cr, layout);
*
* cairo_restore (cr);
* }
*
* /* free the layout object */
* g_object_unref (layout);
* }
*
* int main (int argc, char **argv)
* {
* cairo_t *cr;
* char *filename;
* cairo_status_t status;
* cairo_surface_t *surface;
*
* if (argc != 2)
* {
* g_printerr ("Usage: cairosimple OUTPUT_FILENAME\n");
* return 1;
* }
*
* filename = argv[1];
*
* surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
* 2 * RADIUS, 2 * RADIUS);
* cr = cairo_create (surface);
*
* cairo_set_source_rgb (cr, 1.0, 1.0, 1.0);
* cairo_paint (cr);
* draw_text (cr);
* cairo_destroy (cr);
*
* status = cairo_surface_write_to_png (surface, filename);
* cairo_surface_destroy (surface);
*
* if (status != CAIRO_STATUS_SUCCESS)
* {
* g_printerr ("Could not save png to '%s'\n", filename);
* return 1;
* }
*
* return 0;
* }
* ]|
*
* Once you build and run the example code above, you should see the
* following result:
*
* ![Output of rotated-example](rotated-text.png)
*/
#include "config.h"
#include
#include "pango-font-private.h"
#include "pangocairo-private.h"
#include "pango-glyph-item.h"
#include "pango-impl-utils.h"
typedef struct _PangoCairoRendererClass PangoCairoRendererClass;
#define PANGO_CAIRO_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), PANGO_TYPE_CAIRO_RENDERER, PangoCairoRendererClass))
#define PANGO_IS_CAIRO_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), PANGO_TYPE_CAIRO_RENDERER))
#define PANGO_CAIRO_RENDERER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), PANGO_TYPE_CAIRO_RENDERER, PangoCairoRendererClass))
struct _PangoCairoRenderer
{
PangoRenderer parent_instance;
cairo_t *cr;
gboolean do_path;
gboolean has_show_text_glyphs;
double x_offset, y_offset;
/* house-keeping options */
gboolean is_cached_renderer;
gboolean cr_had_current_point;
};
struct _PangoCairoRendererClass
{
PangoRendererClass parent_class;
};
G_DEFINE_TYPE (PangoCairoRenderer, pango_cairo_renderer, PANGO_TYPE_RENDERER)
static void
set_color (PangoCairoRenderer *crenderer,
PangoRenderPart part)
{
PangoColor *color = pango_renderer_get_color ((PangoRenderer *) (crenderer), part);
guint16 a = pango_renderer_get_alpha ((PangoRenderer *) (crenderer), part);
gdouble red, green, blue, alpha;
if (!a && !color)
return;
if (color)
{
red = color->red / 65535.;
green = color->green / 65535.;
blue = color->blue / 65535.;
alpha = 1.;
}
else
{
cairo_pattern_t *pattern = cairo_get_source (crenderer->cr);
if (pattern && cairo_pattern_get_type (pattern) == CAIRO_PATTERN_TYPE_SOLID)
cairo_pattern_get_rgba (pattern, &red, &green, &blue, &alpha);
else
{
red = 0.;
green = 0.;
blue = 0.;
alpha = 1.;
}
}
if (a)
alpha = a / 65535.;
cairo_set_source_rgba (crenderer->cr, red, green, blue, alpha);
}
/* note: modifies crenderer->cr without doing cairo_save/restore() */
static void
_pango_cairo_renderer_draw_frame (PangoCairoRenderer *crenderer,
double x,
double y,
double width,
double height,
double line_width,
gboolean invalid)
{
cairo_t *cr = crenderer->cr;
if (crenderer->do_path)
{
double d2 = line_width * .5, d = line_width;
/* we draw an outer box in one winding direction and an inner one in the
* opposite direction. This works for both cairo windings rules.
*
* what we really want is cairo_stroke_to_path(), but that's not
* implemented in cairo yet.
*/
/* outer */
cairo_rectangle (cr, x-d2, y-d2, width+d, height+d);
/* inner */
if (invalid)
{
/* delicacies of computing the joint... this is REALLY slow */
double alpha, tan_alpha2, cos_alpha;
double sx, sy;
alpha = atan2 (height, width);
tan_alpha2 = tan (alpha * .5);
if (tan_alpha2 < 1e-5 || (sx = d2 / tan_alpha2, 2. * sx > width - d))
sx = (width - d) * .5;
cos_alpha = cos (alpha);
if (cos_alpha < 1e-5 || (sy = d2 / cos_alpha, 2. * sy > height - d))
sy = (height - d) * .5;
/* top triangle */
cairo_new_sub_path (cr);
cairo_line_to (cr, x+width-sx, y+d2);
cairo_line_to (cr, x+sx, y+d2);
cairo_line_to (cr, x+.5*width, y+.5*height-sy);
cairo_close_path (cr);
/* bottom triangle */
cairo_new_sub_path (cr);
cairo_line_to (cr, x+width-sx, y+height-d2);
cairo_line_to (cr, x+.5*width, y+.5*height+sy);
cairo_line_to (cr, x+sx, y+height-d2);
cairo_close_path (cr);
alpha = G_PI_2 - alpha;
tan_alpha2 = tan (alpha * .5);
if (tan_alpha2 < 1e-5 || (sy = d2 / tan_alpha2, 2. * sy > height - d))
sy = (height - d) * .5;
cos_alpha = cos (alpha);
if (cos_alpha < 1e-5 || (sx = d2 / cos_alpha, 2. * sx > width - d))
sx = (width - d) * .5;
/* left triangle */
cairo_new_sub_path (cr);
cairo_line_to (cr, x+d2, y+sy);
cairo_line_to (cr, x+d2, y+height-sy);
cairo_line_to (cr, x+.5*width-sx, y+.5*height);
cairo_close_path (cr);
/* right triangle */
cairo_new_sub_path (cr);
cairo_line_to (cr, x+width-d2, y+sy);
cairo_line_to (cr, x+.5*width+sx, y+.5*height);
cairo_line_to (cr, x+width-d2, y+height-sy);
cairo_close_path (cr);
}
else
cairo_rectangle (cr, x+width-d2, y+d2, - (width-d), height-d);
}
else
{
cairo_rectangle (cr, x, y, width, height);
if (invalid)
{
/* draw an X */
cairo_new_sub_path (cr);
cairo_move_to (cr, x, y);
cairo_rel_line_to (cr, width, height);
cairo_new_sub_path (cr);
cairo_move_to (cr, x + width, y);
cairo_rel_line_to (cr, -width, height);
cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
}
cairo_set_line_width (cr, line_width);
cairo_set_line_join (cr, CAIRO_LINE_JOIN_MITER);
cairo_set_miter_limit (cr, 2.);
cairo_stroke (cr);
}
}
static void
_pango_cairo_renderer_draw_box_glyph (PangoCairoRenderer *crenderer,
PangoGlyphInfo *gi,
double cx,
double cy,
gboolean invalid)
{
cairo_save (crenderer->cr);
_pango_cairo_renderer_draw_frame (crenderer,
cx + 1.5,
cy + 1.5 - PANGO_UNKNOWN_GLYPH_HEIGHT,
(double)gi->geometry.width / PANGO_SCALE - 3.0,
PANGO_UNKNOWN_GLYPH_HEIGHT - 3.0,
1.0,
invalid);
cairo_restore (crenderer->cr);
}
static void
_pango_cairo_renderer_draw_unknown_glyph (PangoCairoRenderer *crenderer,
PangoFont *font,
PangoGlyphInfo *gi,
double cx,
double cy)
{
char buf[7];
double x0, y0;
int row, col;
int rows, cols;
double width, lsb;
char hexbox_string[2] = { 0, 0 };
PangoCairoFontHexBoxInfo *hbi;
gunichar ch;
gboolean invalid_input;
const char *p;
const char *name;
cairo_save (crenderer->cr);
ch = gi->glyph & ~PANGO_GLYPH_UNKNOWN_FLAG;
invalid_input = G_UNLIKELY (gi->glyph == PANGO_GLYPH_INVALID_INPUT || ch > 0x10FFFF);
hbi = _pango_cairo_font_get_hex_box_info ((PangoCairoFont *)font);
if (!hbi || !_pango_cairo_font_install ((PangoFont *)(hbi->font), crenderer->cr))
{
_pango_cairo_renderer_draw_box_glyph (crenderer, gi, cx, cy, invalid_input);
goto done;
}
if (G_UNLIKELY (invalid_input))
{
rows = hbi->rows;
cols = 1;
}
else if (ch == 0x2423 ||
g_unichar_type (ch) == G_UNICODE_SPACE_SEPARATOR)
{
/* We never want to show a hex box or other drawing for
* space. If we want space to be visible, we replace 0x20
* by 0x2423 (visible space).
*
* Since we don't want to rely on glyph availability,
* we render a centered dot ourselves.
*/
double x = cx + 0.5 *((double)gi->geometry.width / PANGO_SCALE);
double y = cy + hbi->box_descent - 0.5 * hbi->box_height;
cairo_new_sub_path (crenderer->cr);
cairo_arc (crenderer->cr, x, y, 1.5 * hbi->line_width, 0, 2 * G_PI);
cairo_close_path (crenderer->cr);
cairo_fill (crenderer->cr);
goto done;
}
else if (ch == '\t')
{
/* Since we don't want to rely on glyph availability,
* we render an arrow like ↦ ourselves.
*/
double y = cy + hbi->box_descent - 0.5 * hbi->box_height;
double width = (double)gi->geometry.width / PANGO_SCALE;
double offset = 0.2 * width;
double x = cx + offset;
double al = width - 2 * offset; /* arrow length */
double tl = MIN (hbi->digit_width, 0.75 * al); /* tip length */
double tw2 = 2.5 * hbi->line_width; /* tip width / 2 */
double lw2 = 0.5 * hbi->line_width; /* line width / 2 */
cairo_move_to (crenderer->cr, x - lw2, y - tw2);
cairo_line_to (crenderer->cr, x + lw2, y - tw2);
cairo_line_to (crenderer->cr, x + lw2, y - lw2);
cairo_line_to (crenderer->cr, x + al - tl, y - lw2);
cairo_line_to (crenderer->cr, x + al - tl, y - tw2);
cairo_line_to (crenderer->cr, x + al, y);
cairo_line_to (crenderer->cr, x + al - tl, y + tw2);
cairo_line_to (crenderer->cr, x + al - tl, y + lw2);
cairo_line_to (crenderer->cr, x + lw2, y + lw2);
cairo_line_to (crenderer->cr, x + lw2, y + tw2);
cairo_line_to (crenderer->cr, x - lw2, y + tw2);
cairo_close_path (crenderer->cr);
cairo_fill (crenderer->cr);
goto done;
}
else if (ch == '\n' || ch == 0x2028 || ch == 0x2029)
{
/* Since we don't want to rely on glyph availability,
* we render an arrow like ↵ ourselves.
*/
double width = (double)gi->geometry.width / PANGO_SCALE;
double offset = 0.2 * width;
double al = width - 2 * offset; /* arrow length */
double tl = MIN (hbi->digit_width, 0.75 * al); /* tip length */
double ah = al - 0.5 * tl; /* arrow height */
double tw2 = 2.5 * hbi->line_width; /* tip width / 2 */
double x = cx + offset;
double y = cy - (hbi->box_height - al) / 2;
double lw2 = 0.5 * hbi->line_width; /* line width / 2 */
cairo_move_to (crenderer->cr, x, y);
cairo_line_to (crenderer->cr, x + tl, y - tw2);
cairo_line_to (crenderer->cr, x + tl, y - lw2);
cairo_line_to (crenderer->cr, x + al - lw2, y - lw2);
cairo_line_to (crenderer->cr, x + al - lw2, y - ah);
cairo_line_to (crenderer->cr, x + al + lw2, y - ah);
cairo_line_to (crenderer->cr, x + al + lw2, y + lw2);
cairo_line_to (crenderer->cr, x + tl, y + lw2);
cairo_line_to (crenderer->cr, x + tl, y + tw2);
cairo_close_path (crenderer->cr);
cairo_fill (crenderer->cr);
goto done;
}
else if ((name = pango_get_ignorable_size (ch, &rows, &cols)))
{
/* Nothing else to do, we render 'default ignorable' chars
* as hex box with their nick.
*/
}
else
{
/* Everything else gets a traditional hex box. */
rows = hbi->rows;
cols = (ch > 0xffff ? 6 : 4) / rows;
g_snprintf (buf, sizeof(buf), (ch > 0xffff) ? "%06X" : "%04X", ch);
name = buf;
}
width = (3 * hbi->pad_x + cols * (hbi->digit_width + hbi->pad_x));
lsb = ((double)gi->geometry.width / PANGO_SCALE - width) * .5;
lsb = floor (lsb / hbi->pad_x) * hbi->pad_x;
_pango_cairo_renderer_draw_frame (crenderer,
cx + lsb + .5 * hbi->pad_x,
cy + hbi->box_descent - hbi->box_height + hbi->pad_y * 0.5,
width - hbi->pad_x,
(hbi->box_height - hbi->pad_y),
hbi->line_width,
invalid_input);
if (invalid_input)
goto done;
x0 = cx + lsb + hbi->pad_x * 2;
y0 = cy + hbi->box_descent - hbi->pad_y * 2 - ((hbi->rows - rows) * hbi->digit_height / 2);
for (row = 0, p = name; row < rows; row++)
{
double y = y0 - (rows - 1 - row) * (hbi->digit_height + hbi->pad_y);
for (col = 0; col < cols; col++, p++)
{
double x = x0 + col * (hbi->digit_width + hbi->pad_x);
if (!p)
goto done;
cairo_move_to (crenderer->cr, x, y);
hexbox_string[0] = p[0];
if (crenderer->do_path)
cairo_text_path (crenderer->cr, hexbox_string);
else
cairo_show_text (crenderer->cr, hexbox_string);
}
}
done:
cairo_restore (crenderer->cr);
}
#ifndef STACK_BUFFER_SIZE
#define STACK_BUFFER_SIZE (512 * sizeof (int))
#endif
#define STACK_ARRAY_LENGTH(T) (STACK_BUFFER_SIZE / sizeof(T))
static void
pango_cairo_renderer_show_text_glyphs (PangoRenderer *renderer,
const char *text,
int text_len,
PangoGlyphString *glyphs,
cairo_text_cluster_t *clusters,
int num_clusters,
gboolean backward,
PangoFont *font,
int x,
int y)
{
PangoCairoRenderer *crenderer = (PangoCairoRenderer *) (renderer);
int i, count;
int x_position = 0;
cairo_glyph_t *cairo_glyphs;
cairo_glyph_t stack_glyphs[STACK_ARRAY_LENGTH (cairo_glyph_t)];
double base_x = crenderer->x_offset + (double)x / PANGO_SCALE;
double base_y = crenderer->y_offset + (double)y / PANGO_SCALE;
cairo_save (crenderer->cr);
if (!crenderer->do_path)
set_color (crenderer, PANGO_RENDER_PART_FOREGROUND);
if (!_pango_cairo_font_install (font, crenderer->cr))
{
for (i = 0; i < glyphs->num_glyphs; i++)
{
PangoGlyphInfo *gi = &glyphs->glyphs[i];
if (gi->glyph != PANGO_GLYPH_EMPTY)
{
double cx = base_x + (double)(x_position + gi->geometry.x_offset) / PANGO_SCALE;
double cy = gi->geometry.y_offset == 0 ?
base_y :
base_y + (double)(gi->geometry.y_offset) / PANGO_SCALE;
_pango_cairo_renderer_draw_unknown_glyph (crenderer, font, gi, cx, cy);
}
x_position += gi->geometry.width;
}
goto done;
}
if (glyphs->num_glyphs > (int) G_N_ELEMENTS (stack_glyphs))
cairo_glyphs = g_new (cairo_glyph_t, glyphs->num_glyphs);
else
cairo_glyphs = stack_glyphs;
count = 0;
for (i = 0; i < glyphs->num_glyphs; i++)
{
PangoGlyphInfo *gi = &glyphs->glyphs[i];
if (gi->glyph != PANGO_GLYPH_EMPTY)
{
double cx = base_x + (double)(x_position + gi->geometry.x_offset) / PANGO_SCALE;
double cy = gi->geometry.y_offset == 0 ?
base_y :
base_y + (double)(gi->geometry.y_offset) / PANGO_SCALE;
if (gi->glyph & PANGO_GLYPH_UNKNOWN_FLAG)
{
if (gi->glyph == (0x20 | PANGO_GLYPH_UNKNOWN_FLAG))
; /* no hex boxes for space, please */
else
_pango_cairo_renderer_draw_unknown_glyph (crenderer, font, gi, cx, cy);
}
else
{
cairo_glyphs[count].index = gi->glyph;
cairo_glyphs[count].x = cx;
cairo_glyphs[count].y = cy;
count++;
}
}
x_position += gi->geometry.width;
}
if (G_UNLIKELY (crenderer->do_path))
cairo_glyph_path (crenderer->cr, cairo_glyphs, count);
else
if (G_UNLIKELY (clusters))
cairo_show_text_glyphs (crenderer->cr,
text, text_len,
cairo_glyphs, count,
clusters, num_clusters,
backward ? CAIRO_TEXT_CLUSTER_FLAG_BACKWARD : 0);
else
cairo_show_glyphs (crenderer->cr, cairo_glyphs, count);
if (cairo_glyphs != stack_glyphs)
g_free (cairo_glyphs);
done:
cairo_restore (crenderer->cr);
}
static void
pango_cairo_renderer_draw_glyphs (PangoRenderer *renderer,
PangoFont *font,
PangoGlyphString *glyphs,
int x,
int y)
{
pango_cairo_renderer_show_text_glyphs (renderer,
NULL, 0,
glyphs,
NULL, 0,
FALSE,
font,
x, y);
}
static void
pango_cairo_renderer_draw_glyph_item (PangoRenderer *renderer,
const char *text,
PangoGlyphItem *glyph_item,
int x,
int y)
{
PangoCairoRenderer *crenderer = (PangoCairoRenderer *) (renderer);
PangoFont *font = glyph_item->item->analysis.font;
PangoGlyphString *glyphs = glyph_item->glyphs;
PangoItem *item = glyph_item->item;
gboolean backward = (item->analysis.level & 1) != 0;
PangoGlyphItemIter iter;
cairo_text_cluster_t *cairo_clusters;
cairo_text_cluster_t stack_clusters[STACK_ARRAY_LENGTH (cairo_text_cluster_t)];
int num_clusters;
if (!crenderer->has_show_text_glyphs || crenderer->do_path)
{
pango_cairo_renderer_show_text_glyphs (renderer,
NULL, 0,
glyphs,
NULL, 0,
FALSE,
font,
x, y);
return;
}
if (glyphs->num_glyphs > (int) G_N_ELEMENTS (stack_clusters))
cairo_clusters = g_new (cairo_text_cluster_t, glyphs->num_glyphs);
else
cairo_clusters = stack_clusters;
num_clusters = 0;
if (pango_glyph_item_iter_init_start (&iter, glyph_item, text))
{
do {
int num_bytes, num_glyphs, i;
num_bytes = iter.end_index - iter.start_index;
num_glyphs = backward ? iter.start_glyph - iter.end_glyph : iter.end_glyph - iter.start_glyph;
if (num_bytes < 1)
g_warning ("pango_cairo_renderer_draw_glyph_item: bad cluster has num_bytes %d", num_bytes);
if (num_glyphs < 1)
g_warning ("pango_cairo_renderer_draw_glyph_item: bad cluster has num_glyphs %d", num_glyphs);
/* Discount empty and unknown glyphs */
for (i = MIN (iter.start_glyph, iter.end_glyph+1);
i < MAX (iter.start_glyph+1, iter.end_glyph);
i++)
{
PangoGlyphInfo *gi = &glyphs->glyphs[i];
if (gi->glyph == PANGO_GLYPH_EMPTY ||
gi->glyph & PANGO_GLYPH_UNKNOWN_FLAG)
num_glyphs--;
}
cairo_clusters[num_clusters].num_bytes = num_bytes;
cairo_clusters[num_clusters].num_glyphs = num_glyphs;
num_clusters++;
} while (pango_glyph_item_iter_next_cluster (&iter));
}
pango_cairo_renderer_show_text_glyphs (renderer,
text + item->offset, item->length,
glyphs,
cairo_clusters, num_clusters,
backward,
font,
x, y);
if (cairo_clusters != stack_clusters)
g_free (cairo_clusters);
}
static void
pango_cairo_renderer_draw_rectangle (PangoRenderer *renderer,
PangoRenderPart part,
int x,
int y,
int width,
int height)
{
PangoCairoRenderer *crenderer = (PangoCairoRenderer *) (renderer);
if (!crenderer->do_path)
{
cairo_save (crenderer->cr);
set_color (crenderer, part);
}
cairo_rectangle (crenderer->cr,
crenderer->x_offset + (double)x / PANGO_SCALE,
crenderer->y_offset + (double)y / PANGO_SCALE,
(double)width / PANGO_SCALE, (double)height / PANGO_SCALE);
if (!crenderer->do_path)
{
cairo_fill (crenderer->cr);
cairo_restore (crenderer->cr);
}
}
static void
pango_cairo_renderer_draw_trapezoid (PangoRenderer *renderer,
PangoRenderPart part,
double y1_,
double x11,
double x21,
double y2,
double x12,
double x22)
{
PangoCairoRenderer *crenderer = (PangoCairoRenderer *) (renderer);
cairo_t *cr;
double x, y;
cr = crenderer->cr;
cairo_save (cr);
if (!crenderer->do_path)
set_color (crenderer, part);
x = crenderer->x_offset,
y = crenderer->y_offset;
cairo_user_to_device_distance (cr, &x, &y);
cairo_identity_matrix (cr);
cairo_translate (cr, x, y);
cairo_move_to (cr, x11, y1_);
cairo_line_to (cr, x21, y1_);
cairo_line_to (cr, x22, y2);
cairo_line_to (cr, x12, y2);
cairo_close_path (cr);
if (!crenderer->do_path)
cairo_fill (cr);
cairo_restore (cr);
}
/* Draws an error underline that looks like one of:
* H E H
* /\ /\ /\ /\ /\ -
* A/ \ / \ / \ A/ \ / \ |
* \ \ / \ / /D \ \ / \ |
* \ \/ C \/ / \ \/ C \ | height = HEIGHT_SQUARES * square
* \ /\ F / \ F /\ \ |
* \ / \ / \ / \ \G |
* \ / \ / \ / \ / |
* \/ \/ \/ \/ -
* B B
* |---|
* unit_width = (HEIGHT_SQUARES - 1) * square
*
* The x, y, width, height passed in give the desired bounding box;
* x/width are adjusted to make the underline a integer number of units
* wide.
*/
#define HEIGHT_SQUARES 2.5
static void
draw_error_underline (cairo_t *cr,
double x,
double y,
double width,
double height)
{
double square = height / HEIGHT_SQUARES;
double unit_width = (HEIGHT_SQUARES - 1) * square;
double double_width = 2 * unit_width;
int width_units = (width + unit_width / 2) / unit_width;
double y_top, y_bottom;
double x_left, x_middle, x_right;
int i;
x += (width - width_units * unit_width) / 2;
y_top = y;
y_bottom = y + height;
/* Bottom of squiggle */
x_middle = x + unit_width;
x_right = x + double_width;
cairo_move_to (cr, x - square / 2, y_top + square / 2); /* A */
for (i = 0; i < width_units-2; i += 2)
{
cairo_line_to (cr, x_middle, y_bottom); /* B */
cairo_line_to (cr, x_right, y_top + square); /* C */
x_middle += double_width;
x_right += double_width;
}
cairo_line_to (cr, x_middle, y_bottom); /* B */
if (i + 1 == width_units)
cairo_line_to (cr, x_middle + square / 2, y_bottom - square / 2); /* G */
else if (i + 2 == width_units) {
cairo_line_to (cr, x_right + square / 2, y_top + square / 2); /* D */
cairo_line_to (cr, x_right, y_top); /* E */
}
/* Top of squiggle */
x_left = x_middle - unit_width;
for (; i >= 0; i -= 2)
{
cairo_line_to (cr, x_middle, y_bottom - square); /* F */
cairo_line_to (cr, x_left, y_top); /* H */
x_left -= double_width;
x_middle -= double_width;
}
}
static void
pango_cairo_renderer_draw_error_underline (PangoRenderer *renderer,
int x,
int y,
int width,
int height)
{
PangoCairoRenderer *crenderer = (PangoCairoRenderer *) (renderer);
cairo_t *cr = crenderer->cr;
if (!crenderer->do_path)
{
cairo_save (cr);
set_color (crenderer, PANGO_RENDER_PART_UNDERLINE);
cairo_new_path (cr);
}
draw_error_underline (cr,
crenderer->x_offset + (double)x / PANGO_SCALE,
crenderer->y_offset + (double)y / PANGO_SCALE,
(double)width / PANGO_SCALE, (double)height / PANGO_SCALE);
if (!crenderer->do_path)
{
cairo_fill (cr);
cairo_restore (cr);
}
}
static void
pango_cairo_renderer_draw_shape (PangoRenderer *renderer,
PangoAttrShape *attr,
int x,
int y)
{
PangoCairoRenderer *crenderer = (PangoCairoRenderer *) (renderer);
cairo_t *cr = crenderer->cr;
PangoLayout *layout;
PangoCairoShapeRendererFunc shape_renderer;
gpointer shape_renderer_data;
double base_x, base_y;
layout = pango_renderer_get_layout (renderer);
if (!layout)
return;
shape_renderer = pango_cairo_context_get_shape_renderer (pango_layout_get_context (layout),
&shape_renderer_data);
if (!shape_renderer)
return;
base_x = crenderer->x_offset + (double)x / PANGO_SCALE;
base_y = crenderer->y_offset + (double)y / PANGO_SCALE;
cairo_save (cr);
if (!crenderer->do_path)
set_color (crenderer, PANGO_RENDER_PART_FOREGROUND);
cairo_move_to (cr, base_x, base_y);
shape_renderer (cr, attr, crenderer->do_path, shape_renderer_data);
cairo_restore (cr);
}
static void
pango_cairo_renderer_init (PangoCairoRenderer *renderer G_GNUC_UNUSED)
{
}
static void
pango_cairo_renderer_class_init (PangoCairoRendererClass *klass)
{
PangoRendererClass *renderer_class = PANGO_RENDERER_CLASS (klass);
renderer_class->draw_glyphs = pango_cairo_renderer_draw_glyphs;
renderer_class->draw_glyph_item = pango_cairo_renderer_draw_glyph_item;
renderer_class->draw_rectangle = pango_cairo_renderer_draw_rectangle;
renderer_class->draw_trapezoid = pango_cairo_renderer_draw_trapezoid;
renderer_class->draw_error_underline = pango_cairo_renderer_draw_error_underline;
renderer_class->draw_shape = pango_cairo_renderer_draw_shape;
}
static PangoCairoRenderer *cached_renderer = NULL; /* MT-safe */
G_LOCK_DEFINE_STATIC (cached_renderer);
static PangoCairoRenderer *
acquire_renderer (void)
{
PangoCairoRenderer *renderer;
if (G_LIKELY (G_TRYLOCK (cached_renderer)))
{
if (G_UNLIKELY (!cached_renderer))
{
cached_renderer = g_object_new (PANGO_TYPE_CAIRO_RENDERER, NULL);
cached_renderer->is_cached_renderer = TRUE;
}
renderer = cached_renderer;
}
else
{
renderer = g_object_new (PANGO_TYPE_CAIRO_RENDERER, NULL);
}
return renderer;
}
static void
release_renderer (PangoCairoRenderer *renderer)
{
if (G_LIKELY (renderer->is_cached_renderer))
{
renderer->cr = NULL;
renderer->do_path = FALSE;
renderer->has_show_text_glyphs = FALSE;
renderer->x_offset = 0.;
renderer->y_offset = 0.;
G_UNLOCK (cached_renderer);
}
else
g_object_unref (renderer);
}
static void
save_current_point (PangoCairoRenderer *renderer)
{
renderer->cr_had_current_point = cairo_has_current_point (renderer->cr);
cairo_get_current_point (renderer->cr, &renderer->x_offset, &renderer->y_offset);
/* abuse save_current_point() to cache cairo_has_show_text_glyphs() result */
renderer->has_show_text_glyphs = cairo_surface_has_show_text_glyphs (cairo_get_target (renderer->cr));
}
static void
restore_current_point (PangoCairoRenderer *renderer)
{
if (renderer->cr_had_current_point)
/* XXX should do cairo_set_current_point() when we have that function */
cairo_move_to (renderer->cr, renderer->x_offset, renderer->y_offset);
else
cairo_new_sub_path (renderer->cr);
}
/* convenience wrappers using the default renderer */
static void
_pango_cairo_do_glyph_string (cairo_t *cr,
PangoFont *font,
PangoGlyphString *glyphs,
gboolean do_path)
{
PangoCairoRenderer *crenderer = acquire_renderer ();
PangoRenderer *renderer = (PangoRenderer *) crenderer;
crenderer->cr = cr;
crenderer->do_path = do_path;
save_current_point (crenderer);
if (!do_path)
{
/* unset all part colors, since when drawing just a glyph string,
* prepare_run() isn't called.
*/
pango_renderer_activate (renderer);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_FOREGROUND, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_BACKGROUND, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_UNDERLINE, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_STRIKETHROUGH, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_OVERLINE, NULL);
}
pango_renderer_draw_glyphs (renderer, font, glyphs, 0, 0);
if (!do_path)
{
pango_renderer_deactivate (renderer);
}
restore_current_point (crenderer);
release_renderer (crenderer);
}
static void
_pango_cairo_do_glyph_item (cairo_t *cr,
const char *text,
PangoGlyphItem *glyph_item,
gboolean do_path)
{
PangoCairoRenderer *crenderer = acquire_renderer ();
PangoRenderer *renderer = (PangoRenderer *) crenderer;
crenderer->cr = cr;
crenderer->do_path = do_path;
save_current_point (crenderer);
if (!do_path)
{
/* unset all part colors, since when drawing just a glyph string,
* prepare_run() isn't called.
*/
pango_renderer_activate (renderer);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_FOREGROUND, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_BACKGROUND, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_UNDERLINE, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_STRIKETHROUGH, NULL);
pango_renderer_set_color (renderer, PANGO_RENDER_PART_OVERLINE, NULL);
}
pango_renderer_draw_glyph_item (renderer, text, glyph_item, 0, 0);
if (!do_path)
{
pango_renderer_deactivate (renderer);
}
restore_current_point (crenderer);
release_renderer (crenderer);
}
static void
_pango_cairo_do_layout_line (cairo_t *cr,
PangoLayoutLine *line,
gboolean do_path)
{
PangoCairoRenderer *crenderer = acquire_renderer ();
PangoRenderer *renderer = (PangoRenderer *) crenderer;
crenderer->cr = cr;
crenderer->do_path = do_path;
save_current_point (crenderer);
pango_renderer_draw_layout_line (renderer, line, 0, 0);
restore_current_point (crenderer);
release_renderer (crenderer);
}
static void
_pango_cairo_do_layout (cairo_t *cr,
PangoLayout *layout,
gboolean do_path)
{
PangoCairoRenderer *crenderer = acquire_renderer ();
PangoRenderer *renderer = (PangoRenderer *) crenderer;
crenderer->cr = cr;
crenderer->do_path = do_path;
save_current_point (crenderer);
pango_renderer_draw_layout (renderer, layout, 0, 0);
restore_current_point (crenderer);
release_renderer (crenderer);
}
static void
_pango_cairo_do_error_underline (cairo_t *cr,
double x,
double y,
double width,
double height,
gboolean do_path)
{
/* We don't use a renderer here, for a simple reason:
* the only renderer we can get is the default renderer, that
* is all implemented here, so we shortcircuit and make our
* life way easier.
*/
if (!do_path)
cairo_new_path (cr);
draw_error_underline (cr, x, y, width, height);
if (!do_path)
cairo_fill (cr);
}
/* public wrapper of above to show or append path */
/**
* pango_cairo_show_glyph_string:
* @cr: a Cairo context
* @font: a #PangoFont from a #PangoCairoFontMap
* @glyphs: a #PangoGlyphString
*
* Draws the glyphs in @glyphs in the specified cairo context.
* The origin of the glyphs (the left edge of the baseline) will
* be drawn at the current point of the cairo context.
*
* Since: 1.10
**/
void
pango_cairo_show_glyph_string (cairo_t *cr,
PangoFont *font,
PangoGlyphString *glyphs)
{
g_return_if_fail (cr != NULL);
g_return_if_fail (glyphs != NULL);
_pango_cairo_do_glyph_string (cr, font, glyphs, FALSE);
}
/**
* pango_cairo_show_glyph_item:
* @cr: a Cairo context
* @text: the UTF-8 text that @glyph_item refers to
* @glyph_item: a #PangoGlyphItem
*
* Draws the glyphs in @glyph_item in the specified cairo context,
* embedding the text associated with the glyphs in the output if the
* output format supports it (PDF for example), otherwise it acts
* similar to pango_cairo_show_glyph_string().
*
* The origin of the glyphs (the left edge of the baseline) will
* be drawn at the current point of the cairo context.
*
* Note that @text is the start of the text for layout, which is then
* indexed by @glyph_item->item->offset.
*
* Since: 1.22
**/
void
pango_cairo_show_glyph_item (cairo_t *cr,
const char *text,
PangoGlyphItem *glyph_item)
{
g_return_if_fail (cr != NULL);
g_return_if_fail (text != NULL);
g_return_if_fail (glyph_item != NULL);
_pango_cairo_do_glyph_item (cr, text, glyph_item, FALSE);
}
/**
* pango_cairo_show_layout_line:
* @cr: a Cairo context
* @line: a #PangoLayoutLine
*
* Draws a #PangoLayoutLine in the specified cairo context.
* The origin of the glyphs (the left edge of the line) will
* be drawn at the current point of the cairo context.
*
* Since: 1.10
**/
void
pango_cairo_show_layout_line (cairo_t *cr,
PangoLayoutLine *line)
{
g_return_if_fail (cr != NULL);
g_return_if_fail (line != NULL);
_pango_cairo_do_layout_line (cr, line, FALSE);
}
/**
* pango_cairo_show_layout:
* @cr: a Cairo context
* @layout: a Pango layout
*
* Draws a #PangoLayout in the specified cairo context.
* The top-left corner of the #PangoLayout will be drawn
* at the current point of the cairo context.
*
* Since: 1.10
**/
void
pango_cairo_show_layout (cairo_t *cr,
PangoLayout *layout)
{
g_return_if_fail (cr != NULL);
g_return_if_fail (PANGO_IS_LAYOUT (layout));
_pango_cairo_do_layout (cr, layout, FALSE);
}
/**
* pango_cairo_show_error_underline:
* @cr: a Cairo context
* @x: The X coordinate of one corner of the rectangle
* @y: The Y coordinate of one corner of the rectangle
* @width: Non-negative width of the rectangle
* @height: Non-negative height of the rectangle
*
* Draw a squiggly line in the specified cairo context that approximately
* covers the given rectangle in the style of an underline used to indicate a
* spelling error. (The width of the underline is rounded to an integer
* number of up/down segments and the resulting rectangle is centered in the
* original rectangle)
*
* Since: 1.14
**/
void
pango_cairo_show_error_underline (cairo_t *cr,
double x,
double y,
double width,
double height)
{
g_return_if_fail (cr != NULL);
g_return_if_fail ((width >= 0) && (height >= 0));
_pango_cairo_do_error_underline (cr, x, y, width, height, FALSE);
}
/**
* pango_cairo_glyph_string_path:
* @cr: a Cairo context
* @font: a #PangoFont from a #PangoCairoFontMap
* @glyphs: a #PangoGlyphString
*
* Adds the glyphs in @glyphs to the current path in the specified
* cairo context. The origin of the glyphs (the left edge of the baseline)
* will be at the current point of the cairo context.
*
* Since: 1.10
**/
void
pango_cairo_glyph_string_path (cairo_t *cr,
PangoFont *font,
PangoGlyphString *glyphs)
{
g_return_if_fail (cr != NULL);
g_return_if_fail (glyphs != NULL);
_pango_cairo_do_glyph_string (cr, font, glyphs, TRUE);
}
/**
* pango_cairo_layout_line_path:
* @cr: a Cairo context
* @line: a #PangoLayoutLine
*
* Adds the text in #PangoLayoutLine to the current path in the
* specified cairo context. The origin of the glyphs (the left edge
* of the line) will be at the current point of the cairo context.
*
* Since: 1.10
**/
void
pango_cairo_layout_line_path (cairo_t *cr,
PangoLayoutLine *line)
{
g_return_if_fail (cr != NULL);
g_return_if_fail (line != NULL);
_pango_cairo_do_layout_line (cr, line, TRUE);
}
/**
* pango_cairo_layout_path:
* @cr: a Cairo context
* @layout: a Pango layout
*
* Adds the text in a #PangoLayout to the current path in the
* specified cairo context. The top-left corner of the #PangoLayout
* will be at the current point of the cairo context.
*
* Since: 1.10
**/
void
pango_cairo_layout_path (cairo_t *cr,
PangoLayout *layout)
{
g_return_if_fail (cr != NULL);
g_return_if_fail (PANGO_IS_LAYOUT (layout));
_pango_cairo_do_layout (cr, layout, TRUE);
}
/**
* pango_cairo_error_underline_path:
* @cr: a Cairo context
* @x: The X coordinate of one corner of the rectangle
* @y: The Y coordinate of one corner of the rectangle
* @width: Non-negative width of the rectangle
* @height: Non-negative height of the rectangle
*
* Add a squiggly line to the current path in the specified cairo context that
* approximately covers the given rectangle in the style of an underline used
* to indicate a spelling error. (The width of the underline is rounded to an
* integer number of up/down segments and the resulting rectangle is centered
* in the original rectangle)
*
* Since: 1.14
**/
void
pango_cairo_error_underline_path (cairo_t *cr,
double x,
double y,
double width,
double height)
{
g_return_if_fail (cr != NULL);
g_return_if_fail ((width >= 0) && (height >= 0));
_pango_cairo_do_error_underline (cr, x, y, width, height, TRUE);
}