gtkpreview.c 14.2 KB
Newer Older
Elliot Lee's avatar
Elliot Lee committed
1 2 3 4
/* 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
5
 * modify it under the terms of the GNU Lesser General Public
Elliot Lee's avatar
Elliot Lee committed
6 7 8 9 10 11
 * 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
12
 * Lesser General Public License for more details.
Elliot Lee's avatar
Elliot Lee committed
13
 *
14
 * You should have received a copy of the GNU Lesser General Public
15 16 17
 * License along with this library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
Elliot Lee's avatar
Elliot Lee committed
18 19
 */

20
/*
21
 * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
22 23 24 25 26
 * 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/. 
 */

27
#undef GDK_DISABLE_DEPRECATED
28
#undef GTK_DISABLE_DEPRECATED
29

30
#include <config.h>
31

Elliot Lee's avatar
Elliot Lee committed
32 33 34
#include <math.h>
#include <string.h>
#include <sys/types.h>
35
#ifdef HAVE_SYS_PARAM_H
Elliot Lee's avatar
Elliot Lee committed
36
#include <sys/param.h>
37
#endif
38
#include "gdk/gdkrgb.h"
39
#include "gtkalias.h"
Elliot Lee's avatar
Elliot Lee committed
40 41
#include "gtkpreview.h"
#include "gtksignal.h"
42
#include "gtkintl.h"
Elliot Lee's avatar
Elliot Lee committed
43 44 45 46


#define PREVIEW_CLASS(w)      GTK_PREVIEW_CLASS (GTK_OBJECT (w)->klass)

47
enum {
48 49
  PROP_0,
  PROP_EXPAND
50 51
};

Elliot Lee's avatar
Elliot Lee committed
52 53 54

static void   gtk_preview_class_init    (GtkPreviewClass  *klass);
static void   gtk_preview_init          (GtkPreview       *preview);
55 56 57 58 59 60 61 62
static void   gtk_preview_set_property  (GObject          *object,
					 guint             prop_id,
					 const GValue     *value,
					 GParamSpec       *pspec);
static void   gtk_preview_get_property  (GObject          *object,
					 guint             prop_id,
					 GValue           *value,
					 GParamSpec       *pspec);
63
static void   gtk_preview_finalize      (GObject          *object);
Elliot Lee's avatar
Elliot Lee committed
64
static void   gtk_preview_realize       (GtkWidget        *widget);
65 66
static void   gtk_preview_size_allocate (GtkWidget        *widget,
					 GtkAllocation    *allocation);
Elliot Lee's avatar
Elliot Lee committed
67 68 69
static gint   gtk_preview_expose        (GtkWidget        *widget,
				         GdkEventExpose   *event);
static void   gtk_preview_make_buffer   (GtkPreview       *preview);
70
static void   gtk_fill_lookup_array     (guchar           *array);
Elliot Lee's avatar
Elliot Lee committed
71 72 73 74 75 76

static GtkWidgetClass *parent_class = NULL;
static GtkPreviewClass *preview_class = NULL;
static gint install_cmap = FALSE;


Tor Lillqvist's avatar
Tor Lillqvist committed
77
GtkType
78
gtk_preview_get_type (void)
Elliot Lee's avatar
Elliot Lee committed
79
{
Tor Lillqvist's avatar
Tor Lillqvist committed
80
  static GtkType preview_type = 0;
Elliot Lee's avatar
Elliot Lee committed
81 82 83

  if (!preview_type)
    {
84
      static const GtkTypeInfo preview_info =
Elliot Lee's avatar
Elliot Lee committed
85 86 87 88 89 90
      {
        "GtkPreview",
        sizeof (GtkPreview),
        sizeof (GtkPreviewClass),
        (GtkClassInitFunc) gtk_preview_class_init,
        (GtkObjectInitFunc) gtk_preview_init,
91 92
	/* reserved_1 */ NULL,
        /* reserved_2 */ NULL,
93
        (GtkClassInitFunc) NULL,
Elliot Lee's avatar
Elliot Lee committed
94 95
      };

96
      preview_type = gtk_type_unique (GTK_TYPE_WIDGET, &preview_info);
Elliot Lee's avatar
Elliot Lee committed
97 98 99 100 101 102 103 104
    }

  return preview_type;
}

