mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2024-12-25 13:11:13 +00:00
1bbf2ac7ee
Fri Sep 7 11:51:44 2001 Owen Taylor <otaylor@redhat.com> Make gdkx.h the only installed header from gdk/x11. All structures in gdk/x11 are opaque. * gdk/x11/Makefile.am gdk/x11/gdkx.h gdk/x11/gdkprivate-x11.h: Don't install gdk{drawable,pixmap,window}-x11.h. * gdk/x11/{gdkcolormap-x11.c, gdkfont-x11.c, gdkx.h, gdkvisual-x11.c: Move GdkColormapPrivateX11, GdkFontPrivateX GdkImagePrivateX11, GdkVisualClass into C files. * gdk/gdkpixmap-x11.[ch]: Make gdk_pixmap_impl_get_type() static. * gdk/x11/{gdkcolor-x11.c, gdkcursor-x11.c, gdkdrawable-x11.c, gdkfont-x11.c, gdkgc-x11.c, gdkx.h, gdkimage-x11,gdkvisual-x11.c} Add public functions to replace previously exported direct structure access. gdk_x11_colormap_get_{xdisplay,xcolormap} gdk_x11_cursor_get_{xdisplay,xcursor}, gdk_x11_drawable_get_{xdisplay,xcursor,gdk_x11_visual_get_xvisual, gdk_x11_font_get_{xdisplay,xfont}, gdk_x11_image_get_{xdisplay,ximage}, gdk_x11_gc_get_{xdisplay,ximage} * gdk/gdkprivate.h gdk/gdkinternals.h: Move GdkColorInfo, GdkEventFilter, GdkClientFilter, GdkFontPrivate to gdkinternals. Fix a number of variables and functions that were exported "accidentally" from GDK. * gdk/**.[ch]: gdk => _gdk for gdk_visual_init, gdk_events_init, gdk_input_init, gdk_dnd_init, gdk_image_exit, gdk_input_exit, gdk_windowing_exit, gdk_event_func, gdk_event_data, gdk_event_notify, gdk_queued_events, gdk_queued_tail, gdk_event_new, gdk_events_queue, gdk_events_unqueue, gdk_event_queue_find_first, gdk_event_queue_remove_link, gdk_event_queue_append, gdk_event_button_generate, gdk_debug_flags, gdk_default_filters, gdk_parent_root. * gdk/x11/{gdkevents-x11.c, gdkglobals-x11.c, gdkimage-x11.c, gdkmain-x11.c, gdkprivate-x11.h, gdk/x11/gdkwindow-x11.c}: gdk => _gdk for gdk_event_mask_table, gkd_nevent_masks, gdk_wm_window_protocols, gdk_leader_window, gdk_xgrab_window, gdk_use_xshm, gdk_input_ignore_core. * gdk/x11/xsettings-common.h (xsettings_list_insert): Add #defines to namespace functions into the private _gdk_ namespace. * gdk/gdkwindow.[ch] gdk/x11/gdkx.h: Add gdk_get_default_root_window () to replace gdk_parent_root exported variable. Adjust and deprecate GDK_ROOT_PARENT(). * demos/{testpixbuf-drawable.c,testpixbuf-save.c}: Fix GDK_ROOT_PARENT usage, remove includes of port-specific headers. * gdk/{win32,x11,fb}/gdkinput*.[ch]: s/gdk/_gdk/ for _gdk_input_gxid_host, _gdk_input_gxid_port, _gdk_input_ignore_core, gdk_input_devices, _gdk_input_windows, gdk_init_input_core. * gdk/x11/{gdkevents-x11.,c gdkglobals-x11.c, gdkmain-x11.c} docs/Changes-2.0.txt: Remove gdk_wm_protocols, gdk_wm_delete_window functions, gdk_wm_take_focus, use gdk_atom_intern() instead. * gdk/linux-fb/{gdkselection-fb.c, gdkmain-fb.c, gdkprivatefb.h} gdk/win32/{gdkselection-win32.c, gdkmgdkwin32.h, gdkprivate-win32.h} gdk/x11/{gdkselection-x11.c gdkx.h, gtkprivate-x11.h} gtk/gtkselection.c Unexport gdk_selection_property, just use gdk_atom_intern ("GDK_SELECTION"). * gdk/x11/{gdkprivate-x11.h,gdkdrawable-x11h,gdkgc-x11.c,gdkx.h}: Unexport gdk_drawable_impl_x11_get_type, gdk_gc_x11_get_type, GDK_GC_X11 cast macros, GdkGCX11 structures, GdkCursorPrivate, GdkVisualprivate, gdk_x11_gc_flush. Make a number of public exports of variables into functions to increase encapsulation. * gdk/gdkinternals.h gdk/gdkinput.h gdk/gdkevents.h gdk/linux-fb/gdkmouse-fb.c: gdk_core_pointer => _gdk_core_pointer, move to gdkinternals.h. Add gdk_device_get_core_pointer (). * gdk/gdkprivate.h gdk/gdkpango.c gdk/gdkinternals.h docs/Changes-2.0.txt: Unexport gdk_parent_root, gdk_error_code, gdk_error_warnings. * gdk/x11/{gdkcolormap-x11.c, gdkmain-x11.c, gdkx.h} docs/Changes-2.0.txt: s/gdk_screen/_gdk_screen/, add gdk_x11_get_default_screen() s/gdk_root_window/_gdk_root_window/, add gdk_x11_get_default_root_xwindow() Add gdk_x11_get_default_xdisplay(). * gdk/gdk.h gdk/gdk.c linux-fb/gdkfb.h linux-fb/gdkglobals-fb.c win32/gdkwin32.h x11/gdkglobals-x11.c gdk/x11/gdkmain-x11.c gdk/x11/gdkx.h: gdk/gdk.def: Add gdk_get/set_program_class, Don't export gdk_progclass, move --class command line option and handling to common portion of GDK. Miscellaneous fixes: * gdk/x11/gdkwindow-x11.c (gdk_window_set_icon_list): Fix g_return_val_if_fail that should have been g_return_if_fail. * gdk/gdkinternals.h gdk/gdkprivate.h: Move gdk_synthesize_window_state() to the semi-public gdkprivate.h. * gtk/gtkdnd.c (_gtk_drag_source_handle_event): Remove uneeded X11 dependency. * gdk/linux-fb/gdkmain-fb.c gdk/win32/gdkmain-win32.c gdk/TODO: Remove unused gdk_key_repeat_disable/restore. * linux-fb/gdkglobals-fb.c win32/gdkglobals-win32.c x11/gdkglobals-x11.c x11/gdkprivate-x11.h gdk/gdk.def: Remove unused gdk_null_window_warnings variable. * gdk/Makefile.am (DIST_SUBDIRS) nanox/*: cvs remove nanox; it can be retrieved from the repository; it is too far from functional to be worth having people check out; it would be easier to start from scratch, I suspect. * gdk/x11/gdkpixmap-x11.c: Fix lvalue usage of GDK_PIXMAP_XID(). * gdk/x11/gdkkeys-x11.c gdk/gdkrgb.c gdk/gdkwindow.c gdk/x11/gdkpango-x11.c gdk/x11/gdkselection-x11.c: Fix some accidentally global variables and unused global variables. * gdk/x11/gdkkeys-x11.c gdk/gdkrgb.c gdk/gdkwindow.c gdk/x11/gdkpango-x11.c gdk/x11/gdkselection-x11.c: Fix some accidentally global variables and unused global variables. Add some space for future expansion to multihead. * gdk/gdkdrawable.h: Add four reserved function pointers for future expansion of GdkDrawableClass. * gtk/gtkwindow.h gtk/gtkinvisible.h: Add reserved pointer where we can put a GdkScreen * later.
811 lines
24 KiB
C
811 lines
24 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
* Copyright (C) 2000 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser 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.
|
|
*/
|
|
|
|
#include "gdkcolor.h"
|
|
#include "gdkgc.h"
|
|
#include "gdkpango.h"
|
|
#include "gdkrgb.h"
|
|
#include "gdkprivate.h"
|
|
|
|
#define GDK_INFO_KEY "gdk-info"
|
|
|
|
typedef struct _GdkPangoContextInfo GdkPangoContextInfo;
|
|
|
|
struct _GdkPangoContextInfo
|
|
{
|
|
GdkColormap *colormap;
|
|
};
|
|
|
|
static PangoAttrType gdk_pango_attr_stipple_type;
|
|
static PangoAttrType gdk_pango_attr_embossed_type;
|
|
|
|
static void gdk_pango_get_item_properties (PangoItem *item,
|
|
PangoUnderline *uline,
|
|
gboolean *strikethrough,
|
|
gint *rise,
|
|
PangoColor *fg_color,
|
|
gboolean *fg_set,
|
|
PangoColor *bg_color,
|
|
gboolean *bg_set,
|
|
gboolean *embossed,
|
|
GdkBitmap **stipple,
|
|
gboolean *shape_set,
|
|
PangoRectangle *ink_rect,
|
|
PangoRectangle *logical_rect);
|
|
|
|
static void
|
|
gdk_pango_context_destroy (GdkPangoContextInfo *info)
|
|
{
|
|
if (info->colormap)
|
|
gdk_colormap_unref (info->colormap);
|
|
g_free (info);
|
|
}
|
|
|
|
static GdkPangoContextInfo *
|
|
gdk_pango_context_get_info (PangoContext *context, gboolean create)
|
|
{
|
|
GdkPangoContextInfo *info =
|
|
g_object_get_qdata (G_OBJECT (context),
|
|
g_quark_try_string (GDK_INFO_KEY));
|
|
if (!info && create)
|
|
{
|
|
info = g_new (GdkPangoContextInfo, 1);
|
|
info->colormap = NULL;
|
|
|
|
g_object_set_qdata_full (G_OBJECT (context),
|
|
g_quark_from_static_string (GDK_INFO_KEY),
|
|
info, (GDestroyNotify)gdk_pango_context_destroy);
|
|
}
|
|
|
|
return info;
|
|
}
|
|
|
|
static GdkGC *
|
|
gdk_pango_get_gc (PangoContext *context,
|
|
PangoColor *fg_color,
|
|
GdkBitmap *stipple,
|
|
GdkGC *base_gc)
|
|
{
|
|
GdkGC *result;
|
|
GdkPangoContextInfo *info;
|
|
|
|
g_return_val_if_fail (context != NULL, NULL);
|
|
|
|
info = gdk_pango_context_get_info (context, FALSE);
|
|
|
|
if (info == NULL || info->colormap == NULL)
|
|
{
|
|
g_warning ("you must set the colormap on a PangoContext before using it to draw a layout");
|
|
return NULL;
|
|
}
|
|
|
|
result = gdk_gc_new (gdk_get_default_root_window ());
|
|
gdk_gc_copy (result, base_gc);
|
|
|
|
if (fg_color)
|
|
{
|
|
GdkColor color;
|
|
|
|
color.red = fg_color->red;
|
|
color.green = fg_color->green;
|
|
color.blue = fg_color->blue;
|
|
|
|
gdk_rgb_find_color (info->colormap, &color);
|
|
gdk_gc_set_foreground (result, &color);
|
|
}
|
|
|
|
if (stipple)
|
|
{
|
|
gdk_gc_set_fill (result, GDK_STIPPLED);
|
|
gdk_gc_set_stipple (result, stipple);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
gdk_pango_free_gc (PangoContext *context,
|
|
GdkGC *gc)
|
|
{
|
|
gdk_gc_unref (gc);
|
|
}
|
|
|
|
void
|
|
gdk_pango_context_set_colormap (PangoContext *context,
|
|
GdkColormap *colormap)
|
|
{
|
|
GdkPangoContextInfo *info;
|
|
|
|
g_return_if_fail (context != NULL);
|
|
|
|
info = gdk_pango_context_get_info (context, TRUE);
|
|
g_return_if_fail (info != NULL);
|
|
|
|
if (info->colormap != colormap)
|
|
{
|
|
if (info->colormap)
|
|
gdk_colormap_unref (info->colormap);
|
|
|
|
info->colormap = colormap;
|
|
|
|
if (info->colormap)
|
|
gdk_colormap_ref (info->colormap);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gdk_draw_layout_line_with_colors:
|
|
* @drawable: the drawable on which to draw the line
|
|
* @gc: base graphics to use
|
|
* @x: the x position of start of string (in pixels)
|
|
* @y: the y position of baseline (in pixels)
|
|
* @line: a #PangoLayoutLine
|
|
* @foreground: foreground override color, or %NULL for none
|
|
* @background: background override color, or %NULL for none
|
|
*
|
|
* Render a #PangoLayoutLine onto a #GdkDrawable, overriding the
|
|
* layout's normal colors with @foreground and/or @background.
|
|
* @foreground and @background need not be allocated.
|
|
*/
|
|
void
|
|
gdk_draw_layout_line_with_colors (GdkDrawable *drawable,
|
|
GdkGC *gc,
|
|
gint x,
|
|
gint y,
|
|
PangoLayoutLine *line,
|
|
GdkColor *foreground,
|
|
GdkColor *background)
|
|
{
|
|
GSList *tmp_list = line->runs;
|
|
PangoRectangle overall_rect;
|
|
PangoRectangle logical_rect;
|
|
PangoRectangle ink_rect;
|
|
PangoContext *context;
|
|
gint x_off = 0;
|
|
gint rise = 0;
|
|
gboolean embossed;
|
|
GdkBitmap *stipple;
|
|
|
|
g_return_if_fail (GDK_IS_DRAWABLE (drawable));
|
|
g_return_if_fail (GDK_IS_GC (gc));
|
|
g_return_if_fail (line != NULL);
|
|
|
|
context = pango_layout_get_context (line->layout);
|
|
|
|
pango_layout_line_get_extents (line,NULL, &overall_rect);
|
|
|
|
while (tmp_list)
|
|
{
|
|
PangoUnderline uline = PANGO_UNDERLINE_NONE;
|
|
PangoLayoutRun *run = tmp_list->data;
|
|
PangoColor fg_color, bg_color;
|
|
gboolean strike, fg_set, bg_set, shape_set;
|
|
GdkGC *fg_gc;
|
|
gint risen_y;
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
gdk_pango_get_item_properties (run->item, &uline,
|
|
&strike,
|
|
&rise,
|
|
&fg_color, &fg_set,
|
|
&bg_color, &bg_set,
|
|
&embossed,
|
|
&stipple,
|
|
&shape_set,
|
|
&ink_rect,
|
|
&logical_rect);
|
|
|
|
/* we subtract the rise because X coordinates are upside down */
|
|
risen_y = y - rise / PANGO_SCALE;
|
|
|
|
if (!shape_set)
|
|
{
|
|
if (uline == PANGO_UNDERLINE_NONE)
|
|
pango_glyph_string_extents (run->glyphs, run->item->analysis.font,
|
|
NULL, &logical_rect);
|
|
else
|
|
pango_glyph_string_extents (run->glyphs, run->item->analysis.font,
|
|
&ink_rect, &logical_rect);
|
|
}
|
|
|
|
if (bg_set || background)
|
|
{
|
|
GdkGC *bg_gc;
|
|
PangoColor tmp = bg_color;
|
|
|
|
if (background)
|
|
{
|
|
tmp.red = background->red;
|
|
tmp.blue = background->blue;
|
|
tmp.green = background->green;
|
|
}
|
|
|
|
bg_gc = gdk_pango_get_gc (context, &tmp, stipple, gc);
|
|
|
|
gdk_draw_rectangle (drawable, bg_gc, TRUE,
|
|
x + (x_off + logical_rect.x) / PANGO_SCALE,
|
|
risen_y + overall_rect.y / PANGO_SCALE,
|
|
logical_rect.width / PANGO_SCALE,
|
|
overall_rect.height / PANGO_SCALE);
|
|
|
|
if (stipple)
|
|
gdk_gc_set_fill (bg_gc, GDK_SOLID);
|
|
|
|
gdk_pango_free_gc (context, bg_gc);
|
|
}
|
|
|
|
if (fg_set || stipple || foreground)
|
|
{
|
|
PangoColor tmp = fg_color;
|
|
|
|
if (foreground)
|
|
{
|
|
tmp.red = foreground->red;
|
|
tmp.blue = foreground->blue;
|
|
tmp.green = foreground->green;
|
|
}
|
|
|
|
fg_gc = gdk_pango_get_gc (context, fg_set ? &tmp : NULL,
|
|
stipple, gc);
|
|
}
|
|
else
|
|
fg_gc = gc;
|
|
|
|
if (!shape_set)
|
|
{
|
|
gint gx, gy;
|
|
|
|
gx = x + x_off / PANGO_SCALE;
|
|
gy = risen_y;
|
|
|
|
if (embossed)
|
|
{
|
|
PangoColor color = { 65535, 65535, 65535 };
|
|
GdkGC *white_gc = gdk_pango_get_gc (context, &color, stipple, fg_gc);
|
|
gdk_draw_glyphs (drawable, white_gc, run->item->analysis.font,
|
|
gx + 1,
|
|
gy + 1,
|
|
run->glyphs);
|
|
gdk_pango_free_gc (context, white_gc);
|
|
}
|
|
|
|
gdk_draw_glyphs (drawable, fg_gc, run->item->analysis.font,
|
|
gx, gy,
|
|
run->glyphs);
|
|
}
|
|
|
|
switch (uline)
|
|
{
|
|
case PANGO_UNDERLINE_NONE:
|
|
break;
|
|
case PANGO_UNDERLINE_DOUBLE:
|
|
gdk_draw_line (drawable, fg_gc,
|
|
x + (x_off + ink_rect.x) / PANGO_SCALE - 1,
|
|
risen_y + 3,
|
|
x + (x_off + ink_rect.x + ink_rect.width) / PANGO_SCALE,
|
|
risen_y + 3);
|
|
/* Fall through */
|
|
case PANGO_UNDERLINE_SINGLE:
|
|
gdk_draw_line (drawable, fg_gc,
|
|
x + (x_off + ink_rect.x) / PANGO_SCALE - 1,
|
|
risen_y + 1,
|
|
x + (x_off + ink_rect.x + ink_rect.width) / PANGO_SCALE,
|
|
risen_y + 1);
|
|
break;
|
|
case PANGO_UNDERLINE_LOW:
|
|
gdk_draw_line (drawable, fg_gc,
|
|
x + (x_off + ink_rect.x) / PANGO_SCALE - 1,
|
|
risen_y + (ink_rect.y + ink_rect.height) / PANGO_SCALE + 1,
|
|
x + (x_off + ink_rect.x + ink_rect.width) / PANGO_SCALE,
|
|
risen_y + (ink_rect.y + ink_rect.height) / PANGO_SCALE + 1);
|
|
break;
|
|
}
|
|
|
|
if (strike)
|
|
{
|
|
int centerline = logical_rect.y + logical_rect.height / 2;
|
|
|
|
gdk_draw_line (drawable, fg_gc,
|
|
x + (x_off + logical_rect.x) / PANGO_SCALE - 1,
|
|
risen_y + centerline / PANGO_SCALE,
|
|
x + (x_off + logical_rect.x + logical_rect.width) / PANGO_SCALE + 1,
|
|
risen_y + centerline / PANGO_SCALE);
|
|
}
|
|
|
|
if (fg_gc != gc)
|
|
gdk_pango_free_gc (context, fg_gc);
|
|
|
|
x_off += logical_rect.width;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gdk_draw_layout_with_colors:
|
|
* @drawable: the drawable on which to draw string
|
|
* @gc: base graphics context to use
|
|
* @x: the X position of the left of the layout (in pixels)
|
|
* @y: the Y position of the top of the layout (in pixels)
|
|
* @layout: a #PangoLayout
|
|
* @foreground: foreground override color, or %NULL for none
|
|
* @background: background override color, or %NULL for none
|
|
*
|
|
* Render a #PangoLayout onto a #GdkDrawable, overriding the
|
|
* layout's normal colors with @foreground and/or @background.
|
|
* @foreground and @background need not be allocated.
|
|
*/
|
|
void
|
|
gdk_draw_layout_with_colors (GdkDrawable *drawable,
|
|
GdkGC *gc,
|
|
int x,
|
|
int y,
|
|
PangoLayout *layout,
|
|
GdkColor *foreground,
|
|
GdkColor *background)
|
|
{
|
|
PangoLayoutIter *iter;
|
|
|
|
g_return_if_fail (GDK_IS_DRAWABLE (drawable));
|
|
g_return_if_fail (GDK_IS_GC (gc));
|
|
g_return_if_fail (PANGO_IS_LAYOUT (layout));
|
|
|
|
iter = pango_layout_get_iter (layout);
|
|
|
|
do
|
|
{
|
|
PangoRectangle logical_rect;
|
|
PangoLayoutLine *line;
|
|
int baseline;
|
|
|
|
line = pango_layout_iter_get_line (iter);
|
|
|
|
pango_layout_iter_get_line_extents (iter, NULL, &logical_rect);
|
|
baseline = pango_layout_iter_get_baseline (iter);
|
|
|
|
gdk_draw_layout_line_with_colors (drawable, gc,
|
|
x + logical_rect.x / PANGO_SCALE,
|
|
y + baseline / PANGO_SCALE,
|
|
line,
|
|
foreground,
|
|
background);
|
|
}
|
|
while (pango_layout_iter_next_line (iter));
|
|
|
|
pango_layout_iter_free (iter);
|
|
}
|
|
|
|
/**
|
|
* gdk_draw_layout_line:
|
|
* @drawable: the drawable on which to draw the line
|
|
* @gc: base graphics to use
|
|
* @x: the x position of start of string (in pixels)
|
|
* @y: the y position of baseline (in pixels)
|
|
* @line: a #PangoLayoutLine
|
|
*
|
|
* Render a #PangoLayoutLine onto an GDK drawable
|
|
*/
|
|
void
|
|
gdk_draw_layout_line (GdkDrawable *drawable,
|
|
GdkGC *gc,
|
|
gint x,
|
|
gint y,
|
|
PangoLayoutLine *line)
|
|
{
|
|
g_return_if_fail (GDK_IS_DRAWABLE (drawable));
|
|
g_return_if_fail (GDK_IS_GC (gc));
|
|
g_return_if_fail (line != NULL);
|
|
|
|
gdk_draw_layout_line_with_colors (drawable, gc, x, y, line, NULL, NULL);
|
|
}
|
|
|
|
/**
|
|
* gdk_draw_layout:
|
|
* @drawable: the drawable on which to draw string
|
|
* @gc: base graphics context to use
|
|
* @x: the X position of the left of the layout (in pixels)
|
|
* @y: the Y position of the top of the layout (in pixels)
|
|
* @layout: a #PangoLayout
|
|
*
|
|
* Render a #PangoLayout onto a GDK drawable
|
|
*/
|
|
void
|
|
gdk_draw_layout (GdkDrawable *drawable,
|
|
GdkGC *gc,
|
|
int x,
|
|
int y,
|
|
PangoLayout *layout)
|
|
{
|
|
g_return_if_fail (GDK_IS_DRAWABLE (drawable));
|
|
g_return_if_fail (GDK_IS_GC (gc));
|
|
g_return_if_fail (PANGO_IS_LAYOUT (layout));
|
|
|
|
gdk_draw_layout_with_colors (drawable, gc, x, y, layout, NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
gdk_pango_get_item_properties (PangoItem *item,
|
|
PangoUnderline *uline,
|
|
gboolean *strikethrough,
|
|
gint *rise,
|
|
PangoColor *fg_color,
|
|
gboolean *fg_set,
|
|
PangoColor *bg_color,
|
|
gboolean *bg_set,
|
|
gboolean *embossed,
|
|
GdkBitmap **stipple,
|
|
gboolean *shape_set,
|
|
PangoRectangle *ink_rect,
|
|
PangoRectangle *logical_rect)
|
|
{
|
|
GSList *tmp_list = item->analysis.extra_attrs;
|
|
|
|
if (strikethrough)
|
|
*strikethrough = FALSE;
|
|
|
|
if (fg_set)
|
|
*fg_set = FALSE;
|
|
|
|
if (bg_set)
|
|
*bg_set = FALSE;
|
|
|
|
if (shape_set)
|
|
*shape_set = FALSE;
|
|
|
|
if (rise)
|
|
*rise = 0;
|
|
|
|
if (embossed)
|
|
*embossed = FALSE;
|
|
|
|
if (stipple)
|
|
*stipple = NULL;
|
|
|
|
while (tmp_list)
|
|
{
|
|
PangoAttribute *attr = tmp_list->data;
|
|
|
|
switch (attr->klass->type)
|
|
{
|
|
case PANGO_ATTR_UNDERLINE:
|
|
if (uline)
|
|
*uline = ((PangoAttrInt *)attr)->value;
|
|
break;
|
|
|
|
case PANGO_ATTR_STRIKETHROUGH:
|
|
if (strikethrough)
|
|
*strikethrough = ((PangoAttrInt *)attr)->value;
|
|
break;
|
|
|
|
case PANGO_ATTR_FOREGROUND:
|
|
if (fg_color)
|
|
*fg_color = ((PangoAttrColor *)attr)->color;
|
|
if (fg_set)
|
|
*fg_set = TRUE;
|
|
|
|
break;
|
|
|
|
case PANGO_ATTR_BACKGROUND:
|
|
if (bg_color)
|
|
*bg_color = ((PangoAttrColor *)attr)->color;
|
|
if (bg_set)
|
|
*bg_set = TRUE;
|
|
|
|
break;
|
|
|
|
case PANGO_ATTR_SHAPE:
|
|
if (shape_set)
|
|
*shape_set = TRUE;
|
|
if (logical_rect)
|
|
*logical_rect = ((PangoAttrShape *)attr)->logical_rect;
|
|
if (ink_rect)
|
|
*ink_rect = ((PangoAttrShape *)attr)->ink_rect;
|
|
break;
|
|
|
|
case PANGO_ATTR_RISE:
|
|
if (rise)
|
|
*rise = ((PangoAttrInt *)attr)->value;
|
|
break;
|
|
|
|
default:
|
|
/* stipple_type and embossed_type aren't necessarily
|
|
* initialized, but they are 0, which is an
|
|
* invalid type so won't occur.
|
|
*/
|
|
if (stipple && attr->klass->type == gdk_pango_attr_stipple_type)
|
|
{
|
|
*stipple = ((GdkPangoAttrStipple*)attr)->stipple;
|
|
}
|
|
else if (embossed && attr->klass->type == gdk_pango_attr_embossed_type)
|
|
{
|
|
*embossed = ((GdkPangoAttrEmbossed*)attr)->embossed;
|
|
}
|
|
break;
|
|
}
|
|
tmp_list = tmp_list->next;
|
|
}
|
|
}
|
|
|
|
|
|
static PangoAttribute *
|
|
gdk_pango_attr_stipple_copy (const PangoAttribute *attr)
|
|
{
|
|
const GdkPangoAttrStipple *src = (const GdkPangoAttrStipple*) attr;
|
|
|
|
return gdk_pango_attr_stipple_new (src->stipple);
|
|
}
|
|
|
|
static void
|
|
gdk_pango_attr_stipple_destroy (PangoAttribute *attr)
|
|
{
|
|
GdkPangoAttrStipple *st = (GdkPangoAttrStipple*) attr;
|
|
|
|
if (st->stipple)
|
|
g_object_unref (G_OBJECT (st->stipple));
|
|
|
|
g_free (attr);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_pango_attr_stipple_compare (const PangoAttribute *attr1,
|
|
const PangoAttribute *attr2)
|
|
{
|
|
const GdkPangoAttrStipple *a = (const GdkPangoAttrStipple*) attr1;
|
|
const GdkPangoAttrStipple *b = (const GdkPangoAttrStipple*) attr2;
|
|
|
|
return a->stipple == b->stipple;
|
|
}
|
|
|
|
/**
|
|
* gdk_pango_attr_stipple_new:
|
|
* @stipple: a bitmap to be set as stipple
|
|
*
|
|
* Creates a new attribute containing a stipple bitmap to be used when
|
|
* rendering the text.
|
|
*
|
|
* Return value: new #PangoAttribute
|
|
**/
|
|
|
|
PangoAttribute *
|
|
gdk_pango_attr_stipple_new (GdkBitmap *stipple)
|
|
{
|
|
GdkPangoAttrStipple *result;
|
|
|
|
static PangoAttrClass klass = {
|
|
0,
|
|
gdk_pango_attr_stipple_copy,
|
|
gdk_pango_attr_stipple_destroy,
|
|
gdk_pango_attr_stipple_compare
|
|
};
|
|
|
|
if (!klass.type)
|
|
klass.type = gdk_pango_attr_stipple_type =
|
|
pango_attr_type_register ("GdkPangoAttrStipple");
|
|
|
|
result = g_new (GdkPangoAttrStipple, 1);
|
|
result->attr.klass = &klass;
|
|
|
|
if (stipple)
|
|
g_object_ref (stipple);
|
|
|
|
result->stipple = stipple;
|
|
|
|
return (PangoAttribute *)result;
|
|
}
|
|
|
|
static PangoAttribute *
|
|
gdk_pango_attr_embossed_copy (const PangoAttribute *attr)
|
|
{
|
|
const GdkPangoAttrEmbossed *e = (const GdkPangoAttrEmbossed*) attr;
|
|
|
|
return gdk_pango_attr_embossed_new (e->embossed);
|
|
}
|
|
|
|
static void
|
|
gdk_pango_attr_embossed_destroy (PangoAttribute *attr)
|
|
{
|
|
g_free (attr);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_pango_attr_embossed_compare (const PangoAttribute *attr1,
|
|
const PangoAttribute *attr2)
|
|
{
|
|
const GdkPangoAttrEmbossed *e1 = (const GdkPangoAttrEmbossed*) attr1;
|
|
const GdkPangoAttrEmbossed *e2 = (const GdkPangoAttrEmbossed*) attr2;
|
|
|
|
return e1->embossed == e2->embossed;
|
|
}
|
|
|
|
/**
|
|
* gdk_pango_attr_embossed_new:
|
|
* @embossed: a bitmap to be set as embossed
|
|
*
|
|
* Creates a new attribute containing a embossed bitmap to be used when
|
|
* rendering the text.
|
|
*
|
|
* Return value: new #PangoAttribute
|
|
**/
|
|
|
|
PangoAttribute *
|
|
gdk_pango_attr_embossed_new (gboolean embossed)
|
|
{
|
|
GdkPangoAttrEmbossed *result;
|
|
|
|
static PangoAttrClass klass = {
|
|
0,
|
|
gdk_pango_attr_embossed_copy,
|
|
gdk_pango_attr_embossed_destroy,
|
|
gdk_pango_attr_embossed_compare
|
|
};
|
|
|
|
if (!klass.type)
|
|
klass.type = gdk_pango_attr_embossed_type =
|
|
pango_attr_type_register ("GdkPangoAttrEmbossed");
|
|
|
|
result = g_new (GdkPangoAttrEmbossed, 1);
|
|
result->attr.klass = &klass;
|
|
result->embossed = embossed;
|
|
|
|
return (PangoAttribute *)result;
|
|
}
|
|
|
|
/* Get a clip region to draw only part of a layout. index_ranges
|
|
* contains alternating range starts/stops. The region is the
|
|
* region which contains the given ranges, i.e. if you draw with the
|
|
* region as clip, only the given ranges are drawn.
|
|
*/
|
|
|
|
/**
|
|
* gdk_pango_layout_line_get_clip_region:
|
|
* @line: a #PangoLayoutLine
|
|
* @x_origin: X pixel where you intend to draw the layout line with this clip
|
|
* @y_origin: baseline pixel where you intend to draw the layout line with this clip
|
|
* @index_ranges: array of byte indexes into the layout, where even members of array are start indexes and odd elements are end indexes
|
|
* @n_ranges: number of ranges in @index_ranges, i.e. half the size of @index_ranges
|
|
*
|
|
* Obtains a clip region which contains the areas where the given
|
|
* ranges of text would be drawn. @x_origin and @y_origin are the same
|
|
* position you would pass to gdk_draw_layout_line(). @index_ranges
|
|
* should contain ranges of bytes in the layout's text. The clip
|
|
* region will include space to the left or right of the line (to the
|
|
* layout bounding box) if you have indexes above or below the indexes
|
|
* contained inside the line. This is to draw the selection all the way
|
|
* to the side of the layout. However, the clip region is in line coordinates,
|
|
* not layout coordinates.
|
|
*
|
|
* Return value: a clip region containing the given ranges
|
|
**/
|
|
GdkRegion*
|
|
gdk_pango_layout_line_get_clip_region (PangoLayoutLine *line,
|
|
gint x_origin,
|
|
gint y_origin,
|
|
gint *index_ranges,
|
|
gint n_ranges)
|
|
{
|
|
GdkRegion *clip_region;
|
|
gint i;
|
|
PangoRectangle logical_rect;
|
|
PangoLayoutIter *iter;
|
|
gint baseline;
|
|
|
|
g_return_val_if_fail (line != NULL, NULL);
|
|
g_return_val_if_fail (index_ranges != NULL, NULL);
|
|
|
|
clip_region = gdk_region_new ();
|
|
|
|
iter = pango_layout_get_iter (line->layout);
|
|
while (pango_layout_iter_get_line (iter) != line)
|
|
pango_layout_iter_next_line (iter);
|
|
|
|
pango_layout_iter_get_line_extents (iter, NULL, &logical_rect);
|
|
baseline = pango_layout_iter_get_baseline (iter);
|
|
|
|
i = 0;
|
|
while (i < n_ranges)
|
|
{
|
|
gint *pixel_ranges = NULL;
|
|
gint n_pixel_ranges = 0;
|
|
gint j;
|
|
|
|
/* Note that get_x_ranges returns layout coordinates
|
|
*/
|
|
pango_layout_line_get_x_ranges (line,
|
|
index_ranges[i*2],
|
|
index_ranges[i*2+1],
|
|
&pixel_ranges, &n_pixel_ranges);
|
|
|
|
for (j=0; j < n_pixel_ranges; j++)
|
|
{
|
|
GdkRectangle rect;
|
|
|
|
rect.x = x_origin + pixel_ranges[2*j] / PANGO_SCALE - logical_rect.x / PANGO_SCALE;
|
|
rect.y = y_origin - (baseline / PANGO_SCALE - logical_rect.y / PANGO_SCALE);
|
|
rect.width = (pixel_ranges[2*j + 1] - pixel_ranges[2*j]) / PANGO_SCALE;
|
|
rect.height = logical_rect.height / PANGO_SCALE;
|
|
|
|
gdk_region_union_with_rect (clip_region, &rect);
|
|
}
|
|
|
|
g_free (pixel_ranges);
|
|
++i;
|
|
}
|
|
|
|
return clip_region;
|
|
}
|
|
|
|
/**
|
|
* gdk_pango_layout_get_clip_region:
|
|
* @layout: a #PangoLayout
|
|
* @x_origin: X pixel where you intend to draw the layout with this clip
|
|
* @y_origin: Y pixel where you intend to draw the layout with this clip
|
|
* @index_ranges: array of byte indexes into the layout, where even members of array are start indexes and odd elements are end indexes
|
|
* @n_ranges: number of ranges in @index_ranges, i.e. half the size of @index_ranges
|
|
*
|
|
* Obtains a clip region which contains the areas where the given ranges
|
|
* of text would be drawn. @x_origin and @y_origin are the same position
|
|
* you would pass to gdk_draw_layout_line(). @index_ranges should contain
|
|
* ranges of bytes in the layout's text.
|
|
*
|
|
* Return value: a clip region containing the given ranges
|
|
**/
|
|
GdkRegion*
|
|
gdk_pango_layout_get_clip_region (PangoLayout *layout,
|
|
gint x_origin,
|
|
gint y_origin,
|
|
gint *index_ranges,
|
|
gint n_ranges)
|
|
{
|
|
PangoLayoutIter *iter;
|
|
GdkRegion *clip_region;
|
|
|
|
g_return_val_if_fail (PANGO_IS_LAYOUT (layout), NULL);
|
|
g_return_val_if_fail (index_ranges != NULL, NULL);
|
|
|
|
clip_region = gdk_region_new ();
|
|
|
|
iter = pango_layout_get_iter (layout);
|
|
|
|
do
|
|
{
|
|
PangoRectangle logical_rect;
|
|
PangoLayoutLine *line;
|
|
GdkRegion *line_region;
|
|
gint baseline;
|
|
|
|
line = pango_layout_iter_get_line (iter);
|
|
|
|
pango_layout_iter_get_line_extents (iter, NULL, &logical_rect);
|
|
baseline = pango_layout_iter_get_baseline (iter);
|
|
|
|
line_region = gdk_pango_layout_line_get_clip_region (line,
|
|
x_origin + logical_rect.x / PANGO_SCALE,
|
|
y_origin + baseline / PANGO_SCALE,
|
|
index_ranges,
|
|
n_ranges);
|
|
|
|
gdk_region_union (clip_region, line_region);
|
|
gdk_region_destroy (line_region);
|
|
}
|
|
while (pango_layout_iter_next_line (iter));
|
|
|
|
pango_layout_iter_free (iter);
|
|
|
|
return clip_region;
|
|
}
|