mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-13 14:00:09 +00:00
cebfe95660
Wed Nov 15 21:56:28 2000 Owen Taylor <otaylor@redhat.com> * src/pixbuf-*: A few updates for GTypePlugin.
372 lines
9.6 KiB
C
372 lines
9.6 KiB
C
/* GTK+ Pixbuf Engine
|
|
* Copyright (C) 1998-2000 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.
|
|
*
|
|
* Written by Owen Taylor <otaylor@redhat.com>, based on code by
|
|
* Carsten Haitzler <raster@rasterman.com>
|
|
*/
|
|
|
|
#include "pixbuf.h"
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
|
|
GCache *pixbuf_cache = NULL;
|
|
|
|
/* Scale the rectangle (src_x, src_y, src_width, src_height)
|
|
* onto the rectangle (dest_x, dest_y, dest_width, dest_height)
|
|
* of the destination, clip by clip_rect and render
|
|
*/
|
|
static void
|
|
pixbuf_render (GdkPixbuf *src,
|
|
GdkWindow *window,
|
|
GdkBitmap *mask,
|
|
GdkRectangle *clip_rect,
|
|
gint src_x,
|
|
gint src_y,
|
|
gint src_width,
|
|
gint src_height,
|
|
gint dest_x,
|
|
gint dest_y,
|
|
gint dest_width,
|
|
gint dest_height)
|
|
{
|
|
GdkPixbuf *tmp_pixbuf;
|
|
GdkRectangle rect;
|
|
int x_offset, y_offset;
|
|
gboolean has_alpha = gdk_pixbuf_get_has_alpha (src);
|
|
gint src_rowstride = gdk_pixbuf_get_rowstride (src);
|
|
gint src_n_channels = gdk_pixbuf_get_n_channels (src);
|
|
|
|
if (dest_width <= 0 || dest_height <= 0)
|
|
return;
|
|
|
|
rect.x = dest_x;
|
|
rect.y = dest_y;
|
|
rect.width = dest_width;
|
|
rect.height = dest_height;
|
|
|
|
/* FIXME: Because we use the mask to shape windows, we don't use
|
|
* clip_rect to clip what we draw to the mask, only to clip
|
|
* what we actually draw. But this leads to the horrible ineffiency
|
|
* of scale the whole image to get a little bit of it.
|
|
*/
|
|
if (!mask && clip_rect)
|
|
{
|
|
/* The temporary is necessary only for GDK+-1.2, and not
|
|
* for later versions of GDK, where you can have the
|
|
* same source and dest for gdk_rectangle_intersect().
|
|
*/
|
|
GdkRectangle tmp_rect = rect;
|
|
|
|
if (!gdk_rectangle_intersect (clip_rect, &tmp_rect, &rect))
|
|
return;
|
|
}
|
|
|
|
if (dest_width != src_width ||
|
|
dest_height != src_height)
|
|
{
|
|
double x_scale = (double)dest_width / src_width;
|
|
double y_scale = (double)dest_height / src_height;
|
|
guchar *pixels;
|
|
GdkPixbuf *partial_src;
|
|
|
|
pixels = (gdk_pixbuf_get_pixels (src)
|
|
+ src_y * src_rowstride
|
|
+ src_x * src_n_channels);
|
|
|
|
partial_src = gdk_pixbuf_new_from_data (pixels, GDK_COLORSPACE_RGB,
|
|
has_alpha,
|
|
8, src_width, src_height,
|
|
src_rowstride,
|
|
NULL, NULL);
|
|
|
|
tmp_pixbuf = gdk_pixbuf_new (GDK_COLORSPACE_RGB,
|
|
has_alpha, 8,
|
|
rect.width, rect.height);
|
|
|
|
gdk_pixbuf_scale (partial_src, tmp_pixbuf,
|
|
0, 0, rect.width, rect.height,
|
|
dest_x - rect.x, dest_y - rect.y,
|
|
x_scale, y_scale,
|
|
GDK_INTERP_BILINEAR);
|
|
|
|
gdk_pixbuf_unref (partial_src);
|
|
|
|
x_offset = 0;
|
|
y_offset = 0;
|
|
}
|
|
else
|
|
{
|
|
tmp_pixbuf = src;
|
|
gdk_pixbuf_ref (tmp_pixbuf);
|
|
|
|
x_offset = src_x + rect.x - dest_x;
|
|
y_offset = src_y + rect.y - dest_y;
|
|
}
|
|
|
|
if (mask)
|
|
{
|
|
GdkGC *tmp_gc;
|
|
|
|
gdk_pixbuf_render_threshold_alpha (tmp_pixbuf, mask,
|
|
x_offset, y_offset,
|
|
rect.x, rect.y,
|
|
rect.width, rect.height,
|
|
128);
|
|
|
|
tmp_gc = gdk_gc_new (window);
|
|
if (clip_rect)
|
|
gdk_gc_set_clip_rectangle (tmp_gc, clip_rect);
|
|
|
|
gdk_pixbuf_render_to_drawable (tmp_pixbuf, window, tmp_gc,
|
|
x_offset, y_offset,
|
|
rect.x, rect.y,
|
|
rect.width, rect.height,
|
|
GDK_RGB_DITHER_NORMAL,
|
|
0, 0);
|
|
gdk_gc_unref (tmp_gc);
|
|
}
|
|
else
|
|
gdk_pixbuf_render_to_drawable_alpha (tmp_pixbuf, window,
|
|
x_offset, y_offset,
|
|
rect.x, rect.y,
|
|
rect.width, rect.height,
|
|
GDK_PIXBUF_ALPHA_BILEVEL, 128,
|
|
GDK_RGB_DITHER_NORMAL,
|
|
0, 0);
|
|
gdk_pixbuf_unref (tmp_pixbuf);
|
|
}
|
|
|
|
ThemePixbuf *
|
|
theme_pixbuf_new (void)
|
|
{
|
|
ThemePixbuf *result = g_new (ThemePixbuf, 1);
|
|
result->filename = NULL;
|
|
result->pixbuf = NULL;
|
|
|
|
result->stretch = TRUE;
|
|
result->border_left = 0;
|
|
result->border_right = 0;
|
|
result->border_bottom = 0;
|
|
result->border_top = 0;
|
|
|
|
return result;
|
|
}
|
|
|
|
void
|
|
theme_pixbuf_destroy (ThemePixbuf *theme_pb)
|
|
{
|
|
if (theme_pb->pixbuf)
|
|
g_cache_remove (pixbuf_cache, theme_pb->pixbuf);
|
|
}
|
|
|
|
void
|
|
theme_pixbuf_set_filename (ThemePixbuf *theme_pb,
|
|
const char *filename)
|
|
{
|
|
if (theme_pb->pixbuf)
|
|
{
|
|
g_cache_remove (pixbuf_cache, theme_pb->pixbuf);
|
|
theme_pb->pixbuf = NULL;
|
|
}
|
|
|
|
if (theme_pb->filename)
|
|
g_free (theme_pb->filename);
|
|
|
|
theme_pb->filename = g_strdup (filename);
|
|
}
|
|
|
|
void
|
|
theme_pixbuf_set_border (ThemePixbuf *theme_pb,
|
|
gint left,
|
|
gint right,
|
|
gint top,
|
|
gint bottom)
|
|
{
|
|
theme_pb->border_left = left;
|
|
theme_pb->border_right = right;
|
|
theme_pb->border_top = top;
|
|
theme_pb->border_bottom = bottom;
|
|
}
|
|
|
|
void
|
|
theme_pixbuf_set_stretch (ThemePixbuf *theme_pb,
|
|
gboolean stretch)
|
|
{
|
|
theme_pb->stretch = stretch;
|
|
}
|
|
|
|
GdkPixbuf *
|
|
pixbuf_cache_value_new (gchar *filename)
|
|
{
|
|
GError *err = NULL;
|
|
|
|
GdkPixbuf *result = gdk_pixbuf_new_from_file (filename, &err);
|
|
if (!result)
|
|
{
|
|
g_warning ("Pixbuf theme: Cannot load pixmap file %s: %s\n",
|
|
filename, err->message);
|
|
g_error_free (err);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
GdkPixbuf *
|
|
theme_pixbuf_get_pixbuf (ThemePixbuf *theme_pb)
|
|
{
|
|
if (!theme_pb->pixbuf)
|
|
{
|
|
if (!pixbuf_cache)
|
|
pixbuf_cache = g_cache_new ((GCacheNewFunc)pixbuf_cache_value_new,
|
|
(GCacheDestroyFunc)gdk_pixbuf_unref,
|
|
(GCacheDupFunc)g_strdup,
|
|
(GCacheDestroyFunc)g_free,
|
|
g_str_hash, g_direct_hash, g_str_equal);
|
|
|
|
theme_pb->pixbuf = g_cache_insert (pixbuf_cache, theme_pb->filename);
|
|
}
|
|
|
|
return theme_pb->pixbuf;
|
|
}
|
|
|
|
void
|
|
theme_pixbuf_render (ThemePixbuf *theme_pb,
|
|
GdkWindow *window,
|
|
GdkBitmap *mask,
|
|
GdkRectangle *clip_rect,
|
|
guint component_mask,
|
|
gboolean center,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GdkPixbuf *pixbuf = theme_pixbuf_get_pixbuf (theme_pb);
|
|
gint src_x[4], src_y[4], dest_x[4], dest_y[4];
|
|
gint pixbuf_width = gdk_pixbuf_get_width (pixbuf);
|
|
gint pixbuf_height = gdk_pixbuf_get_height (pixbuf);
|
|
|
|
if (!pixbuf)
|
|
return;
|
|
|
|
if (theme_pb->stretch)
|
|
{
|
|
src_x[0] = 0;
|
|
src_x[1] = theme_pb->border_left;
|
|
src_x[2] = pixbuf_width - theme_pb->border_right;
|
|
src_x[3] = pixbuf_width;
|
|
|
|
src_y[0] = 0;
|
|
src_y[1] = theme_pb->border_top;
|
|
src_y[2] = pixbuf_height - theme_pb->border_bottom;
|
|
src_y[3] = pixbuf_height;
|
|
|
|
dest_x[0] = x;
|
|
dest_x[1] = x + theme_pb->border_left;
|
|
dest_x[2] = x + width - theme_pb->border_right;
|
|
dest_x[3] = x + width;
|
|
|
|
dest_y[0] = y;
|
|
dest_y[1] = y + theme_pb->border_top;
|
|
dest_y[2] = y + height - theme_pb->border_bottom;
|
|
dest_y[3] = y + height;
|
|
|
|
if (component_mask & COMPONENT_ALL)
|
|
component_mask = (COMPONENT_ALL - 1) & ~component_mask;
|
|
|
|
#define RENDER_COMPONENT(X1,X2,Y1,Y2) \
|
|
pixbuf_render (pixbuf, window, mask, clip_rect, \
|
|
src_x[X1], src_y[Y1], \
|
|
src_x[X2] - src_x[X1], src_y[Y2] - src_y[Y1], \
|
|
dest_x[X1], dest_y[Y1], \
|
|
dest_x[X2] - dest_x[X1], dest_y[Y2] - dest_y[Y1]);
|
|
|
|
if (component_mask & COMPONENT_NORTH_WEST)
|
|
RENDER_COMPONENT (0, 1, 0, 1);
|
|
|
|
if (component_mask & COMPONENT_NORTH)
|
|
RENDER_COMPONENT (1, 2, 0, 1);
|
|
|
|
if (component_mask & COMPONENT_NORTH_EAST)
|
|
RENDER_COMPONENT (2, 3, 0, 1);
|
|
|
|
if (component_mask & COMPONENT_WEST)
|
|
RENDER_COMPONENT (0, 1, 1, 2);
|
|
|
|
if (component_mask & COMPONENT_CENTER)
|
|
RENDER_COMPONENT (1, 2, 1, 2);
|
|
|
|
if (component_mask & COMPONENT_EAST)
|
|
RENDER_COMPONENT (2, 3, 1, 2);
|
|
|
|
if (component_mask & COMPONENT_SOUTH_WEST)
|
|
RENDER_COMPONENT (0, 1, 2, 3);
|
|
|
|
if (component_mask & COMPONENT_SOUTH)
|
|
RENDER_COMPONENT (1, 2, 2, 3);
|
|
|
|
if (component_mask & COMPONENT_SOUTH_EAST)
|
|
RENDER_COMPONENT (2, 3, 2, 3);
|
|
}
|
|
else
|
|
{
|
|
if (center)
|
|
{
|
|
x += (width - pixbuf_width) / 2;
|
|
y += (height - pixbuf_height) / 2;
|
|
|
|
pixbuf_render (pixbuf, window, NULL, clip_rect,
|
|
0, 0,
|
|
pixbuf_width, pixbuf_height,
|
|
x, y,
|
|
pixbuf_width, pixbuf_height);
|
|
}
|
|
else
|
|
{
|
|
GdkPixmap *tmp_pixmap;
|
|
GdkGC *tmp_gc;
|
|
GdkGCValues gc_values;
|
|
|
|
tmp_pixmap = gdk_pixmap_new (window,
|
|
pixbuf_width,
|
|
pixbuf_height,
|
|
-1);
|
|
tmp_gc = gdk_gc_new (tmp_pixmap);
|
|
gdk_pixbuf_render_to_drawable (pixbuf, tmp_pixmap, tmp_gc,
|
|
0, 0,
|
|
0, 0,
|
|
pixbuf_width, pixbuf_height,
|
|
GDK_RGB_DITHER_NORMAL,
|
|
0, 0);
|
|
gdk_gc_unref (tmp_gc);
|
|
|
|
gc_values.fill = GDK_TILED;
|
|
gc_values.tile = tmp_pixmap;
|
|
tmp_gc = gdk_gc_new_with_values (window,
|
|
&gc_values, GDK_GC_FILL | GDK_GC_TILE);
|
|
if (clip_rect)
|
|
gdk_draw_rectangle (window, tmp_gc, TRUE,
|
|
clip_rect->x, clip_rect->y, clip_rect->width, clip_rect->height);
|
|
else
|
|
gdk_draw_rectangle (window, tmp_gc, TRUE, x, y, width, height);
|
|
|
|
gdk_gc_unref (tmp_gc);
|
|
gdk_pixmap_unref (tmp_pixmap);
|
|
}
|
|
}
|
|
}
|