static void
gtk_preview_class_init (GtkPreviewClass *klass)
{
105
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Elliot Lee's avatar
Elliot Lee committed
106 107 108 109 110 111
  GtkObjectClass *object_class;
  GtkWidgetClass *widget_class;

  object_class = (GtkObjectClass*) klass;
  widget_class = (GtkWidgetClass*) klass;

112
  parent_class = gtk_type_class (GTK_TYPE_WIDGET);
Elliot Lee's avatar
Elliot Lee committed
113 114
  preview_class = klass;

115 116
  gobject_class->finalize = gtk_preview_finalize;

117 118
  gobject_class->set_property = gtk_preview_set_property;
  gobject_class->get_property = gtk_preview_get_property;
Elliot Lee's avatar
Elliot Lee committed
119 120

  widget_class->realize = gtk_preview_realize;
121
  widget_class->size_allocate = gtk_preview_size_allocate;
Elliot Lee's avatar
Elliot Lee committed
122 123
  widget_class->expose_event = gtk_preview_expose;

124
  klass->info.lookup = NULL;
Elliot Lee's avatar
Elliot Lee committed
125

126 127 128
  klass->info.gamma = 1.0;

  gdk_rgb_init ();
129

130 131 132
  g_object_class_install_property (gobject_class,
                                   PROP_EXPAND,
                                   g_param_spec_boolean ("expand",
133 134
							 P_("Expand"),
							 P_("Whether the preview widget should take up the entire space it is allocated"),
135 136
							 FALSE,
							 G_PARAM_READWRITE));
137 138 139
}

static void
140 141 142 143
gtk_preview_set_property (GObject      *object,
			  guint         prop_id,
			  const GValue *value,
			  GParamSpec   *pspec)
144 145 146
{
  GtkPreview *preview = GTK_PREVIEW (object);
  
147
  switch (prop_id)
148
    {
149 150 151 152 153
    case PROP_EXPAND:
      gtk_preview_set_expand (preview, g_value_get_boolean (value));
      break;
    default:
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
154 155 156 157 158
      break;
    }
}

static void
159 160 161 162
gtk_preview_get_property (GObject      *object,
			  guint         prop_id,
			  GValue       *value,
			  GParamSpec   *pspec)
163 164 165 166 167
{
  GtkPreview *preview;
  
  preview = GTK_PREVIEW (object);
  
168
  switch (prop_id)
169
    {
170 171
    case PROP_EXPAND:
      g_value_set_boolean (value, preview->expand);
172 173
      break;
    default:
174
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
175 176
      break;
    }
Elliot Lee's avatar
Elliot Lee committed
177 178
}

179 180 181
void
gtk_preview_reset (void)
{
182
  /* unimplemented */
183 184
}

Elliot Lee's avatar
Elliot Lee committed
185 186 187 188 189 190 191 192 193 194
static void
gtk_preview_init (GtkPreview *preview)
{
  preview->buffer = NULL;
  preview->buffer_width = 0;
  preview->buffer_height = 0;
  preview->expand = FALSE;
}

void
195
gtk_preview_uninit (void)
Elliot Lee's avatar
Elliot Lee committed
196
{
197
  /* unimplemented */
Elliot Lee's avatar
Elliot Lee committed
198 199 200 201 202 203 204 205 206 207
}

GtkWidget*
gtk_preview_new (GtkPreviewType type)
{
  GtkPreview *preview;

  preview = gtk_type_new (gtk_preview_get_type ());
  preview->type = type;

208 209 210 211 212 213 214
  if (type == GTK_PREVIEW_COLOR)
    preview->bpp = 3;
  else
    preview->bpp = 1;

  preview->dither = GDK_RGB_DITHER_NORMAL;

Elliot Lee's avatar
Elliot Lee committed
215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249
  return GTK_WIDGET (preview);
}

void
gtk_preview_size (GtkPreview *preview,
		  gint        width,
		  gint        height)
{
  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;
250 251 252
  guchar *src;
  guint bpp;
  guint rowstride;
Elliot Lee's avatar
Elliot Lee committed
253 254 255 256 257 258 259 260 261

  g_return_if_fail (GTK_IS_PREVIEW (preview));
  g_return_if_fail (window != NULL);

  if (!preview->buffer)
    return;

  widget = GTK_WIDGET (preview);

262 263
  r1.x = 0;
  r1.y = 0;
Elliot Lee's avatar
Elliot Lee committed
264 265 266
  r1.width = preview->buffer_width;
  r1.height = preview->buffer_height;

267 268
  r2.x = srcx;
  r2.y = srcy;
Elliot Lee's avatar
Elliot Lee committed
269 270 271 272 273 274
  r2.width = width;
  r2.height = height;

  if (!gdk_rectangle_intersect (&r1, &r2, &r3))
    return;

275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300
  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);
			
