1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
22 #include <sys/types.h>
23 #include <sys/param.h>
25 #include "gdk/gdkrgb.h"
26 #include "gtkpreview.h"
27 #include "gtksignal.h"
30 #define PREVIEW_CLASS(w) GTK_PREVIEW_CLASS (GTK_OBJECT (w)->klass)
33 static void gtk_preview_class_init (GtkPreviewClass *klass);
34 static void gtk_preview_init (GtkPreview *preview);
35 static void gtk_preview_finalize (GtkObject *object);
36 static void gtk_preview_realize (GtkWidget *widget);
37 static gint gtk_preview_expose (GtkWidget *widget,
38 GdkEventExpose *event);
39 static void gtk_preview_make_buffer (GtkPreview *preview);
40 static void gtk_fill_lookup_array (guchar *array);
42 static GtkWidgetClass *parent_class = NULL;
43 static GtkPreviewClass *preview_class = NULL;
44 static gint install_cmap = FALSE;
48 gtk_preview_get_type (void)
50 static guint preview_type = 0;
54 GtkTypeInfo preview_info =
58 sizeof (GtkPreviewClass),
59 (GtkClassInitFunc) gtk_preview_class_init,
60 (GtkObjectInitFunc) gtk_preview_init,
61 /* reserved_1 */ NULL,
62 /* reserved_2 */ NULL,
63 (GtkClassInitFunc) NULL,
66 preview_type = gtk_type_unique (gtk_widget_get_type (), &preview_info);
73 gtk_preview_class_init (GtkPreviewClass *klass)
75 GtkObjectClass *object_class;
76 GtkWidgetClass *widget_class;
78 object_class = (GtkObjectClass*) klass;
79 widget_class = (GtkWidgetClass*) klass;
81 parent_class = gtk_type_class (gtk_widget_get_type ());
82 preview_class = klass;
84 object_class->finalize = gtk_preview_finalize;
86 widget_class->realize = gtk_preview_realize;
87 widget_class->expose_event = gtk_preview_expose;
89 klass->info.visual = NULL;
90 klass->info.cmap = NULL;
92 klass->info.lookup = NULL;
94 klass->info.gamma = 1.0;
97 klass->info.cmap = gdk_rgb_get_cmap ();
98 klass->info.visual = gdk_rgb_get_visual ();
102 gtk_preview_reset (void)
108 gtk_preview_init (GtkPreview *preview)
110 GTK_WIDGET_SET_FLAGS (preview, GTK_BASIC);
112 preview->buffer = NULL;
113 preview->buffer_width = 0;
114 preview->buffer_height = 0;
115 preview->expand = FALSE;
119 gtk_preview_uninit (void)
126 gtk_preview_new (GtkPreviewType type)
130 preview = gtk_type_new (gtk_preview_get_type ());
131 preview->type = type;
133 if (type == GTK_PREVIEW_COLOR)
138 preview->dither = GDK_RGB_DITHER_NORMAL;
140 return GTK_WIDGET (preview);
144 gtk_preview_size (GtkPreview *preview,
148 g_return_if_fail (preview != NULL);
149 g_return_if_fail (GTK_IS_PREVIEW (preview));
151 if ((width != GTK_WIDGET (preview)->requisition.width) ||
152 (height != GTK_WIDGET (preview)->requisition.height))
154 GTK_WIDGET (preview)->requisition.width = width;
155 GTK_WIDGET (preview)->requisition.height = height;
158 g_free (preview->buffer);
159 preview->buffer = NULL;
164 gtk_preview_put (GtkPreview *preview,
175 GdkRectangle r1, r2, r3;
180 g_return_if_fail (preview != NULL);
181 g_return_if_fail (GTK_IS_PREVIEW (preview));
182 g_return_if_fail (window != NULL);
184 if (!preview->buffer)
187 widget = GTK_WIDGET (preview);
191 r1.width = preview->buffer_width;
192 r1.height = preview->buffer_height;
199 if (!gdk_rectangle_intersect (&r1, &r2, &r3))
203 rowstride = preview->rowstride;
205 src = preview->buffer + r3.y * rowstride + r3.x * bpp;
207 if (preview->type == GTK_PREVIEW_COLOR)
208 gdk_draw_rgb_image (window,
210 destx + (r3.x - srcx),
211 desty + (r3.y - srcy),
218 gdk_draw_gray_image (window,
220 destx + (r3.x - srcx),
221 desty + (r3.y - srcy),
231 gtk_preview_put_row (GtkPreview *preview,
238 g_warning ("gtk_preview_put_row not implemented (deprecated)\n");
242 gtk_preview_draw_row (GtkPreview *preview,
251 g_return_if_fail (preview != NULL);
252 g_return_if_fail (GTK_IS_PREVIEW (preview));
253 g_return_if_fail (data != NULL);
254 g_return_if_fail (preview_class->info.visual != NULL);
256 bpp = (preview->type == GTK_PREVIEW_COLOR ? 3 : 1);
257 rowstride = (preview->buffer_width * bpp + 3) & -4;
259 if ((w <= 0) || (y < 0))
262 g_return_if_fail (data != NULL);
264 gtk_preview_make_buffer (preview);
266 if (x + w > preview->buffer_width)
269 if (y + 1 > preview->buffer_height)
272 if (preview_class->info.gamma == 1.0)
273 memcpy (preview->buffer + y * rowstride + x * bpp, data, w * bpp);
280 if (preview_class->info.lookup != NULL)
281 lookup = preview_class->info.lookup;
284 preview_class->info.lookup = g_new (guchar, 256);
285 gtk_fill_lookup_array (preview_class->info.lookup);
286 lookup = preview_class->info.lookup;
290 dst = preview->buffer + y * rowstride + x * bpp;
291 for (i = 0; i < size; i++)
292 *dst++ = lookup[*src++];
297 gtk_preview_set_expand (GtkPreview *preview,
300 g_return_if_fail (preview != NULL);
301 g_return_if_fail (GTK_IS_PREVIEW (preview));
303 preview->expand = (expand != FALSE);
307 gtk_preview_set_gamma (double _gamma)
310 preview_class = gtk_type_class (gtk_preview_get_type ());
312 if (preview_class->info.gamma != _gamma)
314 preview_class->info.gamma = _gamma;
315 if (preview_class->info.lookup != NULL)
317 g_free (preview_class->info.lookup);
318 preview_class->info.lookup = NULL;
324 gtk_preview_set_color_cube (guint nred_shades,
333 gtk_preview_set_install_cmap (gint _install_cmap)
335 /* effectively unimplemented */
336 install_cmap = _install_cmap;
340 gtk_preview_set_reserved (gint nreserved)
347 gtk_preview_set_dither (GtkPreview *preview,
350 preview->dither = dither;
354 gtk_preview_get_visual (void)
357 preview_class = gtk_type_class (gtk_preview_get_type ());
359 return preview_class->info.visual;
363 gtk_preview_get_cmap (void)
366 preview_class = gtk_type_class (gtk_preview_get_type ());
368 return preview_class->info.cmap;
372 gtk_preview_get_info (void)
375 preview_class = gtk_type_class (gtk_preview_get_type ());
377 return &preview_class->info;
382 gtk_preview_finalize (GtkObject *object)
386 g_return_if_fail (object != NULL);
387 g_return_if_fail (GTK_IS_PREVIEW (object));
389 preview = GTK_PREVIEW (object);
391 g_free (preview->buffer);
392 preview->type = (GtkPreviewType) -1;
394 (* GTK_OBJECT_CLASS (parent_class)->finalize) (object);
398 gtk_preview_realize (GtkWidget *widget)
401 GdkWindowAttr attributes;
402 gint attributes_mask;
404 g_return_if_fail (widget != NULL);
405 g_return_if_fail (GTK_IS_PREVIEW (widget));
407 GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
408 preview = GTK_PREVIEW (widget);
410 attributes.window_type = GDK_WINDOW_CHILD;
411 attributes.x = widget->allocation.x;
412 attributes.y = widget->allocation.y;
413 attributes.width = widget->allocation.width;
414 attributes.height = widget->allocation.height;
415 attributes.wclass = GDK_INPUT_OUTPUT;
416 attributes.visual = preview_class->info.visual;
417 attributes.colormap = preview_class->info.cmap;
418 attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
419 attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
421 widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, attributes_mask);
422 gdk_window_set_user_data (widget->window, widget);
424 widget->style = gtk_style_attach (widget->style, widget->window);
425 gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
426 gdk_window_set_back_pixmap (widget->window, NULL, TRUE);
430 gtk_preview_expose (GtkWidget *widget,
431 GdkEventExpose *event)
435 g_return_val_if_fail (widget != NULL, FALSE);
436 g_return_val_if_fail (GTK_IS_PREVIEW (widget), FALSE);
437 g_return_val_if_fail (event != NULL, FALSE);
439 if (GTK_WIDGET_DRAWABLE (widget))
441 preview = GTK_PREVIEW (widget);
443 gtk_preview_put (GTK_PREVIEW (widget),
444 widget->window, widget->style->black_gc,
446 (widget->allocation.width - preview->buffer_width)/2,
448 (widget->allocation.height - preview->buffer_height)/2,
449 event->area.x, event->area.y,
450 event->area.width, event->area.height);
457 gtk_preview_make_buffer (GtkPreview *preview)
463 g_return_if_fail (preview != NULL);
464 g_return_if_fail (GTK_IS_PREVIEW (preview));
466 widget = GTK_WIDGET (preview);
468 if (preview->expand &&
469 (widget->allocation.width != 0) &&
470 (widget->allocation.height != 0))
472 width = widget->allocation.width;
473 height = widget->allocation.height;
477 width = widget->requisition.width;
478 height = widget->requisition.height;
481 if (!preview->buffer ||
482 (preview->buffer_width != width) ||
483 (preview->buffer_height != height))
486 g_free (preview->buffer);
488 preview->buffer_width = width;
489 preview->buffer_height = height;
491 preview->rowstride = (preview->buffer_width * preview->bpp + 3) & -4;
492 preview->buffer = g_new0 (guchar,
493 preview->buffer_height *
498 /* This will be useful for implementing gamma. */
500 gtk_fill_lookup_array (guchar *array)
502 double one_over_gamma;
507 one_over_gamma = 1.0 / preview_class->info.gamma;
509 for (i = 0; i < 256; i++)
511 ind = (double) i / 255.0;
512 val = (int) (255 * pow (ind, one_over_gamma));