|
Packit |
0ec9dd |
/* Pango
|
|
Packit |
0ec9dd |
* pangoxft-render.c: Rendering routines for the Xft library
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Copyright (C) 2004 Red Hat Software
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* This library is free software; you can redistribute it and/or
|
|
Packit |
0ec9dd |
* modify it under the terms of the GNU Library General Public
|
|
Packit |
0ec9dd |
* License as published by the Free Software Foundation; either
|
|
Packit |
0ec9dd |
* version 2 of the License, or (at your option) any later version.
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* This library is distributed in the hope that it will be useful,
|
|
Packit |
0ec9dd |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
0ec9dd |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Packit |
0ec9dd |
* Library General Public License for more details.
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* You should have received a copy of the GNU Library General Public
|
|
Packit |
0ec9dd |
* License along with this library; if not, write to the
|
|
Packit |
0ec9dd |
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
Packit |
0ec9dd |
* Boston, MA 02111-1307, USA.
|
|
Packit |
0ec9dd |
*/
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
#include "config.h"
|
|
Packit |
0ec9dd |
#include <math.h>
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
#include "pangoxft-render.h"
|
|
Packit |
0ec9dd |
#include "pangoxft-private.h"
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
enum {
|
|
Packit |
0ec9dd |
PROP_0,
|
|
Packit |
0ec9dd |
PROP_DISPLAY,
|
|
Packit |
0ec9dd |
PROP_SCREEN
|
|
Packit |
0ec9dd |
};
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
struct _PangoXftRendererPrivate
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoColor default_color;
|
|
Packit |
0ec9dd |
guint16 alpha;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
Picture src_picture;
|
|
Packit |
0ec9dd |
Picture dest_picture;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
XRenderPictFormat *mask_format;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
GArray *trapezoids;
|
|
Packit |
0ec9dd |
PangoRenderPart trapezoid_part;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
GArray *glyphs;
|
|
Packit |
0ec9dd |
PangoFont *glyph_font;
|
|
Packit |
0ec9dd |
};
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_finalize (GObject *object);
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_set_property (GObject *object,
|
|
Packit |
0ec9dd |
guint prop_id,
|
|
Packit |
0ec9dd |
const GValue *value,
|
|
Packit |
0ec9dd |
GParamSpec *pspec);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_real_composite_trapezoids (PangoXftRenderer *xftrenderer,
|
|
Packit |
0ec9dd |
PangoRenderPart part,
|
|
Packit |
0ec9dd |
XTrapezoid *trapezoids,
|
|
Packit |
0ec9dd |
int n_trapezoids);
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_real_composite_glyphs (PangoXftRenderer *xftrenderer,
|
|
Packit |
0ec9dd |
XftFont *xft_font,
|
|
Packit |
0ec9dd |
XftGlyphSpec *glyphs,
|
|
Packit |
0ec9dd |
int n_glyphs);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_draw_glyphs (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoFont *font,
|
|
Packit |
0ec9dd |
PangoGlyphString *glyphs,
|
|
Packit |
0ec9dd |
int x,
|
|
Packit |
0ec9dd |
int y);
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_draw_trapezoid (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoRenderPart part,
|
|
Packit |
0ec9dd |
double y1,
|
|
Packit |
0ec9dd |
double x11,
|
|
Packit |
0ec9dd |
double x21,
|
|
Packit |
0ec9dd |
double y2,
|
|
Packit |
0ec9dd |
double x12,
|
|
Packit |
0ec9dd |
double x22);
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_part_changed (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoRenderPart part);
|
|
Packit |
0ec9dd |
static void pango_xft_renderer_end (PangoRenderer *renderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void flush_trapezoids (PangoXftRenderer *xftrenderer);
|
|
Packit |
0ec9dd |
static void flush_glyphs (PangoXftRenderer *xftrenderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
G_DEFINE_TYPE (PangoXftRenderer, pango_xft_renderer, PANGO_TYPE_RENDERER)
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_init (PangoXftRenderer *xftrenderer)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
xftrenderer->priv = G_TYPE_INSTANCE_GET_PRIVATE (xftrenderer,
|
|
Packit |
0ec9dd |
PANGO_TYPE_XFT_RENDERER,
|
|
Packit |
0ec9dd |
PangoXftRendererPrivate);
|
|
Packit |
0ec9dd |
xftrenderer->priv->alpha = 0xffff;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_class_init (PangoXftRendererClass *klass)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
Packit |
0ec9dd |
PangoRendererClass *renderer_class = PANGO_RENDERER_CLASS (klass);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
klass->composite_glyphs = pango_xft_renderer_real_composite_glyphs;
|
|
Packit |
0ec9dd |
klass->composite_trapezoids = pango_xft_renderer_real_composite_trapezoids;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
renderer_class->draw_glyphs = pango_xft_renderer_draw_glyphs;
|
|
Packit |
0ec9dd |
renderer_class->draw_trapezoid = pango_xft_renderer_draw_trapezoid;
|
|
Packit |
0ec9dd |
renderer_class->part_changed = pango_xft_renderer_part_changed;
|
|
Packit |
0ec9dd |
renderer_class->end = pango_xft_renderer_end;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
object_class->finalize = pango_xft_renderer_finalize;
|
|
Packit |
0ec9dd |
object_class->set_property = pango_xft_renderer_set_property;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_object_class_install_property (object_class, PROP_DISPLAY,
|
|
Packit |
0ec9dd |
g_param_spec_pointer ("display",
|
|
Packit |
0ec9dd |
"Display",
|
|
Packit |
0ec9dd |
"The display being rendered to",
|
|
Packit |
0ec9dd |
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY));
|
|
Packit |
0ec9dd |
g_object_class_install_property (object_class, PROP_SCREEN,
|
|
Packit |
0ec9dd |
g_param_spec_int ("screen",
|
|
Packit |
0ec9dd |
"Screen",
|
|
Packit |
0ec9dd |
"The screen being rendered to",
|
|
Packit |
0ec9dd |
0, G_MAXINT, 0,
|
|
Packit |
0ec9dd |
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_type_class_add_private (object_class, sizeof (PangoXftRendererPrivate));
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_finalize (GObject *object)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *renderer = PANGO_XFT_RENDERER (object);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (renderer->priv->glyphs)
|
|
Packit |
0ec9dd |
g_array_free (renderer->priv->glyphs, TRUE);
|
|
Packit |
0ec9dd |
if (renderer->priv->trapezoids)
|
|
Packit |
0ec9dd |
g_array_free (renderer->priv->trapezoids, TRUE);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
G_OBJECT_CLASS (pango_xft_renderer_parent_class)->finalize (object);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_property (GObject *object,
|
|
Packit |
0ec9dd |
guint prop_id,
|
|
Packit |
0ec9dd |
const GValue *value,
|
|
Packit |
0ec9dd |
GParamSpec *pspec)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer = PANGO_XFT_RENDERER (object);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
switch (prop_id)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
case PROP_DISPLAY:
|
|
Packit |
0ec9dd |
xftrenderer->display = g_value_get_pointer (value);
|
|
Packit |
0ec9dd |
/* We possibly should use ARGB format when subpixel-AA is turned
|
|
Packit |
0ec9dd |
* on for the fontmap; we could discover that using the technique
|
|
Packit |
0ec9dd |
* for FC_DPI in pango_fc_face_list_sizes.
|
|
Packit |
0ec9dd |
*/
|
|
Packit |
0ec9dd |
xftrenderer->priv->mask_format = XRenderFindStandardFormat (xftrenderer->display,
|
|
Packit |
0ec9dd |
PictStandardA8);
|
|
Packit |
0ec9dd |
break;
|
|
Packit |
0ec9dd |
case PROP_SCREEN:
|
|
Packit |
0ec9dd |
xftrenderer->screen = g_value_get_int (value);
|
|
Packit |
0ec9dd |
break;
|
|
Packit |
0ec9dd |
default:
|
|
Packit |
0ec9dd |
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
Packit |
0ec9dd |
break;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_pictures (PangoXftRenderer *renderer,
|
|
Packit |
0ec9dd |
Picture src_picture,
|
|
Packit |
0ec9dd |
Picture dest_picture)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
renderer->priv->src_picture = src_picture;
|
|
Packit |
0ec9dd |
renderer->priv->dest_picture = dest_picture;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
flush_glyphs (PangoXftRenderer *xftrenderer)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
XftFont *xft_font;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (!xftrenderer->priv->glyphs ||
|
|
Packit |
0ec9dd |
xftrenderer->priv->glyphs->len == 0)
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xft_font = pango_xft_font_get_font (xftrenderer->priv->glyph_font);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
PANGO_XFT_RENDERER_GET_CLASS (xftrenderer)->composite_glyphs (xftrenderer,
|
|
Packit |
0ec9dd |
xft_font,
|
|
Packit |
0ec9dd |
(XftGlyphSpec *)xftrenderer->priv->glyphs->data,
|
|
Packit |
0ec9dd |
xftrenderer->priv->glyphs->len);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_array_set_size (xftrenderer->priv->glyphs, 0);
|
|
Packit |
0ec9dd |
g_object_unref (xftrenderer->priv->glyph_font);
|
|
Packit |
0ec9dd |
xftrenderer->priv->glyph_font = NULL;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
#define MAX_GLYPHS 1024
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
draw_glyph (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoFont *font,
|
|
Packit |
0ec9dd |
FT_UInt glyph,
|
|
Packit |
0ec9dd |
int x,
|
|
Packit |
0ec9dd |
int y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer = PANGO_XFT_RENDERER (renderer);
|
|
Packit |
0ec9dd |
XftGlyphSpec gs;
|
|
Packit |
0ec9dd |
int pixel_x, pixel_y;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (renderer->matrix)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
pixel_x = floor (0.5 + (x * renderer->matrix->xx + y * renderer->matrix->xy) / PANGO_SCALE + renderer->matrix->x0);
|
|
Packit |
0ec9dd |
pixel_y = floor (0.5 + (x * renderer->matrix->yx + y * renderer->matrix->yy) / PANGO_SCALE + renderer->matrix->y0);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
else
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
pixel_x = PANGO_PIXELS (x);
|
|
Packit |
0ec9dd |
pixel_y = PANGO_PIXELS (y);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/* Clip glyphs into the X coordinate range; we really
|
|
Packit |
0ec9dd |
* want to clip glyphs with an ink rect outside the
|
|
Packit |
0ec9dd |
* [0,32767] x [0,32767] rectangle but looking up
|
|
Packit |
0ec9dd |
* the ink rect here would be a noticeable speed hit.
|
|
Packit |
0ec9dd |
* This is close enough.
|
|
Packit |
0ec9dd |
*/
|
|
Packit |
0ec9dd |
if (pixel_x < -32768 || pixel_x > 32767 ||
|
|
Packit |
0ec9dd |
pixel_y < -32768 || pixel_y > 32767)
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
flush_trapezoids (xftrenderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (!xftrenderer->priv->glyphs)
|
|
Packit |
0ec9dd |
xftrenderer->priv->glyphs = g_array_new (FALSE, FALSE,
|
|
Packit |
0ec9dd |
sizeof (XftGlyphSpec));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (xftrenderer->priv->glyph_font != font ||
|
|
Packit |
0ec9dd |
xftrenderer->priv->glyphs->len == MAX_GLYPHS)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
flush_glyphs (xftrenderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xftrenderer->priv->glyph_font = g_object_ref (font);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
gs.x = pixel_x;
|
|
Packit |
0ec9dd |
gs.y = pixel_y;
|
|
Packit |
0ec9dd |
gs.glyph = glyph;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_array_append_val (xftrenderer->priv->glyphs, gs);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static gboolean
|
|
Packit |
0ec9dd |
point_in_bounds (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
gint x,
|
|
Packit |
0ec9dd |
gint y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
gdouble pixel_x = (x * renderer->matrix->xx + y * renderer->matrix->xy) / PANGO_SCALE + renderer->matrix->x0;
|
|
Packit |
0ec9dd |
gdouble pixel_y = (x * renderer->matrix->yx + y * renderer->matrix->yy) / PANGO_SCALE + renderer->matrix->y0;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
return (pixel_x >= -32768. && pixel_x < 32768. &&
|
|
Packit |
0ec9dd |
pixel_y >= -32768. && pixel_y < 32768.);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static gboolean
|
|
Packit |
0ec9dd |
box_in_bounds (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
gint x,
|
|
Packit |
0ec9dd |
gint y,
|
|
Packit |
0ec9dd |
gint width,
|
|
Packit |
0ec9dd |
gint height)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
if (!renderer->matrix)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
#define COORD_MIN (PANGO_SCALE * -16384 - PANGO_SCALE / 2)
|
|
Packit |
0ec9dd |
#define COORD_MAX (PANGO_SCALE * 32767 + PANGO_SCALE / 2 - 1)
|
|
Packit |
0ec9dd |
return (x >= COORD_MIN && x + width <= COORD_MAX &&
|
|
Packit |
0ec9dd |
y >= COORD_MIN && y + width <= COORD_MAX);
|
|
Packit |
0ec9dd |
#undef COORD_MIN
|
|
Packit |
0ec9dd |
#undef COORD_MAX
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
else
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
return (point_in_bounds (renderer, x, y) &&
|
|
Packit |
0ec9dd |
point_in_bounds (renderer, x + width, y) &&
|
|
Packit |
0ec9dd |
point_in_bounds (renderer, x + width, y + height) &&
|
|
Packit |
0ec9dd |
point_in_bounds (renderer, x, y + height));
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
get_total_matrix (PangoMatrix *total,
|
|
Packit |
0ec9dd |
const PangoMatrix *global,
|
|
Packit |
0ec9dd |
double x,
|
|
Packit |
0ec9dd |
double y,
|
|
Packit |
0ec9dd |
double width,
|
|
Packit |
0ec9dd |
double height)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoMatrix local = PANGO_MATRIX_INIT;
|
|
Packit |
0ec9dd |
gdouble angle = atan2 (height, width);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_matrix_translate (&local, x, y);
|
|
Packit |
0ec9dd |
pango_matrix_rotate (&local, -angle * (180. / G_PI));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
*total = *global;
|
|
Packit |
0ec9dd |
pango_matrix_concat (total, &local);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
draw_box (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
gint line_width,
|
|
Packit |
0ec9dd |
gint x,
|
|
Packit |
0ec9dd |
gint y,
|
|
Packit |
0ec9dd |
gint width,
|
|
Packit |
0ec9dd |
gint height,
|
|
Packit |
0ec9dd |
gboolean invalid)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
pango_renderer_draw_rectangle (renderer, PANGO_RENDER_PART_FOREGROUND,
|
|
Packit |
0ec9dd |
x, y, width, line_width);
|
|
Packit |
0ec9dd |
pango_renderer_draw_rectangle (renderer, PANGO_RENDER_PART_FOREGROUND,
|
|
Packit |
0ec9dd |
x, y + line_width, line_width, height - line_width * 2);
|
|
Packit |
0ec9dd |
pango_renderer_draw_rectangle (renderer, PANGO_RENDER_PART_FOREGROUND,
|
|
Packit |
0ec9dd |
x + width - line_width, y + line_width, line_width, height - line_width * 2);
|
|
Packit |
0ec9dd |
pango_renderer_draw_rectangle (renderer, PANGO_RENDER_PART_FOREGROUND,
|
|
Packit |
0ec9dd |
x, y + height - line_width, width, line_width);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (invalid)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
int length;
|
|
Packit |
0ec9dd |
double in_width, in_height;
|
|
Packit |
0ec9dd |
PangoMatrix orig_matrix = PANGO_MATRIX_INIT, new_matrix;
|
|
Packit |
0ec9dd |
const PangoMatrix *porig_matrix;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
in_width = pango_units_to_double (width - 2 * line_width);
|
|
Packit |
0ec9dd |
in_height = pango_units_to_double (height - 2 * line_width);
|
|
Packit |
0ec9dd |
length = PANGO_SCALE * sqrt (in_width*in_width + in_height*in_height);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
porig_matrix = pango_renderer_get_matrix (renderer);
|
|
Packit |
0ec9dd |
if (porig_matrix)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
orig_matrix = *porig_matrix;
|
|
Packit |
0ec9dd |
porig_matrix = &orig_matrix;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
get_total_matrix (&new_matrix, &orig_matrix,
|
|
Packit |
0ec9dd |
pango_units_to_double (x + line_width), pango_units_to_double (y + line_width),
|
|
Packit |
0ec9dd |
in_width, in_height);
|
|
Packit |
0ec9dd |
pango_renderer_set_matrix (renderer, &new_matrix);
|
|
Packit |
0ec9dd |
pango_renderer_draw_rectangle (renderer, PANGO_RENDER_PART_FOREGROUND,
|
|
Packit |
0ec9dd |
0, -line_width / 2, length, line_width);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
get_total_matrix (&new_matrix, &orig_matrix,
|
|
Packit |
0ec9dd |
pango_units_to_double (x + line_width), pango_units_to_double (y + height - line_width),
|
|
Packit |
0ec9dd |
in_width, -in_height);
|
|
Packit |
0ec9dd |
pango_renderer_set_matrix (renderer, &new_matrix);
|
|
Packit |
0ec9dd |
pango_renderer_draw_rectangle (renderer, PANGO_RENDER_PART_FOREGROUND,
|
|
Packit |
0ec9dd |
0, -line_width / 2, length, line_width);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_renderer_set_matrix (renderer, porig_matrix);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
_pango_xft_renderer_draw_box_glyph (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoGlyphInfo *gi,
|
|
Packit |
0ec9dd |
int glyph_x,
|
|
Packit |
0ec9dd |
int glyph_y,
|
|
Packit |
0ec9dd |
gboolean invalid)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
int x = glyph_x + PANGO_SCALE;
|
|
Packit |
0ec9dd |
int y = glyph_y - PANGO_SCALE * (PANGO_UNKNOWN_GLYPH_HEIGHT - 1);
|
|
Packit |
0ec9dd |
int width = gi->geometry.width - PANGO_SCALE * 2;
|
|
Packit |
0ec9dd |
int height = PANGO_SCALE * (PANGO_UNKNOWN_GLYPH_HEIGHT - 2);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (box_in_bounds (renderer, x, y, width, height))
|
|
Packit |
0ec9dd |
draw_box (renderer, PANGO_SCALE, x, y, width, height, invalid);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
_pango_xft_renderer_draw_unknown_glyph (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoXftFont *xfont,
|
|
Packit |
0ec9dd |
XftFont *xft_font,
|
|
Packit |
0ec9dd |
PangoGlyphInfo *gi,
|
|
Packit |
0ec9dd |
int glyph_x,
|
|
Packit |
0ec9dd |
int glyph_y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
char buf[7];
|
|
Packit |
0ec9dd |
int ys[3];
|
|
Packit |
0ec9dd |
int xs[4];
|
|
Packit |
0ec9dd |
int row, col;
|
|
Packit |
0ec9dd |
int cols;
|
|
Packit |
0ec9dd |
gunichar ch;
|
|
Packit |
0ec9dd |
gboolean invalid_input;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
PangoFont *mini_font;
|
|
Packit |
0ec9dd |
XftFont *mini_xft_font;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
ch = gi->glyph & ~PANGO_GLYPH_UNKNOWN_FLAG;
|
|
Packit |
0ec9dd |
if (G_UNLIKELY (gi->glyph == PANGO_GLYPH_INVALID_INPUT || ch > 0x10FFFF))
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
invalid_input = TRUE;
|
|
Packit |
0ec9dd |
cols = 1;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
else
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
invalid_input = FALSE;
|
|
Packit |
0ec9dd |
cols = ch > 0xffff ? 3 : 2;
|
|
Packit |
0ec9dd |
g_snprintf (buf, sizeof(buf), (ch > 0xffff) ? "%06X" : "%04X", ch);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
mini_font = _pango_xft_font_get_mini_font (xfont);
|
|
Packit |
0ec9dd |
mini_xft_font = pango_xft_font_get_font (mini_font);
|
|
Packit |
0ec9dd |
if (!mini_xft_font)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
_pango_xft_renderer_draw_box_glyph (renderer, gi, glyph_x, glyph_y, invalid_input);
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
ys[0] = glyph_y - PANGO_SCALE * xft_font->ascent + PANGO_SCALE * (((xft_font->ascent + xft_font->descent) - (xfont->mini_height * 2 + xfont->mini_pad * 5 + PANGO_SCALE / 2) / PANGO_SCALE) / 2);
|
|
Packit |
0ec9dd |
ys[1] = ys[0] + 2 * xfont->mini_pad + xfont->mini_height;
|
|
Packit |
0ec9dd |
ys[2] = ys[1] + xfont->mini_height + xfont->mini_pad;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xs[0] = glyph_x;
|
|
Packit |
0ec9dd |
xs[1] = xs[0] + 2 * xfont->mini_pad;
|
|
Packit |
0ec9dd |
xs[2] = xs[1] + xfont->mini_width + xfont->mini_pad;
|
|
Packit |
0ec9dd |
xs[3] = xs[2] + xfont->mini_width + xfont->mini_pad;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (box_in_bounds (renderer,
|
|
Packit |
0ec9dd |
xs[0], ys[0],
|
|
Packit |
0ec9dd |
xfont->mini_width * cols + xfont->mini_pad * (2 * cols + 1),
|
|
Packit |
0ec9dd |
xfont->mini_height * 2 + xfont->mini_pad * 5))
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
if (xfont->mini_pad)
|
|
Packit |
0ec9dd |
draw_box (renderer, xfont->mini_pad,
|
|
Packit |
0ec9dd |
xs[0], ys[0],
|
|
Packit |
0ec9dd |
xfont->mini_width * cols + xfont->mini_pad * (2 * cols + 1),
|
|
Packit |
0ec9dd |
xfont->mini_height * 2 + xfont->mini_pad * 5,
|
|
Packit |
0ec9dd |
invalid_input);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (invalid_input)
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
for (row = 0; row < 2; row++)
|
|
Packit |
0ec9dd |
for (col = 0; col < cols; col++)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
draw_glyph (renderer, mini_font,
|
|
Packit |
0ec9dd |
XftCharIndex (NULL, mini_xft_font,
|
|
Packit |
0ec9dd |
buf[row * cols + col] & 0xff),
|
|
Packit |
0ec9dd |
xs[col+1],
|
|
Packit |
0ec9dd |
ys[row+1]);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_draw_glyphs (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoFont *font,
|
|
Packit |
0ec9dd |
PangoGlyphString *glyphs,
|
|
Packit |
0ec9dd |
int x,
|
|
Packit |
0ec9dd |
int y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftFont *xfont = PANGO_XFT_FONT (font);
|
|
Packit |
0ec9dd |
PangoFcFont *fcfont = PANGO_FC_FONT (font);
|
|
Packit |
0ec9dd |
XftFont *xft_font = pango_xft_font_get_font (font);
|
|
Packit |
0ec9dd |
int i;
|
|
Packit |
0ec9dd |
int x_off = 0;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (!fcfont)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
for (i=0; i<glyphs->num_glyphs; i++)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoGlyphInfo *gi = &glyphs->glyphs[i];
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (gi->glyph != PANGO_GLYPH_EMPTY)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
int glyph_x = x + x_off + gi->geometry.x_offset;
|
|
Packit |
0ec9dd |
int glyph_y = y + gi->geometry.y_offset;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
_pango_xft_renderer_draw_unknown_glyph (renderer,
|
|
Packit |
0ec9dd |
xfont,
|
|
Packit |
0ec9dd |
xft_font,
|
|
Packit |
0ec9dd |
gi,
|
|
Packit |
0ec9dd |
glyph_x,
|
|
Packit |
0ec9dd |
glyph_y);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
x_off += gi->geometry.width;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (!fcfont->fontmap) /* Display closed */
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
for (i=0; i<glyphs->num_glyphs; i++)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoGlyphInfo *gi = &glyphs->glyphs[i];
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (gi->glyph != PANGO_GLYPH_EMPTY)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
int glyph_x = x + x_off + gi->geometry.x_offset;
|
|
Packit |
0ec9dd |
int glyph_y = y + gi->geometry.y_offset;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (gi->glyph & PANGO_GLYPH_UNKNOWN_FLAG)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
_pango_xft_renderer_draw_unknown_glyph (renderer,
|
|
Packit |
0ec9dd |
xfont,
|
|
Packit |
0ec9dd |
xft_font,
|
|
Packit |
0ec9dd |
gi,
|
|
Packit |
0ec9dd |
glyph_x,
|
|
Packit |
0ec9dd |
glyph_y);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
else
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
draw_glyph (renderer, font, gi->glyph, glyph_x, glyph_y);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
x_off += gi->geometry.width;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
flush_trapezoids (PangoXftRenderer *xftrenderer)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
if (!xftrenderer->priv->trapezoids ||
|
|
Packit |
0ec9dd |
xftrenderer->priv->trapezoids->len == 0)
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
PANGO_XFT_RENDERER_GET_CLASS (xftrenderer)->composite_trapezoids (xftrenderer,
|
|
Packit |
0ec9dd |
xftrenderer->priv->trapezoid_part,
|
|
Packit |
0ec9dd |
(XTrapezoid *)xftrenderer->priv->trapezoids->data,
|
|
Packit |
0ec9dd |
xftrenderer->priv->trapezoids->len);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_array_set_size (xftrenderer->priv->trapezoids, 0);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_draw_trapezoid (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoRenderPart part,
|
|
Packit |
0ec9dd |
double y1,
|
|
Packit |
0ec9dd |
double x11,
|
|
Packit |
0ec9dd |
double x21,
|
|
Packit |
0ec9dd |
double y2,
|
|
Packit |
0ec9dd |
double x12,
|
|
Packit |
0ec9dd |
double x22)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer = PANGO_XFT_RENDERER (renderer);
|
|
Packit |
0ec9dd |
XTrapezoid trap;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
flush_glyphs (xftrenderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (!xftrenderer->priv->trapezoids)
|
|
Packit |
0ec9dd |
xftrenderer->priv->trapezoids = g_array_new (FALSE, FALSE,
|
|
Packit |
0ec9dd |
sizeof (XTrapezoid));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (xftrenderer->draw)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
if (xftrenderer->priv->trapezoids->len > 0 &&
|
|
Packit |
0ec9dd |
xftrenderer->priv->trapezoid_part != part)
|
|
Packit |
0ec9dd |
flush_trapezoids (xftrenderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xftrenderer->priv->trapezoid_part = part;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
trap.top = XDoubleToFixed (y1);
|
|
Packit |
0ec9dd |
trap.bottom = XDoubleToFixed (y2);
|
|
Packit |
0ec9dd |
trap.left.p1.x = XDoubleToFixed (x11);
|
|
Packit |
0ec9dd |
trap.left.p1.y = XDoubleToFixed (y1);
|
|
Packit |
0ec9dd |
trap.left.p2.x = XDoubleToFixed (x12);
|
|
Packit |
0ec9dd |
trap.left.p2.y = XDoubleToFixed (y2);
|
|
Packit |
0ec9dd |
trap.right.p1.x = XDoubleToFixed (x21);
|
|
Packit |
0ec9dd |
trap.right.p1.y = XDoubleToFixed (y1);
|
|
Packit |
0ec9dd |
trap.right.p2.x = XDoubleToFixed (x22);
|
|
Packit |
0ec9dd |
trap.right.p2.y = XDoubleToFixed (y2);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_array_append_val (xftrenderer->priv->trapezoids, trap);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_part_changed (PangoRenderer *renderer,
|
|
Packit |
0ec9dd |
PangoRenderPart part)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer = PANGO_XFT_RENDERER (renderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (part == PANGO_RENDER_PART_FOREGROUND)
|
|
Packit |
0ec9dd |
flush_glyphs (xftrenderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (part == xftrenderer->priv->trapezoid_part)
|
|
Packit |
0ec9dd |
flush_trapezoids (xftrenderer);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_end (PangoRenderer *renderer)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer = PANGO_XFT_RENDERER (renderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
flush_glyphs (xftrenderer);
|
|
Packit |
0ec9dd |
flush_trapezoids (xftrenderer);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_real_composite_trapezoids (PangoXftRenderer *xftrenderer,
|
|
Packit |
0ec9dd |
PangoRenderPart part,
|
|
Packit |
0ec9dd |
XTrapezoid *trapezoids,
|
|
Packit |
0ec9dd |
int n_trapezoids)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
Picture src_picture;
|
|
Packit |
0ec9dd |
Picture dest_picture;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (!XftDefaultHasRender (xftrenderer->display))
|
|
Packit |
0ec9dd |
return;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
if (xftrenderer->priv->src_picture != None)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
src_picture = xftrenderer->priv->src_picture;
|
|
Packit |
0ec9dd |
dest_picture = xftrenderer->priv->dest_picture;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
else
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
XftColor xft_color;
|
|
Packit |
0ec9dd |
PangoColor *color = pango_renderer_get_color (PANGO_RENDERER (xftrenderer),
|
|
Packit |
0ec9dd |
part);
|
|
Packit |
0ec9dd |
if (!color)
|
|
Packit |
0ec9dd |
color = &xftrenderer->priv->default_color;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xft_color.color.red = color->red;
|
|
Packit |
0ec9dd |
xft_color.color.green = color->green;
|
|
Packit |
0ec9dd |
xft_color.color.blue = color->blue;
|
|
Packit |
0ec9dd |
xft_color.color.alpha = xftrenderer->priv->alpha;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
src_picture = XftDrawSrcPicture (xftrenderer->draw, &xft_color);
|
|
Packit |
0ec9dd |
dest_picture = XftDrawPicture (xftrenderer->draw);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
XRenderCompositeTrapezoids (xftrenderer->display,
|
|
Packit |
0ec9dd |
PictOpOver,
|
|
Packit |
0ec9dd |
src_picture, dest_picture,
|
|
Packit |
0ec9dd |
xftrenderer->priv->mask_format,
|
|
Packit |
0ec9dd |
0, 0,
|
|
Packit |
0ec9dd |
trapezoids, n_trapezoids);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
pango_xft_renderer_real_composite_glyphs (PangoXftRenderer *xftrenderer,
|
|
Packit |
0ec9dd |
XftFont *xft_font,
|
|
Packit |
0ec9dd |
XftGlyphSpec *glyphs,
|
|
Packit |
0ec9dd |
int n_glyphs)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
if (xftrenderer->priv->src_picture != None)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
XftGlyphSpecRender (xftrenderer->display, PictOpOver,
|
|
Packit |
0ec9dd |
xftrenderer->priv->src_picture,
|
|
Packit |
0ec9dd |
xft_font,
|
|
Packit |
0ec9dd |
xftrenderer->priv->dest_picture, 0, 0,
|
|
Packit |
0ec9dd |
glyphs, n_glyphs);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
else
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
XftColor xft_color;
|
|
Packit |
0ec9dd |
PangoColor *color = pango_renderer_get_color (PANGO_RENDERER (xftrenderer),
|
|
Packit |
0ec9dd |
PANGO_RENDER_PART_FOREGROUND);
|
|
Packit |
0ec9dd |
if (!color)
|
|
Packit |
0ec9dd |
color = &xftrenderer->priv->default_color;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xft_color.color.red = color->red;
|
|
Packit |
0ec9dd |
xft_color.color.green = color->green;
|
|
Packit |
0ec9dd |
xft_color.color.blue = color->blue;
|
|
Packit |
0ec9dd |
xft_color.color.alpha = xftrenderer->priv->alpha;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
XftDrawGlyphSpec (xftrenderer->draw, &xft_color,
|
|
Packit |
0ec9dd |
xft_font,
|
|
Packit |
0ec9dd |
glyphs, n_glyphs);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static PangoRenderer *
|
|
Packit |
0ec9dd |
get_renderer (PangoFontMap *fontmap,
|
|
Packit |
0ec9dd |
XftDraw *draw,
|
|
Packit |
0ec9dd |
XftColor *color)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoRenderer *renderer;
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer;
|
|
Packit |
0ec9dd |
PangoColor pango_color;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
renderer = _pango_xft_font_map_get_renderer (PANGO_XFT_FONT_MAP (fontmap));
|
|
Packit |
0ec9dd |
xftrenderer = PANGO_XFT_RENDERER (renderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_draw (xftrenderer, draw);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_color.red = color->color.red;
|
|
Packit |
0ec9dd |
pango_color.green = color->color.green;
|
|
Packit |
0ec9dd |
pango_color.blue = color->color.blue;
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_default_color (xftrenderer, &pango_color);
|
|
Packit |
0ec9dd |
xftrenderer->priv->alpha = color->color.alpha;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
return renderer;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
static void
|
|
Packit |
0ec9dd |
release_renderer (PangoRenderer *renderer)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer = PANGO_XFT_RENDERER (renderer);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xftrenderer->priv->alpha = 0xffff;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_render_layout:
|
|
Packit |
0ec9dd |
* @draw: an #XftDraw
|
|
Packit |
0ec9dd |
* @color: the foreground color in which to draw the layout
|
|
Packit |
0ec9dd |
* (may be overridden by color attributes)
|
|
Packit |
0ec9dd |
* @layout: a #PangoLayout
|
|
Packit |
0ec9dd |
* @x: the X position of the left of the layout (in Pango units)
|
|
Packit |
0ec9dd |
* @y: the Y position of the top of the layout (in Pango units)
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Render a #PangoLayout onto a #XftDraw
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Since: 1.8
|
|
Packit |
0ec9dd |
*/
|
|
Packit |
0ec9dd |
void
|
|
Packit |
0ec9dd |
pango_xft_render_layout (XftDraw *draw,
|
|
Packit |
0ec9dd |
XftColor *color,
|
|
Packit |
0ec9dd |
PangoLayout *layout,
|
|
Packit |
0ec9dd |
int x,
|
|
Packit |
0ec9dd |
int y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoContext *context;
|
|
Packit |
0ec9dd |
PangoFontMap *fontmap;
|
|
Packit |
0ec9dd |
PangoRenderer *renderer;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_return_if_fail (draw != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (color != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (PANGO_IS_LAYOUT (layout));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
context = pango_layout_get_context (layout);
|
|
Packit |
0ec9dd |
fontmap = pango_context_get_font_map (context);
|
|
Packit |
0ec9dd |
renderer = get_renderer (fontmap, draw, color);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_renderer_draw_layout (renderer, layout, x, y);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
release_renderer (renderer);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_render_layout_line:
|
|
Packit |
0ec9dd |
* @draw: an #XftDraw
|
|
Packit |
0ec9dd |
* @color: the foreground color in which to draw the layout line
|
|
Packit |
0ec9dd |
* (may be overridden by color attributes)
|
|
Packit |
0ec9dd |
* @line: a #PangoLayoutLine
|
|
Packit |
0ec9dd |
* @x: the x position of start of string (in Pango units)
|
|
Packit |
0ec9dd |
* @y: the y position of baseline (in Pango units)
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Render a #PangoLayoutLine onto a #XftDraw
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Since: 1.8
|
|
Packit |
0ec9dd |
*/
|
|
Packit |
0ec9dd |
void
|
|
Packit |
0ec9dd |
pango_xft_render_layout_line (XftDraw *draw,
|
|
Packit |
0ec9dd |
XftColor *color,
|
|
Packit |
0ec9dd |
PangoLayoutLine *line,
|
|
Packit |
0ec9dd |
int x,
|
|
Packit |
0ec9dd |
int y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoContext *context;
|
|
Packit |
0ec9dd |
PangoFontMap *fontmap;
|
|
Packit |
0ec9dd |
PangoRenderer *renderer;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_return_if_fail (draw != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (color != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (line != NULL);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
context = pango_layout_get_context (line->layout);
|
|
Packit |
0ec9dd |
fontmap = pango_context_get_font_map (context);
|
|
Packit |
0ec9dd |
renderer = get_renderer (fontmap, draw, color);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_renderer_draw_layout_line (renderer, line, x, y);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
release_renderer (renderer);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_render_transformed:
|
|
Packit |
0ec9dd |
* @draw: an #XftDraw
|
|
Packit |
0ec9dd |
* @color: the color in which to draw the glyphs
|
|
Packit |
0ec9dd |
* @font: the font in which to draw the string
|
|
Packit |
0ec9dd |
* @matrix: (nullable): a #PangoMatrix, or %NULL to use an identity
|
|
Packit |
0ec9dd |
* transformation
|
|
Packit |
0ec9dd |
* @glyphs: the glyph string to draw
|
|
Packit |
0ec9dd |
* @x: the x position of the start of the string (in Pango
|
|
Packit |
0ec9dd |
* units in user space coordinates)
|
|
Packit |
0ec9dd |
* @y: the y position of the baseline (in Pango units
|
|
Packit |
0ec9dd |
* in user space coordinates)
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Renders a #PangoGlyphString onto a #XftDraw, possibly
|
|
Packit |
0ec9dd |
* transforming the layed-out coordinates through a transformation
|
|
Packit |
0ec9dd |
* matrix. Note that the transformation matrix for @font is not
|
|
Packit |
0ec9dd |
* changed, so to produce correct rendering results, the @font
|
|
Packit |
0ec9dd |
* must have been loaded using a #PangoContext with an identical
|
|
Packit |
0ec9dd |
* transformation matrix to that passed in to this function.
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Since: 1.8
|
|
Packit |
0ec9dd |
**/
|
|
Packit |
0ec9dd |
void
|
|
Packit |
0ec9dd |
pango_xft_render_transformed (XftDraw *draw,
|
|
Packit |
0ec9dd |
XftColor *color,
|
|
Packit |
0ec9dd |
PangoMatrix *matrix,
|
|
Packit |
0ec9dd |
PangoFont *font,
|
|
Packit |
0ec9dd |
PangoGlyphString *glyphs,
|
|
Packit |
0ec9dd |
int x,
|
|
Packit |
0ec9dd |
int y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoFontMap *fontmap;
|
|
Packit |
0ec9dd |
PangoRenderer *renderer;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_return_if_fail (draw != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (color != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (PANGO_XFT_IS_FONT (font));
|
|
Packit |
0ec9dd |
g_return_if_fail (glyphs != NULL);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
fontmap = PANGO_FC_FONT (font)->fontmap;
|
|
Packit |
0ec9dd |
renderer = get_renderer (fontmap, draw, color);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_renderer_set_matrix (renderer, matrix);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_renderer_draw_glyphs (renderer, font, glyphs, x, y);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
release_renderer (renderer);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_render:
|
|
Packit |
0ec9dd |
* @draw: the <type>XftDraw</type> object.
|
|
Packit |
0ec9dd |
* @color: the color in which to draw the string
|
|
Packit |
0ec9dd |
* @font: the font in which to draw the string
|
|
Packit |
0ec9dd |
* @glyphs: the glyph string to draw
|
|
Packit |
0ec9dd |
* @x: the x position of start of string (in pixels)
|
|
Packit |
0ec9dd |
* @y: the y position of baseline (in pixels)
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Renders a #PangoGlyphString onto an <type>XftDraw</type> object wrapping an X drawable.
|
|
Packit |
0ec9dd |
*/
|
|
Packit |
0ec9dd |
void
|
|
Packit |
0ec9dd |
pango_xft_render (XftDraw *draw,
|
|
Packit |
0ec9dd |
XftColor *color,
|
|
Packit |
0ec9dd |
PangoFont *font,
|
|
Packit |
0ec9dd |
PangoGlyphString *glyphs,
|
|
Packit |
0ec9dd |
gint x,
|
|
Packit |
0ec9dd |
gint y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
g_return_if_fail (draw != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (color != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (PANGO_XFT_IS_FONT (font));
|
|
Packit |
0ec9dd |
g_return_if_fail (glyphs != NULL);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_xft_render_transformed (draw, color, NULL, font, glyphs,
|
|
Packit |
0ec9dd |
x * PANGO_SCALE, y * PANGO_SCALE);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_picture_render:
|
|
Packit |
0ec9dd |
* @display: an X display
|
|
Packit |
0ec9dd |
* @src_picture: the source picture to draw the string with
|
|
Packit |
0ec9dd |
* @dest_picture: the destination picture to draw the string onto
|
|
Packit |
0ec9dd |
* @font: the font in which to draw the string
|
|
Packit |
0ec9dd |
* @glyphs: the glyph string to draw
|
|
Packit |
0ec9dd |
* @x: the x position of start of string (in pixels)
|
|
Packit |
0ec9dd |
* @y: the y position of baseline (in pixels)
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Renders a #PangoGlyphString onto an Xrender <type>Picture</type> object.
|
|
Packit |
0ec9dd |
*/
|
|
Packit |
0ec9dd |
void
|
|
Packit |
0ec9dd |
pango_xft_picture_render (Display *display,
|
|
Packit |
0ec9dd |
Picture src_picture,
|
|
Packit |
0ec9dd |
Picture dest_picture,
|
|
Packit |
0ec9dd |
PangoFont *font,
|
|
Packit |
0ec9dd |
PangoGlyphString *glyphs,
|
|
Packit |
0ec9dd |
gint x,
|
|
Packit |
0ec9dd |
gint y)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoFontMap *fontmap;
|
|
Packit |
0ec9dd |
PangoRenderer *renderer;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
g_return_if_fail (display != NULL);
|
|
Packit |
0ec9dd |
g_return_if_fail (src_picture != None);
|
|
Packit |
0ec9dd |
g_return_if_fail (dest_picture != None);
|
|
Packit |
0ec9dd |
g_return_if_fail (PANGO_XFT_IS_FONT (font));
|
|
Packit |
0ec9dd |
g_return_if_fail (glyphs != NULL);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
fontmap = PANGO_FC_FONT (font)->fontmap;
|
|
Packit |
0ec9dd |
renderer = _pango_xft_font_map_get_renderer (PANGO_XFT_FONT_MAP (fontmap));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_pictures (PANGO_XFT_RENDERER (renderer),
|
|
Packit |
0ec9dd |
src_picture, dest_picture);
|
|
Packit |
0ec9dd |
pango_renderer_set_matrix (renderer, NULL);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_renderer_draw_glyphs (renderer, font, glyphs, x * PANGO_SCALE, y * PANGO_SCALE);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_pictures (PANGO_XFT_RENDERER (renderer),
|
|
Packit |
0ec9dd |
None, None);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_renderer_new:
|
|
Packit |
0ec9dd |
* @display: an X display
|
|
Packit |
0ec9dd |
* @screen: the index of the screen for @display to which rendering will be done
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Create a new #PangoXftRenderer to allow rendering Pango objects
|
|
Packit |
0ec9dd |
* with the Xft library. You must call pango_xft_renderer_set_draw() before
|
|
Packit |
0ec9dd |
* using the renderer.
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Return value: the newly created #PangoXftRenderer, which should
|
|
Packit |
0ec9dd |
* be freed with g_object_unref().
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Since: 1.8
|
|
Packit |
0ec9dd |
**/
|
|
Packit |
0ec9dd |
PangoRenderer *
|
|
Packit |
0ec9dd |
pango_xft_renderer_new (Display *display,
|
|
Packit |
0ec9dd |
int screen)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
PangoXftRenderer *xftrenderer;
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xftrenderer = g_object_new (PANGO_TYPE_XFT_RENDERER,
|
|
Packit |
0ec9dd |
"display", display,
|
|
Packit |
0ec9dd |
"screen", screen,
|
|
Packit |
0ec9dd |
NULL);
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
return PANGO_RENDERER (xftrenderer);
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_renderer_set_draw:
|
|
Packit |
0ec9dd |
* @xftrenderer: a #PangoXftRenderer
|
|
Packit |
0ec9dd |
* @draw: a #XftDraw
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Sets the #XftDraw object that the renderer is drawing to.
|
|
Packit |
0ec9dd |
* The renderer must not be currently active.
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Since: 1.8
|
|
Packit |
0ec9dd |
**/
|
|
Packit |
0ec9dd |
void
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_draw (PangoXftRenderer *xftrenderer,
|
|
Packit |
0ec9dd |
XftDraw *draw)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
g_return_if_fail (PANGO_IS_XFT_RENDERER (xftrenderer));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xftrenderer->draw = draw;
|
|
Packit |
0ec9dd |
}
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
/**
|
|
Packit |
0ec9dd |
* pango_xft_renderer_set_default_color:
|
|
Packit |
0ec9dd |
* @xftrenderer: a #XftRenderer
|
|
Packit |
0ec9dd |
* @default_color: the default foreground color
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Sets the default foreground color for a #XftRenderer.
|
|
Packit |
0ec9dd |
*
|
|
Packit |
0ec9dd |
* Since: 1.8
|
|
Packit |
0ec9dd |
**/
|
|
Packit |
0ec9dd |
void
|
|
Packit |
0ec9dd |
pango_xft_renderer_set_default_color (PangoXftRenderer *xftrenderer,
|
|
Packit |
0ec9dd |
PangoColor *default_color)
|
|
Packit |
0ec9dd |
{
|
|
Packit |
0ec9dd |
g_return_if_fail (PANGO_IS_XFT_RENDERER (xftrenderer));
|
|
Packit |
0ec9dd |
|
|
Packit |
0ec9dd |
xftrenderer->priv->default_color = *default_color;
|
|
Packit |
0ec9dd |
}
|