Elliot Lee's avatar
Elliot Lee committed
301 302 303 304 305 306 307 308 309
}

void
gtk_preview_draw_row (GtkPreview *preview,
		      guchar     *data,
		      gint        x,
		      gint        y,
		      gint        w)
{
310 311
  guint bpp;
  guint rowstride;
Elliot Lee's avatar
Elliot Lee committed
312 313 314

  g_return_if_fail (GTK_IS_PREVIEW (preview));
  g_return_if_fail (data != NULL);
315
  
316 317 318
  bpp = (preview->type == GTK_PREVIEW_COLOR ? 3 : 1);
  rowstride = (preview->buffer_width * bpp + 3) & -4;

Elliot Lee's avatar
Elliot Lee committed
319 320 321 322 323 324 325
  if ((w <= 0) || (y < 0))
    return;

  g_return_if_fail (data != NULL);

  gtk_preview_make_buffer (preview);

326 327 328 329
  if (x + w > preview->buffer_width)
    return;

  if (y + 1 > preview->buffer_height)
Elliot Lee's avatar
Elliot Lee committed
330 331
    return;

332 333 334
  if (preview_class->info.gamma == 1.0)
    memcpy (preview->buffer + y * rowstride + x * bpp, data, w * bpp);
  else
Elliot Lee's avatar
Elliot Lee committed
335
    {
336 337 338 339 340 341 342
      guint i, size;
      guchar *src, *dst;
      guchar *lookup;

      if (preview_class->info.lookup != NULL)
	lookup = preview_class->info.lookup;
      else
Elliot Lee's avatar
Elliot Lee committed
343
	{
344 345 346
	  preview_class->info.lookup = g_new (guchar, 256);
	  gtk_fill_lookup_array (preview_class->info.lookup);
	  lookup = preview_class->info.lookup;
Elliot Lee's avatar
Elliot Lee committed
347
	}
348 349 350 351 352
      size = w * bpp;
      src = data;
      dst = preview->buffer + y * rowstride + x * bpp;
      for (i = 0; i < size; i++)
	*dst++ = lookup[*src++];
Elliot Lee's avatar
Elliot Lee committed
353 354 355 356 357
    }
}

void
gtk_preview_set_expand (GtkPreview *preview,
358
			gboolean    expand)
Elliot Lee's avatar
Elliot Lee committed
359 360 361
{
  g_return_if_fail (GTK_IS_PREVIEW (preview));

362 363 364 365 366 367
  expand = expand != FALSE;

  if (preview->expand != expand)
    {
      preview->expand = expand;
      gtk_widget_queue_resize (GTK_WIDGET (preview));
368 369
 
      g_object_notify (G_OBJECT (preview), "expand"); 
370
    }
Elliot Lee's avatar
Elliot Lee committed
371 372 373 374 375
}

void
gtk_preview_set_gamma (double _gamma)
{
376 377 378 379
  if (!preview_class)
    preview_class = gtk_type_class (gtk_preview_get_type ());

  if (preview_class->info.gamma != _gamma)
Elliot Lee's avatar
Elliot Lee committed
380
    {
381 382 383 384 385 386
      preview_class->info.gamma = _gamma;
      if (preview_class->info.lookup != NULL)
	{
	  g_free (preview_class->info.lookup);
	  preview_class->info.lookup = NULL;
	}
Elliot Lee's avatar
Elliot Lee committed
387 388 389 390 391 392 393 394 395
    }
}

void
gtk_preview_set_color_cube (guint nred_shades,
			    guint ngreen_shades,
			    guint nblue_shades,
			    guint ngray_shades)
{
396
  /* unimplemented */
Elliot Lee's avatar
Elliot Lee committed
397 398 399 400 401
}

void
gtk_preview_set_install_cmap (gint _install_cmap)
{
402
  /* effectively unimplemented */
Elliot Lee's avatar
Elliot Lee committed
403 404 405 406 407 408 409
  install_cmap = _install_cmap;
}

void
gtk_preview_set_reserved (gint nreserved)
{

410 411 412 413 414 415 416 417
  /* unimplemented */
}

void
gtk_preview_set_dither (GtkPreview      *preview,
			GdkRgbDither     dither)
{
  preview->dither = dither;
Elliot Lee's avatar
Elliot Lee committed
418 419 420
}

GdkVisual*
421
gtk_preview_get_visual (void)
Elliot Lee's avatar
Elliot Lee committed
422
{
Owen Taylor's avatar
Owen Taylor committed
423
  return gdk_screen_get_rgb_visual (gdk_screen_get_default ());
Elliot Lee's avatar
Elliot Lee committed
424 425 426
}

