/* GDK - The GIMP Drawing Kit * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald * * 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. */ /* * Modified by the GTK+ Team and others 1997-1999. See the AUTHORS * file for a list of people on the GTK+ Team. See the ChangeLog * files for a list of changes. These files are distributed with * GTK+ at ftp://ftp.gtk.org/pub/gtk/. */ #include #include #include "gdkx.h" #include "gdkdrawable.h" #include "gdkprivate.h" #include "gdkwindow.h" /* Manipulation of drawables */ void gdk_drawable_set_data (GdkDrawable *drawable, const gchar *key, gpointer data, GDestroyNotify destroy_func) { g_dataset_set_data_full (drawable, key, data, destroy_func); } void gdk_drawable_get_data (GdkDrawable *drawable, const gchar *key) { g_dataset_get_data (drawable, key); } GdkDrawableType gdk_drawable_get_type (GdkDrawable *drawable) { g_return_val_if_fail (drawable != NULL, (GdkDrawableType) -1); return GDK_DRAWABLE_TYPE (drawable); } void gdk_drawable_get_size (GdkDrawable *drawable, gint *width, gint *height) { GdkDrawablePrivate *drawable_private; g_return_if_fail (drawable != NULL); drawable_private = (GdkDrawablePrivate*) drawable; if (width) *width = drawable_private->width; if (height) *height = drawable_private->height; } void gdk_drawable_set_colormap (GdkDrawable *drawable, GdkColormap *colormap) { GdkDrawablePrivate *drawable_private; GdkColormapPrivate *colormap_private; g_return_if_fail (drawable != NULL); g_return_if_fail (colormap != NULL); drawable_private = (GdkDrawablePrivate*) drawable; colormap_private = (GdkColormapPrivate*) colormap; if (!GDK_DRAWABLE_DESTROYED (drawable)) { if (GDK_IS_WINDOW (drawable)) { g_return_if_fail (colormap_private->visual != ((GdkColormapPrivate *)(drawable_private->colormap))->visual); XSetWindowColormap (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), colormap_private->xcolormap); } if (drawable_private->colormap) gdk_colormap_unref (drawable_private->colormap); drawable_private->colormap = colormap; gdk_colormap_ref (drawable_private->colormap); if (GDK_IS_WINDOW (drawable) && drawable_private->window_type != GDK_WINDOW_TOPLEVEL) gdk_window_add_colormap_windows (drawable); } } GdkColormap* gdk_drawable_get_colormap (GdkDrawable *drawable) { GdkDrawablePrivate *drawable_private; XWindowAttributes window_attributes; g_return_val_if_fail (drawable != NULL, NULL); drawable_private = (GdkDrawablePrivate*) drawable; if (!GDK_DRAWABLE_DESTROYED (drawable)) { if (drawable_private->colormap == NULL && GDK_IS_WINDOW (drawable)) { XGetWindowAttributes (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), &window_attributes); drawable_private->colormap = gdk_colormap_lookup (window_attributes.colormap); } return drawable_private->colormap; } return NULL; } GdkVisual* gdk_drawable_get_visual (GdkDrawable *drawable) { GdkColormap *colormap; g_return_val_if_fail (drawable != NULL, NULL); colormap = gdk_drawable_get_colormap (drawable); return colormap ? gdk_colormap_get_visual (colormap) : NULL; } /* Drawing */ void gdk_draw_point (GdkDrawable *drawable, GdkGC *gc, gint x, gint y) { GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (gc != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; XDrawPoint (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y); } void gdk_draw_line (GdkDrawable *drawable, GdkGC *gc, gint x1, gint y1, gint x2, gint y2) { GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (gc != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; XDrawLine (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x1, y1, x2, y2); } void gdk_draw_rectangle (GdkDrawable *drawable, GdkGC *gc, gint filled, gint x, gint y, gint width, gint height) { GdkDrawablePrivate *drawable_private; GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (gc != NULL); drawable_private = (GdkDrawablePrivate*) drawable; if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; if (width == -1) width = drawable_private->width; if (height == -1) height = drawable_private->height; if (filled) XFillRectangle (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, width, height); else XDrawRectangle (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, width, height); } void gdk_draw_arc (GdkDrawable *drawable, GdkGC *gc, gint filled, gint x, gint y, gint width, gint height, gint angle1, gint angle2) { GdkDrawablePrivate *drawable_private; GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (gc != NULL); drawable_private = (GdkDrawablePrivate*) drawable; if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; if (width == -1) width = drawable_private->width; if (height == -1) height = drawable_private->height; if (filled) XFillArc (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, width, height, angle1, angle2); else XDrawArc (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, width, height, angle1, angle2); } void gdk_draw_polygon (GdkDrawable *drawable, GdkGC *gc, gint filled, GdkPoint *points, gint npoints) { GdkGCPrivate *gc_private; GdkPoint *local_points = points; gint local_npoints = npoints; gint local_alloc = 0; g_return_if_fail (drawable != NULL); g_return_if_fail (gc != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; if (filled) { XFillPolygon (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, (XPoint*) points, npoints, Complex, CoordModeOrigin); } else { if ((points[0].x != points[npoints-1].x) || (points[0].y != points[npoints-1].y)) { local_alloc = 1; ++local_npoints; local_points = (GdkPoint*) g_malloc (local_npoints * sizeof(GdkPoint)); memcpy (local_points, points, npoints * sizeof(GdkPoint)); local_points[npoints].x = points[0].x; local_points[npoints].y = points[0].y; } XDrawLines (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, (XPoint*) local_points, local_npoints, CoordModeOrigin); if (local_alloc) g_free (local_points); } } /* gdk_draw_string * * Modified by Li-Da Lho to draw 16 bits and Multibyte strings * * Interface changed: add "GdkFont *font" to specify font or fontset explicitely */ void gdk_draw_string (GdkDrawable *drawable, GdkFont *font, GdkGC *gc, gint x, gint y, const gchar *string) { GdkFontPrivate *font_private; GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (font != NULL); g_return_if_fail (gc != NULL); g_return_if_fail (string != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; font_private = (GdkFontPrivate*) font; if (font->type == GDK_FONT_FONT) { XFontStruct *xfont = (XFontStruct *) font_private->xfont; XSetFont(GDK_DRAWABLE_XDISPLAY (drawable), gc_private->xgc, xfont->fid); if ((xfont->min_byte1 == 0) && (xfont->max_byte1 == 0)) { XDrawString (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, string, strlen (string)); } else { XDrawString16 (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, (XChar2b *) string, strlen (string) / 2); } } else if (font->type == GDK_FONT_FONTSET) { XFontSet fontset = (XFontSet) font_private->xfont; XmbDrawString (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), fontset, gc_private->xgc, x, y, string, strlen (string)); } else g_error("undefined font type\n"); } /* gdk_draw_text * * Modified by Li-Da Lho to draw 16 bits and Multibyte strings * * Interface changed: add "GdkFont *font" to specify font or fontset explicitely */ void gdk_draw_text (GdkDrawable *drawable, GdkFont *font, GdkGC *gc, gint x, gint y, const gchar *text, gint text_length) { GdkFontPrivate *font_private; GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (font != NULL); g_return_if_fail (gc != NULL); g_return_if_fail (text != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; font_private = (GdkFontPrivate*) font; if (font->type == GDK_FONT_FONT) { XFontStruct *xfont = (XFontStruct *) font_private->xfont; XSetFont(GDK_DRAWABLE_XDISPLAY (drawable), gc_private->xgc, xfont->fid); if ((xfont->min_byte1 == 0) && (xfont->max_byte1 == 0)) { XDrawString (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, text, text_length); } else { XDrawString16 (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, x, y, (XChar2b *) text, text_length / 2); } } else if (font->type == GDK_FONT_FONTSET) { XFontSet fontset = (XFontSet) font_private->xfont; XmbDrawString (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), fontset, gc_private->xgc, x, y, text, text_length); } else g_error("undefined font type\n"); } void gdk_draw_text_wc (GdkDrawable *drawable, GdkFont *font, GdkGC *gc, gint x, gint y, const GdkWChar *text, gint text_length) { GdkFontPrivate *font_private; GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (font != NULL); g_return_if_fail (gc != NULL); g_return_if_fail (text != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; font_private = (GdkFontPrivate*) font; if (font->type == GDK_FONT_FONT) { XFontStruct *xfont = (XFontStruct *) font_private->xfont; gchar *text_8bit; gint i; XSetFont(GDK_DRAWABLE_XDISPLAY (drawable), gc_private->xgc, xfont->fid); text_8bit = g_new (gchar, text_length); for (i=0; ixgc, x, y, text_8bit, text_length); g_free (text_8bit); } else if (font->type == GDK_FONT_FONTSET) { if (sizeof(GdkWChar) == sizeof(wchar_t)) { XwcDrawString (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), (XFontSet) font_private->xfont, gc_private->xgc, x, y, (wchar_t *)text, text_length); } else { wchar_t *text_wchar; gint i; text_wchar = g_new (wchar_t, text_length); for (i=0; ixfont, gc_private->xgc, x, y, text_wchar, text_length); g_free (text_wchar); } } else g_error("undefined font type\n"); } void gdk_draw_pixmap (GdkDrawable *drawable, GdkGC *gc, GdkPixmap *src, gint xsrc, gint ysrc, gint xdest, gint ydest, gint width, gint height) { GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail (src != NULL); g_return_if_fail (gc != NULL); if (GDK_DRAWABLE_DESTROYED (drawable) || GDK_DRAWABLE_DESTROYED (src)) return; gc_private = (GdkGCPrivate*) gc; if (width == -1) width = ((GdkDrawablePrivate *)src)->width; if (height == -1) height = ((GdkDrawablePrivate *)src)->height; XCopyArea (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (src), GDK_DRAWABLE_XID (drawable), gc_private->xgc, xsrc, ysrc, width, height, xdest, ydest); } void gdk_draw_image (GdkDrawable *drawable, GdkGC *gc, GdkImage *image, gint xsrc, gint ysrc, gint xdest, gint ydest, gint width, gint height) { GdkImagePrivate *image_private; g_return_if_fail (drawable != NULL); g_return_if_fail (image != NULL); g_return_if_fail (gc != NULL); image_private = (GdkImagePrivate*) image; g_return_if_fail (image_private->image_put != NULL); if (width == -1) width = image->width; if (height == -1) height = image->height; (* image_private->image_put) (drawable, gc, image, xsrc, ysrc, xdest, ydest, width, height); } void gdk_draw_points (GdkDrawable *drawable, GdkGC *gc, GdkPoint *points, gint npoints) { GdkGCPrivate *gc_private; g_return_if_fail (drawable != NULL); g_return_if_fail ((points != NULL) && (npoints > 0)); g_return_if_fail (gc != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; XDrawPoints (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, (XPoint *) points, npoints, CoordModeOrigin); } void gdk_draw_segments (GdkDrawable *drawable, GdkGC *gc, GdkSegment *segs, gint nsegs) { GdkGCPrivate *gc_private; if (nsegs <= 0) return; g_return_if_fail (drawable != NULL); g_return_if_fail (segs != NULL); g_return_if_fail (gc != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; XDrawSegments (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, (XSegment *) segs, nsegs); } void gdk_draw_lines (GdkDrawable *drawable, GdkGC *gc, GdkPoint *points, gint npoints) { GdkGCPrivate *gc_private; if (npoints <= 0) return; g_return_if_fail (drawable != NULL); g_return_if_fail (points != NULL); g_return_if_fail (gc != NULL); if (GDK_DRAWABLE_DESTROYED (drawable)) return; gc_private = (GdkGCPrivate*) gc; XDrawLines (GDK_DRAWABLE_XDISPLAY (drawable), GDK_DRAWABLE_XID (drawable), gc_private->xgc, (XPoint *) points, npoints, CoordModeOrigin); }