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.
21 * Modified by the GTK+ Team and others 1997-1999. See the AUTHORS
22 * file for a list of people on the GTK+ Team. See the ChangeLog
23 * files for a list of changes. These files are distributed with
24 * GTK+ at ftp://ftp.gtk.org/pub/gtk/.
30 #include <sys/types.h>
31 #include <sys/param.h>
33 #include "gdk/gdkrgb.h"
34 #include "gtkpreview.h"
35 #include "gtksignal.h"
38 #define PREVIEW_CLASS(w) GTK_PREVIEW_CLASS (GTK_OBJECT (w)->klass)
41 static void gtk_preview_class_init (GtkPreviewClass *klass);
42 static void gtk_preview_init (GtkPreview *preview);
43 static void gtk_preview_finalize (GtkObject *object);
44 static void gtk_preview_realize (GtkWidget *widget);
45 static void gtk_preview_size_allocate (GtkWidget *widget,
46 GtkAllocation *allocation);
47 static gint gtk_preview_expose (GtkWidget *widget,
48 GdkEventExpose *event);
49 static void gtk_preview_make_buffer (GtkPreview *preview);
50 static void gtk_fill_lookup_array (guchar *array);
52 static GtkWidgetClass *parent_class = NULL;
53 static GtkPreviewClass *preview_class = NULL;
54 static gint install_cmap = FALSE;
58 gtk_preview_get_type (void)
60 static guint preview_type = 0;
64 static const GtkTypeInfo preview_info =
68 sizeof (GtkPreviewClass),
69 (GtkClassInitFunc) gtk_preview_class_init,
70 (GtkObjectInitFunc) gtk_preview_init,
71 /* reserved_1 */ NULL,
72 /* reserved_2 */ NULL,
73 (GtkClassInitFunc) NULL,
76 preview_type = gtk_type_unique (gtk_widget_get_type (), &preview_info);
83 gtk_preview_class_init (GtkPreviewClass *klass)
85 GtkObjectClass *object_class;
86 GtkWidgetClass *widget_class;
88 object_class = (GtkObjectClass*) klass;
89 widget_class = (GtkWidgetClass*) klass;
91 parent_class = gtk_type_class (gtk_widget_get_type ());
92 preview_class = klass;
94 object_class->finalize = gtk_preview_finalize;
96 widget_class->realize = gtk_preview_realize;
97 widget_class->size_allocate = gtk_preview_size_allocate;
98 widget_class->expose_event = gtk_preview_expose;
100 klass->info.visual = NULL;
101 klass->info.cmap = NULL;
103 klass->info.lookup = NULL;
105 klass->info.gamma = 1.0;
108 klass->info.cmap = gdk_rgb_get_cmap ();
109 klass->info.visual = gdk_rgb_get_visual ();
113 gtk_preview_reset (void)
119 gtk_preview_init (GtkPreview *preview)
121 preview->buffer = NULL;
122 preview->buffer_width = 0;
123 preview->buffer_height = 0;
124 preview->expand = FALSE;
128 gtk_preview_uninit (void)
135 gtk_preview_new (GtkPreviewType type)
139 preview = gtk_type_new (gtk_preview_get_type ());
140 preview->type = type;
142 if (type == GTK_PREVIEW_COLOR)
147 preview->dither = GDK_RGB_DITHER_NORMAL;
149 return GTK_WIDGET (preview);
153 gtk_preview_size (GtkPreview *preview,
157 g_return_if_fail (preview != NULL);
158 g_return_if_fail (GTK_IS_PREVIEW (preview));
160 if ((width != GTK_WIDGET (preview)->requisition.width) ||
161 (height != GTK_WIDGET (preview)->requisition.height))
163 GTK_WIDGET (preview)->requisition.width = width;
164 GTK_WIDGET (preview)->requisition.height = height;
167 g_free (preview->buffer);
168 preview->buffer = NULL;
173 gtk_preview_put (GtkPreview *preview,
184 GdkRectangle r1, r2, r3;
189 g_return_if_fail (preview != NULL);
190 g_return_if_fail (GTK_IS_PREVIEW (preview));
191 g_return_if_fail (window != NULL);
193 if (!preview->buffer)
196 widget = GTK_WIDGET (preview);
200 r1.width = preview->buffer_width;
201 r1.height = preview->buffer_height;
208 if (!gdk_rectangle_intersect (&r1, &r2, &r3))
212 rowstride = preview->rowstride;
214 src = preview->buffer + r3.y * rowstride + r3.x * bpp;
216 if (preview->type == GTK_PREVIEW_COLOR)
217 gdk_draw_rgb_image (window,
219 destx + (r3.x - srcx),
220 desty + (r3.y - srcy),
227 gdk_draw_gray_image (window,
229 destx + (r3.x - srcx),
230 desty + (r3.y - srcy),
240 gtk_preview_draw_row (GtkPreview *preview,
249 g_return_if_fail (preview != NULL);
250 g_return_if_fail (GTK_IS_PREVIEW (preview));
251 g_return_if_fail (data != NULL);
252 g_return_if_fail (preview_class->info.visual != NULL);
254 bpp = (preview->type == GTK_PREVIEW_COLOR ? 3 : 1);
255 rowstride = (preview->buffer_width * bpp + 3) & -4;
257 if ((w <= 0) || (y < 0))
260 g_return_if_fail (data != NULL);
262 gtk_preview_make_buffer (preview);
264 if (x + w > preview->buffer_width)
267 if (y + 1 > preview->buffer_height)
270 if (preview_class->info.gamma == 1.0)
271 memcpy (preview->buffer + y * rowstride + x * bpp, data, w * bpp);
278 if (preview_class->info.lookup != NULL)
279 lookup = preview_class->info.lookup;
282 preview_class->info.lookup = g_new (guchar, 256);
283 gtk_fill_lookup_array (preview_class->info.lookup);
284 lookup = preview_class->info.lookup;
288 dst = preview->buffer + y * rowstride + x * bpp;
289 for (i = 0; i < size; i++)
290 *dst++ = lookup[*src++];
295 gtk_preview_set_expand (GtkPreview *preview,
298 g_return_if_fail (preview != NULL);
299 g_return_if_fail (GTK_IS_PREVIEW (preview));
301 preview->expand = (expand != FALSE);
305 gtk_preview_set_gamma (double _gamma)
308 preview_class = gtk_type_class (gtk_preview_get_type ());
310 if (preview_class->info.gamma != _gamma)
312 preview_class->info.gamma = _gamma;
313 if (preview_class->info.lookup != NULL)
315 g_free (preview_class->info.lookup);
316 preview_class->info.lookup = NULL;
322 gtk_preview_set_color_cube (guint nred_shades,
331 gtk_preview_set_install_cmap (gint _install_cmap)
333 /* effectively unimplemented */
334 install_cmap = _install_cmap;
338 gtk_preview_set_reserved (gint nreserved)
345 gtk_preview_set_dither (GtkPreview *preview,
348 preview->dither = dither;
352 gtk_preview_get_visual (void)
355 preview_class = gtk_type_class (gtk_preview_get_type ());
357 return preview_class->info.visual;
361 gtk_preview_get_cmap (void)
364 preview_class = gtk_type_class (gtk_preview_get_type ());
366 return preview_class->info.cmap;
370 gtk_preview_get_info (void)
373 preview_class = gtk_type_class (gtk_preview_get_type ());
375 return &preview_class->info;
380 gtk_preview_finalize (GtkObject *object)
384 g_return_if_fail (object != NULL);
385 g_return_if_fail (GTK_IS_PREVIEW (object));
387 preview = GTK_PREVIEW (object);
389 g_free (preview->buffer);
390 preview->type = (GtkPreviewType) -1;
392 (* GTK_OBJECT_CLASS (parent_class)->finalize) (object);
396 gtk_preview_realize (GtkWidget *widget)
399 GdkWindowAttr attributes;
400 gint attributes_mask;
402 g_return_if_fail (widget != NULL);
403 g_return_if_fail (GTK_IS_PREVIEW (widget));
405 GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
406 preview = GTK_PREVIEW (widget);
408 attributes.window_type = GDK_WINDOW_CHILD;
412 attributes.width = widget->allocation.width;
413 attributes.height = widget->allocation.height;
417 attributes.width = MIN (widget->requisition.width, widget->allocation.width);
418 attributes.height = MIN (widget->requisition.height, widget->allocation.height);
421 attributes.x = widget->allocation.x + (widget->allocation.width - attributes.width) / 2;
422 attributes.y = widget->allocation.y + (widget->allocation.height - attributes.height) / 2;;
424 attributes.wclass = GDK_INPUT_OUTPUT;
425 attributes.visual = preview_class->info.visual;
426 attributes.colormap = preview_class->info.cmap;
427 attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
428 attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
430 widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, attributes_mask);
431 gdk_window_set_user_data (widget->window, widget);
433 widget->style = gtk_style_attach (widget->style, widget->window);
434 gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
438 gtk_preview_size_allocate (GtkWidget *widget,
439 GtkAllocation *allocation)
444 g_return_if_fail (widget != NULL);
445 g_return_if_fail (GTK_IS_PREVIEW (widget));
447 preview = GTK_PREVIEW (widget);
448 widget->allocation = *allocation;
450 if (GTK_WIDGET_REALIZED (widget))
454 width = widget->allocation.width;
455 height = widget->allocation.height;
459 width = MIN (widget->allocation.width, widget->requisition.width);
460 height = MIN (widget->allocation.height, widget->requisition.height);
463 gdk_window_move_resize (widget->window,
464 widget->allocation.x + (widget->allocation.width - width) / 2,
465 widget->allocation.y + (widget->allocation.height - height) / 2,
471 gtk_preview_expose (GtkWidget *widget,
472 GdkEventExpose *event)
477 g_return_val_if_fail (widget != NULL, FALSE);
478 g_return_val_if_fail (GTK_IS_PREVIEW (widget), FALSE);
479 g_return_val_if_fail (event != NULL, FALSE);
481 if (GTK_WIDGET_DRAWABLE (widget))
483 preview = GTK_PREVIEW (widget);
485 gdk_window_get_size (widget->window, &width, &height);
487 gtk_preview_put (GTK_PREVIEW (widget),
488 widget->window, widget->style->black_gc,
489 event->area.x - (width - preview->buffer_width)/2,
490 event->area.y - (height - preview->buffer_height)/2,
491 event->area.x, event->area.y,
492 event->area.width, event->area.height);
499 gtk_preview_make_buffer (GtkPreview *preview)
505 g_return_if_fail (preview != NULL);
506 g_return_if_fail (GTK_IS_PREVIEW (preview));
508 widget = GTK_WIDGET (preview);
510 if (preview->expand &&
511 (widget->allocation.width != 0) &&
512 (widget->allocation.height != 0))
514 width = widget->allocation.width;
515 height = widget->allocation.height;
519 width = widget->requisition.width;
520 height = widget->requisition.height;
523 if (!preview->buffer ||
524 (preview->buffer_width != width) ||
525 (preview->buffer_height != height))
528 g_free (preview->buffer);
530 preview->buffer_width = width;
531 preview->buffer_height = height;
533 preview->rowstride = (preview->buffer_width * preview->bpp + 3) & -4;
534 preview->buffer = g_new0 (guchar,
535 preview->buffer_height *
540 /* This will be useful for implementing gamma. */
542 gtk_fill_lookup_array (guchar *array)
544 double one_over_gamma;
549 one_over_gamma = 1.0 / preview_class->info.gamma;
551 for (i = 0; i < 256; i++)
553 ind = (double) i / 255.0;
554 val = (int) (255 * pow (ind, one_over_gamma));