mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2024-11-17 14:30:15 +00:00
dfba226908
Sun Nov 5 04:24:53 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtkcellrenderertextpixbuf.c: Make parent_class static. Tue Sep 19 10:54:22 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtkimcontext*.[ch] gtk/gtkimmulticontext.[ch] gtk/gtktextlayout.[ch] gtk/gtktextview.c gtk/gtkentry.c: Add support for positioning the cursor within the preedit string. Mon Sep 18 23:56:32 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtktextview.c: Check for bindings after passing events to im context filter. Mon Sep 18 11:50:51 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtktextlayout.c (add_preedit_attrs): Handle empty attribute lists properly. Sun Sep 17 10:08:16 2000 Owen Taylor <otaylor@redhat.com> * gtk/queryimmodules.c (main): Return non-zero exit status if errors were encountered querying any modules. Sat Sep 16 14:01:52 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtk.h: include gtkmodule.h gtkoldeditable.h, don't include gtkthemes.h. * gtk/testgtk.c gtk/testtext.c: Set environment variables to point * gtk/Makefile.am: Add new .c and .h files, build gtk-query-immodules and use it to create a gtk.immodules file for use of test programs. * gtk/gtkpreview.c: remove extra blank line. Sat Sep 16 13:21:04 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtkimcontextsimple.c (gtk_im_context_simple_add_table): Add the ability to add extra tables beyond the default one, and also the ability to have compose sequences that are prefixes of other compose sequences. * gtk/gtkimcontextsimple.c: Export a preedit string which consists of possible candidates for keystrokes that have been entered but not yet committed. * gtk/gtkimcontext.[ch] gtk/immulticontext.[ch] gtk/gtkimcontextsimple.[ch]: add gtk_im_context_reset() * gtk/gtkmulticontext.[ch] (gtk_im_multicontext_append_menuitems): Add a function to add input-method switching menu items to a menu. * gtk/gtkimmulticontext.[ch]: Properly handly set_client_window when switching input methods. * gtk/gtkimcontextsimple.[ch]: Change the format of the compose table to allow compose tables of different lengths / sequence. Sat Sep 16 13:05:48 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtkimmodule.[ch]: Support routines for loading GtkIMContext implementations dynamically at runtime. * gtk/queryimmodules.c: Program to query the available input modules and write the results into a file. * gtk/gtkrc.[ch] (gtk_rc_get_im_module_file): Add extra config options "im_module_file" (cache file for input method modules), and "im_module_path" - path to look for modules when generating cache file. This doesn't scale. Sat Sep 16 13:09:06 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtkthemes.[ch] gtk/gtkmodule.[ch]: Move most of the generic code from gtkthemes into a new abstraction GtkModule which has the logic for implementing a loadable module which implements a number of GObject types. Sat Sep 16 13:07:13 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtkeditable.[ch]: Convert GtkEditable from a class into an interface * gtk/gtkoldeditable.[ch]: Move the old editable implementation into here, so legacy widgets can still rely on the implemenation. GtkOldEditable exports GtkEditable. Make selection handling code use new text conversion functions (and handle UTF-8 as a side-effect). Use GtkClipboard for CLIPBOARD. * gtk/gtktext.[ch] gtk/gtkcombo.c gtk/gtkspinbutton.c: Adopt to match above changes. * gtk/gtkentry.[ch]: Implement GtkEditable directly, avoid GtkOldEditable implementation. Restructure to reduce number of places that modify state directly. Move to GtkBindingSet. Display the preedit string. Queue recomputation of PangoLayout and scroll position to improve effiency of doing complex changes naively. Add a menu with cut/copy/paste and input method selection. Thu Sep 14 22:11:05 2000 Owen Taylor <otaylor@redhat.com> * gtk/gtktextlayout.[ch]: Add gtk_text_layout_set_preedit_string() to set preedit string and attributes; display preedit string by inserting string and attributes at cursor when creating the GtkTextLineDisplay. * gtk/gtktextlayout.c: Move all conversions between byte positions in PangoLayout and GtkTextIter into new functions line_display_iter_to_index/index_to_iter that properly handle the preedit string. * gtk/gtktextmark.[ch]: Restore gtk_text_mark_get_name, modify it to return const char * (eventually will end up as GCONST char *, most likely.) * gtk/gtktextview.[ch]: Handle the preedit string, call gtk_im_context_reset() as necessary, add a menu to switch input methods. * gtk/gtktextlayout.[ch]: Remove useless gtk_text_layout_get_log_attrs() function.
619 lines
14 KiB
C
619 lines
14 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 Lesser 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser 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-2000. 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 "gdk/gdkrgb.h"
|
|
#include "gtkpreview.h"
|
|
#include "gtksignal.h"
|
|
|
|
|
|
#define PREVIEW_CLASS(w) GTK_PREVIEW_CLASS (GTK_OBJECT (w)->klass)
|
|
|
|
enum {
|
|
ARG_0,
|
|
ARG_EXPAND
|
|
};
|
|
|
|
|
|
static void gtk_preview_class_init (GtkPreviewClass *klass);
|
|
static void gtk_preview_init (GtkPreview *preview);
|
|
static void gtk_preview_set_arg (GtkObject *object,
|
|
GtkArg *arg,
|
|
guint arg_id);
|
|
static void gtk_preview_get_arg (GtkObject *object,
|
|
GtkArg *arg,
|
|
guint arg_id);
|
|
static void gtk_preview_finalize (GObject *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;
|
|
|
|
|
|
GtkType
|
|
gtk_preview_get_type (void)
|
|
{
|
|
static GtkType 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_TYPE_WIDGET, &preview_info);
|
|
}
|
|
|
|
return preview_type;
|
|
}
|
|
|
|
static void
|
|
gtk_preview_class_init (GtkPreviewClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
GtkObjectClass *object_class;
|
|
GtkWidgetClass *widget_class;
|
|
|
|
object_class = (GtkObjectClass*) klass;
|
|
widget_class = (GtkWidgetClass*) klass;
|
|
|
|
parent_class = gtk_type_class (GTK_TYPE_WIDGET);
|
|
preview_class = klass;
|
|
|
|
gobject_class->finalize = gtk_preview_finalize;
|
|
|
|
object_class->set_arg = gtk_preview_set_arg;
|
|
object_class->get_arg = gtk_preview_get_arg;
|
|
|
|
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 ();
|
|
|
|
gtk_object_add_arg_type ("GtkPreview::expand",
|
|
GTK_TYPE_BOOL,
|
|
GTK_ARG_READWRITE,
|
|
ARG_EXPAND);
|
|
}
|
|
|
|
static void
|
|
gtk_preview_set_arg (GtkObject *object,
|
|
GtkArg *arg,
|
|
guint arg_id)
|
|
{
|
|
GtkPreview *preview = GTK_PREVIEW (object);
|
|
|
|
switch (arg_id)
|
|
{
|
|
case ARG_EXPAND:
|
|
gtk_preview_set_expand (preview, GTK_VALUE_BOOL (*arg));
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_preview_get_arg (GtkObject *object,
|
|
GtkArg *arg,
|
|
guint arg_id)
|
|
{
|
|
GtkPreview *preview;
|
|
|
|
preview = GTK_PREVIEW (object);
|
|
|
|
switch (arg_id)
|
|
{
|
|
case ARG_EXPAND:
|
|
GTK_VALUE_BOOL (*arg) = preview->expand;
|
|
break;
|
|
default:
|
|
arg->type = GTK_TYPE_INVALID;
|
|
break;
|
|
}
|
|
}
|
|
|
|
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,
|
|
gboolean expand)
|
|
{
|
|
g_return_if_fail (preview != NULL);
|
|
g_return_if_fail (GTK_IS_PREVIEW (preview));
|
|
|
|
expand = expand != FALSE;
|
|
|
|
if (preview->expand != expand)
|
|
{
|
|
preview->expand = expand;
|
|
gtk_widget_queue_resize (GTK_WIDGET (preview));
|
|
}
|
|
}
|
|
|
|
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 (GObject *object)
|
|
{
|
|
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;
|
|
|
|
G_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 is used 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;
|
|
}
|
|
}
|