forked from AuroraMiddleware/gtk
0b4d29fd6e
* gdk/gdktypes.h: Merge in Win32 version: Define macro GDKVAR for declaring gdk variables exported/imported from the DLL. New image type enum, GDK_IMAGE_SHARED_PIXMAP, for gdk_imlib. New drag and drop protocol enums, GDK_DRAG_PROTO_WIN32_DROPFILES and GDK_DRAG_PROTO_OLE2. * gdk/gdk.h: Merge in Win32 version: Two new functions, gdk_pixmap_create_on_shared_image and gdk_image_bitmap_new. So far declared only for the Win32 version, but could be in the X11 version as well. (Needed for a Xlib-less gdk_imlib.) gdk_color_hash should have only one parameter. Declare gdk_threads_mutex with GDKVAR. * gdk/gdkcolor.c (gdk_color_hash): As a hash function should have just one parameter. * gdk/gdkimage.c (gdk_image_get): Initialize bpp correctly. Bytes per pixel, not bits. * gdk/gdkrgb.c: Mingle includes somewhat. (gdk_rgb_select_conv): Fetch bpp (which means bits-per-pixel here) from another place on Win32. Accept also depth==32 (which we might get on Win32) with bpp==32. * gtk/{gtkclist,gtkctree,gtkdnd,gtkditable,gtkfontsel, gtkhandlebox,gtklayout,gtkmain,gtkplug,gtkpreview,gtkrc, gtkselection,gtksocket,gtkstyle,gtkwidget,gtkwindow}.c: Include gdx.h from "gdkx.h", not "gdk/gdkx.h", as gdkx.h will be in the backend-dependent directory, not in the common gdk directory. * gtk/testgtk.c: Ditto. Also, don't use ../gdk patchs to gdk headers.
561 lines
13 KiB
C
561 lines
13 KiB
C
/* GTK - The GIMP Toolkit
|
|
* 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 "config.h"
|
|
|
|
#include <math.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#ifdef HAVE_SYS_PARAM_H
|
|
#include <sys/param.h>
|
|
#endif
|
|
#include "gdkx.h"
|
|
#include "gdk/gdkrgb.h"
|
|
#include "gtkpreview.h"
|
|
#include "gtksignal.h"
|
|
|
|
|
|
#define PREVIEW_CLASS(w) GTK_PREVIEW_CLASS (GTK_OBJECT (w)->klass)
|
|
|
|
|
|
static void gtk_preview_class_init (GtkPreviewClass *klass);
|
|
static void gtk_preview_init (GtkPreview *preview);
|
|
static void gtk_preview_finalize (GtkObject *object);
|
|
static void gtk_preview_realize (GtkWidget *widget);
|
|
static void gtk_preview_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation);
|
|
static gint gtk_preview_expose (GtkWidget *widget,
|
|
GdkEventExpose *event);
|
|
static void gtk_preview_make_buffer (GtkPreview *preview);
|
|
static void gtk_fill_lookup_array (guchar *array);
|
|
|
|
static GtkWidgetClass *parent_class = NULL;
|
|
static GtkPreviewClass *preview_class = NULL;
|
|
static gint install_cmap = FALSE;
|
|
|
|
|
|
guint
|
|
gtk_preview_get_type (void)
|
|
{
|
|
static guint preview_type = 0;
|
|
|
|
if (!preview_type)
|
|
{
|
|
static const GtkTypeInfo preview_info =
|
|
{
|
|
"GtkPreview",
|
|
sizeof (GtkPreview),
|
|
sizeof (GtkPreviewClass),
|
|
(GtkClassInitFunc) gtk_preview_class_init,
|
|
(GtkObjectInitFunc) gtk_preview_init,
|
|
/* reserved_1 */ NULL,
|
|
/* reserved_2 */ NULL,
|
|
(GtkClassInitFunc) NULL,
|
|
};
|
|
|
|
preview_type = gtk_type_unique (gtk_widget_get_type (), &preview_info);
|
|
}
|
|
|
|
return preview_type;
|
|
}
|
|
|
|
static void
|
|
gtk_preview_class_init (GtkPreviewClass *klass)
|
|
{
|
|
GtkObjectClass *object_class;
|
|
GtkWidgetClass *widget_class;
|
|
|
|
object_class = (GtkObjectClass*) klass;
|
|
widget_class = (GtkWidgetClass*) klass;
|
|
|
|
parent_class = gtk_type_class (gtk_widget_get_type ());
|
|
preview_class = klass;
|
|
|
|
object_class->finalize = gtk_preview_finalize;
|
|
|
|
widget_class->realize = gtk_preview_realize;
|
|
widget_class->size_allocate = gtk_preview_size_allocate;
|
|
widget_class->expose_event = gtk_preview_expose;
|
|
|
|
klass->info.visual = NULL;
|
|
klass->info.cmap = NULL;
|
|
|
|
klass->info.lookup = NULL;
|
|
|
|
klass->info.gamma = 1.0;
|
|
|
|
gdk_rgb_init ();
|
|
klass->info.cmap = gdk_rgb_get_cmap ();
|
|
klass->info.visual = gdk_rgb_get_visual ();
|
|
}
|
|
|
|
void
|
|
gtk_preview_reset (void)
|
|
{
|
|
/* unimplemented */
|
|
}
|
|
|
|
static void
|
|
gtk_preview_init (GtkPreview *preview)
|
|
{
|
|
preview->buffer = NULL;
|
|
preview->buffer_width = 0;
|
|
preview->buffer_height = 0;
|
|
preview->expand = FALSE;
|
|
}
|
|
|
|
void
|
|
gtk_preview_uninit (void)
|
|
{
|
|
|
|
/* unimplemented */
|
|
}
|
|
|
|
GtkWidget*
|
|
gtk_preview_new (GtkPreviewType type)
|
|
{
|
|
GtkPreview *preview;
|
|
|
|
preview = gtk_type_new (gtk_preview_get_type ());
|
|
preview->type = type;
|
|
|
|
if (type == GTK_PREVIEW_COLOR)
|
|
preview->bpp = 3;
|
|
else
|
|
preview->bpp = 1;
|
|
|
|
preview->dither = GDK_RGB_DITHER_NORMAL;
|
|
|
|
return GTK_WIDGET (preview);
|
|
}
|
|
|
|
void
|
|
gtk_preview_size (GtkPreview *preview,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
g_return_if_fail (preview != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (preview));
|
|
|
|
if ((width != GTK_WIDGET (preview)->requisition.width) ||
|
|
(height != GTK_WIDGET (preview)->requisition.height))
|
|
{
|
|
GTK_WIDGET (preview)->requisition.width = width;
|
|
GTK_WIDGET (preview)->requisition.height = height;
|
|
|
|
if (preview->buffer)
|
|
g_free (preview->buffer);
|
|
preview->buffer = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
gtk_preview_put (GtkPreview *preview,
|
|
GdkWindow *window,
|
|
GdkGC *gc,
|
|
gint srcx,
|
|
gint srcy,
|
|
gint destx,
|
|
gint desty,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GtkWidget *widget;
|
|
GdkRectangle r1, r2, r3;
|
|
guchar *src;
|
|
guint bpp;
|
|
guint rowstride;
|
|
|
|
g_return_if_fail (preview != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (preview));
|
|
g_return_if_fail (window != NULL);
|
|
|
|
if (!preview->buffer)
|
|
return;
|
|
|
|
widget = GTK_WIDGET (preview);
|
|
|
|
r1.x = 0;
|
|
r1.y = 0;
|
|
r1.width = preview->buffer_width;
|
|
r1.height = preview->buffer_height;
|
|
|
|
r2.x = srcx;
|
|
r2.y = srcy;
|
|
r2.width = width;
|
|
r2.height = height;
|
|
|
|
if (!gdk_rectangle_intersect (&r1, &r2, &r3))
|
|
return;
|
|
|
|
bpp = preview->bpp;
|
|
rowstride = preview->rowstride;
|
|
|
|
src = preview->buffer + r3.y * rowstride + r3.x * bpp;
|
|
|
|
if (preview->type == GTK_PREVIEW_COLOR)
|
|
gdk_draw_rgb_image (window,
|
|
gc,
|
|
destx + (r3.x - srcx),
|
|
desty + (r3.y - srcy),
|
|
r3.width,
|
|
r3.height,
|
|
preview->dither,
|
|
src,
|
|
rowstride);
|
|
else
|
|
gdk_draw_gray_image (window,
|
|
gc,
|
|
destx + (r3.x - srcx),
|
|
desty + (r3.y - srcy),
|
|
r3.width,
|
|
r3.height,
|
|
preview->dither,
|
|
src,
|
|
rowstride);
|
|
|
|
}
|
|
|
|
void
|
|
gtk_preview_draw_row (GtkPreview *preview,
|
|
guchar *data,
|
|
gint x,
|
|
gint y,
|
|
gint w)
|
|
{
|
|
guint bpp;
|
|
guint rowstride;
|
|
|
|
g_return_if_fail (preview != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (preview));
|
|
g_return_if_fail (data != NULL);
|
|
g_return_if_fail (preview_class->info.visual != NULL);
|
|
|
|
bpp = (preview->type == GTK_PREVIEW_COLOR ? 3 : 1);
|
|
rowstride = (preview->buffer_width * bpp + 3) & -4;
|
|
|
|
if ((w <= 0) || (y < 0))
|
|
return;
|
|
|
|
g_return_if_fail (data != NULL);
|
|
|
|
gtk_preview_make_buffer (preview);
|
|
|
|
if (x + w > preview->buffer_width)
|
|
return;
|
|
|
|
if (y + 1 > preview->buffer_height)
|
|
return;
|
|
|
|
if (preview_class->info.gamma == 1.0)
|
|
memcpy (preview->buffer + y * rowstride + x * bpp, data, w * bpp);
|
|
else
|
|
{
|
|
guint i, size;
|
|
guchar *src, *dst;
|
|
guchar *lookup;
|
|
|
|
if (preview_class->info.lookup != NULL)
|
|
lookup = preview_class->info.lookup;
|
|
else
|
|
{
|
|
preview_class->info.lookup = g_new (guchar, 256);
|
|
gtk_fill_lookup_array (preview_class->info.lookup);
|
|
lookup = preview_class->info.lookup;
|
|
}
|
|
size = w * bpp;
|
|
src = data;
|
|
dst = preview->buffer + y * rowstride + x * bpp;
|
|
for (i = 0; i < size; i++)
|
|
*dst++ = lookup[*src++];
|
|
}
|
|
}
|
|
|
|
void
|
|
gtk_preview_set_expand (GtkPreview *preview,
|
|
gint expand)
|
|
{
|
|
g_return_if_fail (preview != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (preview));
|
|
|
|
preview->expand = (expand != FALSE);
|
|
}
|
|
|
|
void
|
|
gtk_preview_set_gamma (double _gamma)
|
|
{
|
|
if (!preview_class)
|
|
preview_class = gtk_type_class (gtk_preview_get_type ());
|
|
|
|
if (preview_class->info.gamma != _gamma)
|
|
{
|
|
preview_class->info.gamma = _gamma;
|
|
if (preview_class->info.lookup != NULL)
|
|
{
|
|
g_free (preview_class->info.lookup);
|
|
preview_class->info.lookup = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
gtk_preview_set_color_cube (guint nred_shades,
|
|
guint ngreen_shades,
|
|
guint nblue_shades,
|
|
guint ngray_shades)
|
|
{
|
|
/* unimplemented */
|
|
}
|
|
|
|
void
|
|
gtk_preview_set_install_cmap (gint _install_cmap)
|
|
{
|
|
/* effectively unimplemented */
|
|
install_cmap = _install_cmap;
|
|
}
|
|
|
|
void
|
|
gtk_preview_set_reserved (gint nreserved)
|
|
{
|
|
|
|
/* unimplemented */
|
|
}
|
|
|
|
void
|
|
gtk_preview_set_dither (GtkPreview *preview,
|
|
GdkRgbDither dither)
|
|
{
|
|
preview->dither = dither;
|
|
}
|
|
|
|
GdkVisual*
|
|
gtk_preview_get_visual (void)
|
|
{
|
|
if (!preview_class)
|
|
preview_class = gtk_type_class (gtk_preview_get_type ());
|
|
|
|
return preview_class->info.visual;
|
|
}
|
|
|
|
GdkColormap*
|
|
gtk_preview_get_cmap (void)
|
|
{
|
|
if (!preview_class)
|
|
preview_class = gtk_type_class (gtk_preview_get_type ());
|
|
|
|
return preview_class->info.cmap;
|
|
}
|
|
|
|
GtkPreviewInfo*
|
|
gtk_preview_get_info (void)
|
|
{
|
|
if (!preview_class)
|
|
preview_class = gtk_type_class (gtk_preview_get_type ());
|
|
|
|
return &preview_class->info;
|
|
}
|
|
|
|
|
|
static void
|
|
gtk_preview_finalize (GtkObject *object)
|
|
{
|
|
GtkPreview *preview;
|
|
|
|
g_return_if_fail (object != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (object));
|
|
|
|
preview = GTK_PREVIEW (object);
|
|
if (preview->buffer)
|
|
g_free (preview->buffer);
|
|
preview->type = (GtkPreviewType) -1;
|
|
|
|
(* GTK_OBJECT_CLASS (parent_class)->finalize) (object);
|
|
}
|
|
|
|
static void
|
|
gtk_preview_realize (GtkWidget *widget)
|
|
{
|
|
GtkPreview *preview;
|
|
GdkWindowAttr attributes;
|
|
gint attributes_mask;
|
|
|
|
g_return_if_fail (widget != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (widget));
|
|
|
|
GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
|
|
preview = GTK_PREVIEW (widget);
|
|
|
|
attributes.window_type = GDK_WINDOW_CHILD;
|
|
|
|
if (preview->expand)
|
|
{
|
|
attributes.width = widget->allocation.width;
|
|
attributes.height = widget->allocation.height;
|
|
}
|
|
else
|
|
{
|
|
attributes.width = MIN (widget->requisition.width, widget->allocation.width);
|
|
attributes.height = MIN (widget->requisition.height, widget->allocation.height);
|
|
}
|
|
|
|
attributes.x = widget->allocation.x + (widget->allocation.width - attributes.width) / 2;
|
|
attributes.y = widget->allocation.y + (widget->allocation.height - attributes.height) / 2;;
|
|
|
|
attributes.wclass = GDK_INPUT_OUTPUT;
|
|
attributes.visual = preview_class->info.visual;
|
|
attributes.colormap = preview_class->info.cmap;
|
|
attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
|
|
attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
|
|
|
|
widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, attributes_mask);
|
|
gdk_window_set_user_data (widget->window, widget);
|
|
|
|
widget->style = gtk_style_attach (widget->style, widget->window);
|
|
gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
|
|
}
|
|
|
|
static void
|
|
gtk_preview_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation)
|
|
{
|
|
GtkPreview *preview;
|
|
gint width, height;
|
|
|
|
g_return_if_fail (widget != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (widget));
|
|
|
|
preview = GTK_PREVIEW (widget);
|
|
widget->allocation = *allocation;
|
|
|
|
if (GTK_WIDGET_REALIZED (widget))
|
|
{
|
|
if (preview->expand)
|
|
{
|
|
width = widget->allocation.width;
|
|
height = widget->allocation.height;
|
|
}
|
|
else
|
|
{
|
|
width = MIN (widget->allocation.width, widget->requisition.width);
|
|
height = MIN (widget->allocation.height, widget->requisition.height);
|
|
}
|
|
|
|
gdk_window_move_resize (widget->window,
|
|
widget->allocation.x + (widget->allocation.width - width) / 2,
|
|
widget->allocation.y + (widget->allocation.height - height) / 2,
|
|
width, height);
|
|
}
|
|
}
|
|
|
|
static gint
|
|
gtk_preview_expose (GtkWidget *widget,
|
|
GdkEventExpose *event)
|
|
{
|
|
GtkPreview *preview;
|
|
gint width, height;
|
|
|
|
g_return_val_if_fail (widget != NULL, FALSE);
|
|
g_return_val_if_fail (GTK_IS_PREVIEW (widget), FALSE);
|
|
g_return_val_if_fail (event != NULL, FALSE);
|
|
|
|
if (GTK_WIDGET_DRAWABLE (widget))
|
|
{
|
|
preview = GTK_PREVIEW (widget);
|
|
|
|
gdk_window_get_size (widget->window, &width, &height);
|
|
|
|
gtk_preview_put (GTK_PREVIEW (widget),
|
|
widget->window, widget->style->black_gc,
|
|
event->area.x - (width - preview->buffer_width)/2,
|
|
event->area.y - (height - preview->buffer_height)/2,
|
|
event->area.x, event->area.y,
|
|
event->area.width, event->area.height);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
gtk_preview_make_buffer (GtkPreview *preview)
|
|
{
|
|
GtkWidget *widget;
|
|
gint width;
|
|
gint height;
|
|
|
|
g_return_if_fail (preview != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (preview));
|
|
|
|
widget = GTK_WIDGET (preview);
|
|
|
|
if (preview->expand &&
|
|
(widget->allocation.width != 0) &&
|
|
(widget->allocation.height != 0))
|
|
{
|
|
width = widget->allocation.width;
|
|
height = widget->allocation.height;
|
|
}
|
|
else
|
|
{
|
|
width = widget->requisition.width;
|
|
height = widget->requisition.height;
|
|
}
|
|
|
|
if (!preview->buffer ||
|
|
(preview->buffer_width != width) ||
|
|
(preview->buffer_height != height))
|
|
{
|
|
if (preview->buffer)
|
|
g_free (preview->buffer);
|
|
|
|
preview->buffer_width = width;
|
|
preview->buffer_height = height;
|
|
|
|
preview->rowstride = (preview->buffer_width * preview->bpp + 3) & -4;
|
|
preview->buffer = g_new0 (guchar,
|
|
preview->buffer_height *
|
|
preview->rowstride);
|
|
}
|
|
}
|
|
|
|
/* This will be useful for implementing gamma. */
|
|
static void
|
|
gtk_fill_lookup_array (guchar *array)
|
|
{
|
|
double one_over_gamma;
|
|
double ind;
|
|
int val;
|
|
int i;
|
|
|
|
one_over_gamma = 1.0 / preview_class->info.gamma;
|
|
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
ind = (double) i / 255.0;
|
|
val = (int) (255 * pow (ind, one_over_gamma));
|
|
array[i] = val;
|
|
}
|
|
}
|