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 Lesser 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 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser 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-2000. 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/.
31 #include <sys/types.h>
32 #ifdef HAVE_SYS_PARAM_H
33 #include <sys/param.h>
35 #include "gdk/gdkrgb.h"
36 #include "gtkpreview.h"
37 #include "gtksignal.h"
41 #define PREVIEW_CLASS(w) GTK_PREVIEW_CLASS (GTK_OBJECT (w)->klass)
49 static void gtk_preview_class_init (GtkPreviewClass *klass);
50 static void gtk_preview_init (GtkPreview *preview);
51 static void gtk_preview_set_property (GObject *object,
55 static void gtk_preview_get_property (GObject *object,
59 static void gtk_preview_finalize (GObject *object);
60 static void gtk_preview_realize (GtkWidget *widget);
61 static void gtk_preview_size_allocate (GtkWidget *widget,
62 GtkAllocation *allocation);
63 static gint gtk_preview_expose (GtkWidget *widget,
64 GdkEventExpose *event);
65 static void gtk_preview_make_buffer (GtkPreview *preview);
66 static void gtk_fill_lookup_array (guchar *array);
68 static GtkWidgetClass *parent_class = NULL;
69 static GtkPreviewClass *preview_class = NULL;
70 static gint install_cmap = FALSE;
74 gtk_preview_get_type (void)
76 static GtkType preview_type = 0;
80 static const GtkTypeInfo preview_info =
84 sizeof (GtkPreviewClass),
85 (GtkClassInitFunc) gtk_preview_class_init,
86 (GtkObjectInitFunc) gtk_preview_init,
87 /* reserved_1 */ NULL,
88 /* reserved_2 */ NULL,
89 (GtkClassInitFunc) NULL,
92 preview_type = gtk_type_unique (GTK_TYPE_WIDGET, &preview_info);
99 gtk_preview_class_init (GtkPreviewClass *klass)
101 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
102 GtkObjectClass *object_class;
103 GtkWidgetClass *widget_class;
105 object_class = (GtkObjectClass*) klass;
106 widget_class = (GtkWidgetClass*) klass;
108 parent_class = gtk_type_class (GTK_TYPE_WIDGET);
109 preview_class = klass;
111 gobject_class->finalize = gtk_preview_finalize;
113 gobject_class->set_property = gtk_preview_set_property;
114 gobject_class->get_property = gtk_preview_get_property;
116 widget_class->realize = gtk_preview_realize;
117 widget_class->size_allocate = gtk_preview_size_allocate;
118 widget_class->expose_event = gtk_preview_expose;
120 klass->info.lookup = NULL;
122 klass->info.gamma = 1.0;
126 g_object_class_install_property (gobject_class,
128 g_param_spec_boolean ("expand",
130 _("Whether the preview widget should take up the entire space it is allocated"),
136 gtk_preview_set_property (GObject *object,
141 GtkPreview *preview = GTK_PREVIEW (object);
146 gtk_preview_set_expand (preview, g_value_get_boolean (value));
149 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
155 gtk_preview_get_property (GObject *object,
162 preview = GTK_PREVIEW (object);
167 g_value_set_boolean (value, preview->expand);
170 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
176 gtk_preview_reset (void)
182 gtk_preview_init (GtkPreview *preview)
184 preview->buffer = NULL;
185 preview->buffer_width = 0;
186 preview->buffer_height = 0;
187 preview->expand = FALSE;
191 gtk_preview_uninit (void)
197 gtk_preview_new (GtkPreviewType type)
201 preview = gtk_type_new (gtk_preview_get_type ());
202 preview->type = type;
204 if (type == GTK_PREVIEW_COLOR)
209 preview->dither = GDK_RGB_DITHER_NORMAL;
211 return GTK_WIDGET (preview);
215 gtk_preview_size (GtkPreview *preview,
219 g_return_if_fail (GTK_IS_PREVIEW (preview));
221 if ((width != GTK_WIDGET (preview)->requisition.width) ||
222 (height != GTK_WIDGET (preview)->requisition.height))
224 GTK_WIDGET (preview)->requisition.width = width;
225 GTK_WIDGET (preview)->requisition.height = height;
228 g_free (preview->buffer);
229 preview->buffer = NULL;
234 gtk_preview_put (GtkPreview *preview,
245 GdkRectangle r1, r2, r3;
250 g_return_if_fail (GTK_IS_PREVIEW (preview));
251 g_return_if_fail (window != NULL);
253 if (!preview->buffer)
256 widget = GTK_WIDGET (preview);
260 r1.width = preview->buffer_width;
261 r1.height = preview->buffer_height;
268 if (!gdk_rectangle_intersect (&r1, &r2, &r3))
272 rowstride = preview->rowstride;
274 src = preview->buffer + r3.y * rowstride + r3.x * bpp;
276 if (preview->type == GTK_PREVIEW_COLOR)
277 gdk_draw_rgb_image (window,
279 destx + (r3.x - srcx),
280 desty + (r3.y - srcy),
287 gdk_draw_gray_image (window,
289 destx + (r3.x - srcx),
290 desty + (r3.y - srcy),
300 gtk_preview_draw_row (GtkPreview *preview,
309 g_return_if_fail (GTK_IS_PREVIEW (preview));
310 g_return_if_fail (data != NULL);
312 bpp = (preview->type == GTK_PREVIEW_COLOR ? 3 : 1);
313 rowstride = (preview->buffer_width * bpp + 3) & -4;
315 if ((w <= 0) || (y < 0))
318 g_return_if_fail (data != NULL);
320 gtk_preview_make_buffer (preview);
322 if (x + w > preview->buffer_width)
325 if (y + 1 > preview->buffer_height)
328 if (preview_class->info.gamma == 1.0)
329 memcpy (preview->buffer + y * rowstride + x * bpp, data, w * bpp);
336 if (preview_class->info.lookup != NULL)
337 lookup = preview_class->info.lookup;
340 preview_class->info.lookup = g_new (guchar, 256);
341 gtk_fill_lookup_array (preview_class->info.lookup);
342 lookup = preview_class->info.lookup;
346 dst = preview->buffer + y * rowstride + x * bpp;
347 for (i = 0; i < size; i++)
348 *dst++ = lookup[*src++];
353 gtk_preview_set_expand (GtkPreview *preview,
356 g_return_if_fail (GTK_IS_PREVIEW (preview));
358 expand = expand != FALSE;
360 if (preview->expand != expand)
362 preview->expand = expand;
363 gtk_widget_queue_resize (GTK_WIDGET (preview));
365 g_object_notify (G_OBJECT (preview), "expand");
370 gtk_preview_set_gamma (double _gamma)
373 preview_class = gtk_type_class (gtk_preview_get_type ());
375 if (preview_class->info.gamma != _gamma)
377 preview_class->info.gamma = _gamma;
378 if (preview_class->info.lookup != NULL)
380 g_free (preview_class->info.lookup);
381 preview_class->info.lookup = NULL;
387 gtk_preview_set_color_cube (guint nred_shades,
396 gtk_preview_set_install_cmap (gint _install_cmap)
398 /* effectively unimplemented */
399 install_cmap = _install_cmap;
403 gtk_preview_set_reserved (gint nreserved)
410 gtk_preview_set_dither (GtkPreview *preview,
413 preview->dither = dither;
417 gtk_preview_get_visual (void)
419 return gdk_rgb_get_visual ();
423 gtk_preview_get_cmap (void)
425 return gdk_rgb_get_colormap ();
429 gtk_preview_get_info (void)
432 preview_class = gtk_type_class (gtk_preview_get_type ());
434 return &preview_class->info;
439 gtk_preview_finalize (GObject *object)
443 g_return_if_fail (GTK_IS_PREVIEW (object));
445 preview = GTK_PREVIEW (object);
447 g_free (preview->buffer);
448 preview->type = (GtkPreviewType) -1;
450 G_OBJECT_CLASS (parent_class)->finalize (object);
454 gtk_preview_realize (GtkWidget *widget)
457 GdkWindowAttr attributes;
458 gint attributes_mask;
460 g_return_if_fail (GTK_IS_PREVIEW (widget));
462 GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
463 preview = GTK_PREVIEW (widget);
465 attributes.window_type = GDK_WINDOW_CHILD;
469 attributes.width = widget->allocation.width;
470 attributes.height = widget->allocation.height;
474 attributes.width = MIN (widget->requisition.width, widget->allocation.width);
475 attributes.height = MIN (widget->requisition.height, widget->allocation.height);
478 attributes.x = widget->allocation.x + (widget->allocation.width - attributes.width) / 2;
479 attributes.y = widget->allocation.y + (widget->allocation.height - attributes.height) / 2;;
481 attributes.wclass = GDK_INPUT_OUTPUT;
482 attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
483 attributes_mask = GDK_WA_X | GDK_WA_Y;
485 widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, attributes_mask);
486 gdk_window_set_user_data (widget->window, widget);
488 widget->style = gtk_style_attach (widget->style, widget->window);
489 gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
493 gtk_preview_size_allocate (GtkWidget *widget,
494 GtkAllocation *allocation)
499 g_return_if_fail (GTK_IS_PREVIEW (widget));
501 preview = GTK_PREVIEW (widget);
502 widget->allocation = *allocation;
504 if (GTK_WIDGET_REALIZED (widget))
508 width = widget->allocation.width;
509 height = widget->allocation.height;
513 width = MIN (widget->allocation.width, widget->requisition.width);
514 height = MIN (widget->allocation.height, widget->requisition.height);
517 gdk_window_move_resize (widget->window,
518 widget->allocation.x + (widget->allocation.width - width) / 2,
519 widget->allocation.y + (widget->allocation.height - height) / 2,
525 gtk_preview_expose (GtkWidget *widget,
526 GdkEventExpose *event)
531 g_return_val_if_fail (GTK_IS_PREVIEW (widget), FALSE);
532 g_return_val_if_fail (event != NULL, FALSE);
534 if (GTK_WIDGET_DRAWABLE (widget))
536 preview = GTK_PREVIEW (widget);
538 gdk_window_get_size (widget->window, &width, &height);
540 gtk_preview_put (GTK_PREVIEW (widget),
541 widget->window, widget->style->black_gc,
542 event->area.x - (width - preview->buffer_width)/2,
543 event->area.y - (height - preview->buffer_height)/2,
544 event->area.x, event->area.y,
545 event->area.width, event->area.height);
552 gtk_preview_make_buffer (GtkPreview *preview)
558 g_return_if_fail (GTK_IS_PREVIEW (preview));
560 widget = GTK_WIDGET (preview);
562 if (preview->expand &&
563 (widget->allocation.width != 0) &&
564 (widget->allocation.height != 0))
566 width = widget->allocation.width;
567 height = widget->allocation.height;
571 width = widget->requisition.width;
572 height = widget->requisition.height;
575 if (!preview->buffer ||
576 (preview->buffer_width != width) ||
577 (preview->buffer_height != height))
580 g_free (preview->buffer);
582 preview->buffer_width = width;
583 preview->buffer_height = height;
585 preview->rowstride = (preview->buffer_width * preview->bpp + 3) & -4;
586 preview->buffer = g_new0 (guchar,
587 preview->buffer_height *
592 /* This is used for implementing gamma. */
594 gtk_fill_lookup_array (guchar *array)
596 double one_over_gamma;
601 one_over_gamma = 1.0 / preview_class->info.gamma;
603 for (i = 0; i < 256; i++)
605 ind = (double) i / 255.0;
606 val = (int) (255 * pow (ind, one_over_gamma));