GdkColormap*
427
gtk_preview_get_cmap (void)
Elliot Lee's avatar
Elliot Lee committed
428
{
Owen Taylor's avatar
Owen Taylor committed
429
  return gdk_screen_get_rgb_colormap (gdk_screen_get_default ());
Elliot Lee's avatar
Elliot Lee committed
430 431 432
}

GtkPreviewInfo*
433
gtk_preview_get_info (void)
Elliot Lee's avatar
Elliot Lee committed
434 435 436 437 438 439 440 441 442
{
  if (!preview_class)
    preview_class = gtk_type_class (gtk_preview_get_type ());

  return &preview_class->info;
}


static void
443
gtk_preview_finalize (GObject *object)
Elliot Lee's avatar
Elliot Lee committed
444 445 446 447 448 449 450 451 452 453
{
  GtkPreview *preview;

  g_return_if_fail (GTK_IS_PREVIEW (object));

  preview = GTK_PREVIEW (object);
  if (preview->buffer)
    g_free (preview->buffer);
  preview->type = (GtkPreviewType) -1;

454
  G_OBJECT_CLASS (parent_class)->finalize (object);
Elliot Lee's avatar
Elliot Lee committed
455 456 457 458 459 460 461 462 463 464 465 466 467 468 469
}

static void
gtk_preview_realize (GtkWidget *widget)
{
  GtkPreview *preview;
  GdkWindowAttr attributes;
  gint attributes_mask;

  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;
470 471 472 473 474 475 476 477 478 479 480 481 482 483 484

  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;;

Elliot Lee's avatar
Elliot Lee committed
485 486
  attributes.wclass = GDK_INPUT_OUTPUT;
  attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
487
  attributes_mask = GDK_WA_X | GDK_WA_Y;
Elliot Lee's avatar
Elliot Lee committed
488

489
  widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, attributes_mask);
Elliot Lee's avatar
Elliot Lee committed
490 491 492 493
  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);
494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525
}

static void   
gtk_preview_size_allocate (GtkWidget        *widget,
			   GtkAllocation    *allocation)
{
  GtkPreview *preview;
  gint width, height;

  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);
    }
Elliot Lee's avatar
Elliot Lee committed
526 527 528 529 530 531 532
}

static gint
gtk_preview_expose (GtkWidget      *widget,
		    GdkEventExpose *event)
{
  GtkPreview *preview;
533
  gint width, height;
Elliot Lee's avatar
Elliot Lee committed
534 535 536 537 538 539 540

  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);
541
      
542 543
      gdk_window_get_size (widget->window, &width, &height);

Elliot Lee's avatar
Elliot Lee committed
544 545
      gtk_preview_put (GTK_PREVIEW (widget),
		       widget->window, widget->style->black_gc,
546 547
		       event->area.x - (width - preview->buffer_width)/2,
		       event->area.y - (height - preview->buffer_height)/2,
Elliot Lee's avatar
Elliot Lee committed
548 549 550
		       event->area.x, event->area.y,
		       event->area.width, event->area.height);
    }
551
  
Elliot Lee's avatar
Elliot Lee committed
552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588
  return FALSE;
}

static void
gtk_preview_make_buffer (GtkPreview *preview)
{
  GtkWidget *widget;
  gint width;
  gint height;

  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;

589
      preview->rowstride = (preview->buffer_width * preview->bpp + 3) & -4;
Elliot Lee's avatar
Elliot Lee committed
590 591
      preview->buffer = g_new0 (guchar,
				preview->buffer_height *
592
				preview->rowstride);
Elliot Lee's avatar
Elliot Lee committed
593 594 595
    }
}

596
/* This is used for implementing gamma. */
Elliot Lee's avatar
Elliot Lee committed
597
static void
598
gtk_fill_lookup_array (guchar *array)
Elliot Lee's avatar
Elliot Lee committed
599 600 601 602 603 604
{
  double one_over_gamma;
  double ind;
  int val;
  int i;

605
  one_over_gamma = 1.0 / preview_class->info.gamma;
Elliot Lee's avatar
Elliot Lee committed
606 607 608

  for (i = 0; i < 256; i++)
    {
609 610 611
      ind = (double) i / 255.0;
      val = (int) (255 * pow (ind, one_over_gamma));
      array[i] = val;
Elliot Lee's avatar
Elliot Lee committed
612 613
    }
}