gimp/app/channel_ops.c
BST 1998 Adam D. Moss d1d3697c9d Moved the idlerender stuff into gdisplay.c. Implemented idlerender when
Sat Sep 26 20:46:18 BST 1998  Adam D. Moss <adam@gimp.org>

	* app/channel.c app/channel_ops.c app/drawable.c
	app/floating_sel.c app/gdisplay.c app/gdisplay.h
	app/gimpimage.c app/layers_dialog.c app/undo.c:

	Moved the idlerender stuff into gdisplay.c.  Implemented
	idlerender when doing floating_sel->layer, and undoing/redoing
	layer deletion.

	idlerender would be useful in many other places for improving
	interactivity, if it weren't for the following problems:

		* By definition, idlerender doesn't wait for a
		gdisplays_update() call	before starting work - it just
		runs in idle time, which due to CPU contention with
		plugins may not	be genuinely available idle time when
		things are 'noninteractive'.

		* Most GIMP functions don't know whether they're
		being run interactively or not.  idlerender only
		makes sense for interactive work.  This is why
		it is currently only applied to those functions which
		would normally only be activated manually.

		* Mixing idlerender and drawable_update() /
		gdisplays_update_area() calls can lead to a region
		being rerendered twice.

	Hence, some slogwork is needed before idlerender can be
	applied in the more general case.
1998-09-26 20:07:46 +00:00

660 lines
18 KiB
C

/* The GIMP -- an image manipulation program
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include <stdlib.h>
#include <string.h>
#include <math.h>
#include "appenv.h"
#include "actionarea.h"
#include "channel_ops.h"
#include "drawable.h"
#include "floating_sel.h"
#include "general.h"
#include "gdisplay.h"
#include "interface.h"
#include "palette.h"
#include "channel_pvt.h"
#define ENTRY_WIDTH 60
#define OFFSET_BACKGROUND 0
#define OFFSET_TRANSPARENT 1
typedef struct
{
GtkWidget *dlg;
GtkWidget *fill_options;
GtkWidget *off_x_entry;
GtkWidget *off_y_entry;
int wrap_around;
int transparent;
int background;
GimpImage* gimage;
} OffsetDialog;
/* Local procedures */
static void offset (GImage *gimage,
GimpDrawable *drawable,
int wrap_around,
int fill_type,
int offset_x,
int offset_y);
static void offset_ok_callback (GtkWidget *widget,
gpointer data);
static void offset_cancel_callback (GtkWidget *widget,
gpointer data);
static gint offset_delete_callback (GtkWidget *widget,
GdkEvent *event,
gpointer data);
static void offset_toggle_update (GtkWidget *widget,
gpointer data);
static void offset_wraparound_update (GtkWidget *widget,
gpointer data);
static void offset_halfheight_update (GtkWidget *widget,
gpointer data);
static Argument * channel_ops_offset_invoker (Argument *args);
void
channel_ops_offset (GimpImage* gimage)
{
OffsetDialog *off_d;
GtkWidget *button;
GtkWidget *label;
GtkWidget *check;
GtkWidget *push;
GtkWidget *toggle;
GtkWidget *vbox;
GtkWidget *toggle_vbox;
GtkWidget *table;
GSList *group = NULL;
GimpDrawable *drawable;
drawable = gimage_active_drawable (gimage);
off_d = g_new (OffsetDialog, 1);
off_d->wrap_around = TRUE;
off_d->transparent = drawable_has_alpha (drawable);
off_d->background = !off_d->transparent;
off_d->gimage = gimage;
off_d->dlg = gtk_dialog_new ();
gtk_window_set_wmclass (GTK_WINDOW (off_d->dlg), "offset", "Gimp");
gtk_window_set_title (GTK_WINDOW (off_d->dlg), "Offset");
/* handle the wm close signal */
gtk_signal_connect (GTK_OBJECT (off_d->dlg), "delete_event",
GTK_SIGNAL_FUNC (offset_delete_callback),
off_d);
/* Action area */
button = gtk_button_new_with_label ("OK");
GTK_WIDGET_SET_FLAGS (button, GTK_CAN_DEFAULT);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
(GtkSignalFunc) offset_ok_callback,
off_d);
gtk_box_pack_start (GTK_BOX (GTK_DIALOG (off_d->dlg)->action_area), button, TRUE, TRUE, 0);
gtk_widget_grab_default (button);
gtk_widget_show (button);
button = gtk_button_new_with_label ("Cancel");
GTK_WIDGET_SET_FLAGS (button, GTK_CAN_DEFAULT);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
(GtkSignalFunc) offset_cancel_callback,
off_d);
gtk_box_pack_start (GTK_BOX (GTK_DIALOG (off_d->dlg)->action_area), button, TRUE, TRUE, 0);
gtk_widget_show (button);
/* The vbox for first column of options */
vbox = gtk_vbox_new (FALSE, 1);
gtk_container_border_width (GTK_CONTAINER (vbox), 1);
gtk_box_pack_start (GTK_BOX (GTK_DIALOG (off_d->dlg)->vbox), vbox, TRUE, TRUE, 0);
/* the table for offsets */
table = gtk_table_new (2, 2, FALSE);
gtk_box_pack_start (GTK_BOX (vbox), table, TRUE, TRUE, 0);
label = gtk_label_new ("Offset X:");
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 0, 1, GTK_FILL, GTK_FILL, 2, 2);
off_d->off_x_entry = gtk_entry_new ();
gtk_entry_set_text (GTK_ENTRY (off_d->off_x_entry), "0");
gtk_widget_set_usize (off_d->off_x_entry, ENTRY_WIDTH, 0);
gtk_table_attach (GTK_TABLE (table), off_d->off_x_entry, 1, 2, 0, 1, GTK_FILL | GTK_EXPAND, GTK_FILL, 2, 2);
gtk_widget_show (label);
gtk_widget_show (off_d->off_x_entry);
label = gtk_label_new ("Offset Y:");
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 1, 2, GTK_FILL, GTK_FILL, 2, 2);
off_d->off_y_entry = gtk_entry_new ();
gtk_entry_set_text (GTK_ENTRY (off_d->off_y_entry), "0");
gtk_widget_set_usize (off_d->off_y_entry, ENTRY_WIDTH, 0);
gtk_table_attach (GTK_TABLE (table), off_d->off_y_entry, 1, 2, 1, 2, GTK_FILL | GTK_EXPAND, GTK_FILL, 2, 2);
gtk_widget_show (label);
gtk_widget_show (off_d->off_y_entry);
gtk_widget_show (table);
/* the wrap around option */
check = gtk_check_button_new_with_label ("Wrap-Around");
gtk_box_pack_start (GTK_BOX (vbox), check, FALSE, FALSE, 0);
gtk_widget_show (check);
/* The fill options */
off_d->fill_options = gtk_frame_new ("Fill Options");
gtk_frame_set_shadow_type (GTK_FRAME (off_d->fill_options), GTK_SHADOW_ETCHED_IN);
gtk_box_pack_start (GTK_BOX (vbox), off_d->fill_options, FALSE, TRUE, 0);
toggle_vbox = gtk_vbox_new (FALSE, 1);
gtk_container_border_width (GTK_CONTAINER (toggle_vbox), 5);
gtk_container_add (GTK_CONTAINER (off_d->fill_options), toggle_vbox);
toggle = gtk_radio_button_new_with_label (group, "Background");
group = gtk_radio_button_group (GTK_RADIO_BUTTON (toggle));
gtk_box_pack_start (GTK_BOX (toggle_vbox), toggle, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
(GtkSignalFunc) offset_toggle_update,
&off_d->background);
gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON (toggle), off_d->background);
gtk_widget_show (toggle);
if (drawable_has_alpha (drawable))
{
toggle = gtk_radio_button_new_with_label (group, "Transparent");
group = gtk_radio_button_group (GTK_RADIO_BUTTON (toggle));
gtk_box_pack_start (GTK_BOX (toggle_vbox), toggle, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
(GtkSignalFunc) offset_toggle_update,
&off_d->transparent);
gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON (toggle), off_d->transparent);
gtk_widget_show (toggle);
}
/* the by half height and half width offtion */
push = gtk_button_new_with_label ("Offset by (x/2),(y/2)");
gtk_box_pack_start (GTK_BOX (vbox), push, FALSE, FALSE,0);
gtk_widget_show (push);
gtk_widget_show (toggle_vbox);
gtk_widget_show (off_d->fill_options);
gtk_widget_show (vbox);
gtk_widget_show (off_d->dlg);
/* Hook up the wrap around */
gtk_signal_connect (GTK_OBJECT (check), "toggled",
(GtkSignalFunc) offset_wraparound_update,
off_d);
gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON (check), off_d->wrap_around);
/* Hook up the by half */
gtk_signal_connect (GTK_OBJECT (push), "clicked",
(GtkSignalFunc) offset_halfheight_update,
off_d);
}
static void
offset (GImage *gimage,
GimpDrawable *drawable,
int wrap_around,
int fill_type,
int offset_x,
int offset_y)
{
PixelRegion srcPR, destPR;
TileManager *new_tiles;
int width, height;
int src_x, src_y;
int dest_x, dest_y;
unsigned char fill[MAX_CHANNELS] = { 0 };
if (!drawable)
return;
width = drawable_width (drawable);
height = drawable_height (drawable);
if (wrap_around)
{
offset_x %= width;
offset_y %= height;
}
else
{
offset_x = BOUNDS (offset_x, -width, width);
offset_y = BOUNDS (offset_y, -height, height);
}
if (offset_x == 0 && offset_y == 0)
return;
new_tiles = tile_manager_new (width, height, drawable_bytes (drawable));
if (offset_x >= 0)
{
src_x = 0;
dest_x = offset_x;
width = BOUNDS ((width - offset_x), 0, width);
}
else
{
src_x = -offset_x;
dest_x = 0;
width = BOUNDS ((width + offset_x), 0, width);
}
if (offset_y >= 0)
{
src_y = 0;
dest_y = offset_y;
height = BOUNDS ((height - offset_y), 0, height);
}
else
{
src_y = -offset_y;
dest_y = 0;
height = BOUNDS ((height + offset_y), 0, height);
}
/* Copy the center region */
if (width && height)
{
pixel_region_init (&srcPR, drawable_data (drawable), src_x, src_y, width, height, FALSE);
pixel_region_init (&destPR, new_tiles, dest_x, dest_y, width, height, TRUE);
copy_region (&srcPR, &destPR);
}
/* Copy appropriately for wrap around */
if (wrap_around == TRUE)
{
if (offset_x >= 0 && offset_y >= 0)
{
src_x = drawable_width (drawable) - offset_x;
src_y = drawable_height (drawable) - offset_y;
}
else if (offset_x >= 0 && offset_y < 0)
{
src_x = drawable_width (drawable) - offset_x;
src_y = 0;
}
else if (offset_x < 0 && offset_y >= 0)
{
src_x = 0;
src_y = drawable_height (drawable) - offset_y;
}
else if (offset_x < 0 && offset_y < 0)
{
src_x = 0;
src_y = 0;
}
dest_x = (src_x + offset_x) % drawable_width (drawable);
if (dest_x < 0)
dest_x = drawable_width (drawable) + dest_x;
dest_y = (src_y + offset_y) % drawable_height (drawable);
if (dest_y < 0)
dest_y = drawable_height (drawable) + dest_y;
/* intersecting region */
if (offset_x != 0 && offset_y != 0)
{
pixel_region_init (&srcPR, drawable_data (drawable), src_x, src_y,
ABS (offset_x), ABS (offset_y), FALSE);
pixel_region_init (&destPR, new_tiles, dest_x, dest_y, ABS (offset_x), ABS (offset_y), TRUE);
copy_region (&srcPR, &destPR);
}
/* X offset */
if (offset_x != 0)
{
if (offset_y >= 0)
{
pixel_region_init (&srcPR, drawable_data (drawable), src_x, 0,
ABS (offset_x), drawable_height (drawable) - ABS (offset_y), FALSE);
pixel_region_init (&destPR, new_tiles, dest_x, dest_y + offset_y,
ABS (offset_x), drawable_height (drawable) - ABS (offset_y), TRUE);
}
else if (offset_y < 0)
{
pixel_region_init (&srcPR, drawable_data (drawable), src_x, src_y - offset_y,
ABS (offset_x), drawable_height (drawable) - ABS (offset_y), FALSE);
pixel_region_init (&destPR, new_tiles, dest_x, 0,
ABS (offset_x), drawable_height (drawable) - ABS (offset_y), TRUE);
}
copy_region (&srcPR, &destPR);
}
/* X offset */
if (offset_y != 0)
{
if (offset_x >= 0)
{
pixel_region_init (&srcPR, drawable_data (drawable), 0, src_y,
drawable_width (drawable) - ABS (offset_x), ABS (offset_y), FALSE);
pixel_region_init (&destPR, new_tiles, dest_x + offset_x, dest_y,
drawable_width (drawable) - ABS (offset_x), ABS (offset_y), TRUE);
}
else if (offset_x < 0)
{
pixel_region_init (&srcPR, drawable_data (drawable), src_x - offset_x, src_y,
drawable_width (drawable) - ABS (offset_x), ABS (offset_y), FALSE);
pixel_region_init (&destPR, new_tiles, 0, dest_y,
drawable_width (drawable) - ABS (offset_x), ABS (offset_y), TRUE);
}
copy_region (&srcPR, &destPR);
}
}
/* Otherwise, fill the vacated regions */
else
{
if (fill_type == OFFSET_BACKGROUND)
{
palette_get_background (&fill[0], &fill[1], &fill[2]);
if (drawable_has_alpha (drawable))
fill[drawable_bytes (drawable) - 1] = OPAQUE_OPACITY;
}
if (offset_x >= 0 && offset_y >= 0)
{
dest_x = 0;
dest_y = 0;
}
else if (offset_x >= 0 && offset_y < 0)
{
dest_x = 0;
dest_y = drawable_height (drawable) + offset_y;
}
else if (offset_x < 0 && offset_y >= 0)
{
dest_x = drawable_width (drawable) + offset_x;
dest_y = 0;
}
else if (offset_x < 0 && offset_y < 0)
{
dest_x = drawable_width (drawable) + offset_x;
dest_y = drawable_height (drawable) + offset_y;
}
/* intersecting region */
if (offset_x != 0 && offset_y != 0)
{
pixel_region_init (&destPR, new_tiles, dest_x, dest_y, ABS (offset_x), ABS (offset_y), TRUE);
color_region (&destPR, fill);
}
/* X offset */
if (offset_x != 0)
{
if (offset_y >= 0)
pixel_region_init (&destPR, new_tiles, dest_x, dest_y + offset_y,
ABS (offset_x), drawable_height (drawable) - ABS (offset_y), TRUE);
else if (offset_y < 0)
pixel_region_init (&destPR, new_tiles, dest_x, 0,
ABS (offset_x), drawable_height (drawable) - ABS (offset_y), TRUE);
color_region (&destPR, fill);
}
/* X offset */
if (offset_y != 0)
{
if (offset_x >= 0)
pixel_region_init (&destPR, new_tiles, dest_x + offset_x, dest_y,
drawable_width (drawable) - ABS (offset_x), ABS (offset_y), TRUE);
else if (offset_x < 0)
pixel_region_init (&destPR, new_tiles, 0, dest_y,
drawable_width (drawable) - ABS (offset_x), ABS (offset_y), TRUE);
color_region (&destPR, fill);
}
}
/* push an undo */
drawable_apply_image (drawable, 0, 0, drawable_width (drawable), drawable_height (drawable),
drawable_data (drawable), FALSE);
/* swap the tiles */
drawable->tiles = new_tiles;
printf(" offset ");fflush(stdout);
/* update the drawable */
drawable_update (drawable, 0, 0, drawable_width (drawable), drawable_height (drawable));
}
/*
* Interface callbacks
*/
static void
offset_ok_callback (GtkWidget *widget,
gpointer data)
{
OffsetDialog *off_d;
GImage *gimage;
GimpDrawable *drawable;
int offset_x, offset_y;
int fill_type;
off_d = (OffsetDialog *) data;
if ((gimage = off_d->gimage) != NULL)
{
drawable = gimage_active_drawable (gimage);
offset_x = (int) atof (gtk_entry_get_text (GTK_ENTRY (off_d->off_x_entry)));
offset_y = (int) atof (gtk_entry_get_text (GTK_ENTRY (off_d->off_y_entry)));
if (off_d->transparent)
fill_type = OFFSET_TRANSPARENT;
else
fill_type = OFFSET_BACKGROUND;
offset (gimage, drawable, off_d->wrap_around, fill_type, offset_x, offset_y);
gdisplays_flush ();
}
gtk_widget_destroy (off_d->dlg);
g_free (off_d);
}
static gint
offset_delete_callback (GtkWidget *widget,
GdkEvent *event,
gpointer data)
{
offset_cancel_callback (widget, data);
return TRUE;
}
static void
offset_cancel_callback (GtkWidget *widget,
gpointer data)
{
OffsetDialog *off_d;
off_d = (OffsetDialog *) data;
gtk_widget_destroy (off_d->dlg);
g_free (off_d);
}
static void
offset_toggle_update (GtkWidget *widget,
gpointer data)
{
int *toggle_val;
toggle_val = (int *) data;
if (GTK_TOGGLE_BUTTON (widget)->active)
*toggle_val = TRUE;
else
*toggle_val = FALSE;
}
static void
offset_wraparound_update (GtkWidget *widget,
gpointer data)
{
OffsetDialog *off_d;
off_d = (OffsetDialog *) data;
if (GTK_TOGGLE_BUTTON (widget)->active)
off_d->wrap_around = TRUE;
else
off_d->wrap_around = FALSE;
gtk_widget_set_sensitive (off_d->fill_options, !off_d->wrap_around);
}
static void
offset_halfheight_update (GtkWidget *widget,
gpointer data)
{
OffsetDialog *off_d;
GImage *gimage;
gchar buffer[16];
off_d = (OffsetDialog *) data;
gimage = off_d->gimage;
sprintf (buffer, "%d", gimage->width / 2);
gtk_entry_set_text (GTK_ENTRY (off_d->off_x_entry), buffer);
sprintf (buffer, "%d", gimage->height / 2);
gtk_entry_set_text (GTK_ENTRY (off_d->off_y_entry), buffer);
}
/*
* Procedure database functions and data structures
*/
/*
* Procedure database functions and data structures
*/
/* The offset procedure definition */
ProcArg channel_ops_offset_args[] =
{
{ PDB_IMAGE,
"image",
"the image"
},
{ PDB_DRAWABLE,
"drawable",
"the drawable to offset"
},
{ PDB_INT32,
"wrap_around",
"wrap image around or fill vacated regions"
},
{ PDB_INT32,
"fill_type",
"fill vacated regions of drawable with background or transparent: { OFFSET_BACKGROUND (0), OFFSET_TRANSPARENT (1) }"
},
{ PDB_INT32,
"offset_x",
"offset by this amount in X direction"
},
{ PDB_INT32,
"offset_y",
"offset by this amount in Y direction"
}
};
ProcRecord channel_ops_offset_proc =
{
"gimp_channel_ops_offset",
"Offset the drawable by the specified amounts in the X and Y directions",
"This procedure offsets the specified drawable by the amounts specified by 'offset_x' and 'offset_y'. If 'wrap_around' is set to TRUE, then portions of the drawable which are offset out of bounds are wrapped around. Alternatively, the undefined regions of the drawable can be filled with transparency or the background color, as specified by the 'fill_type' parameter.",
"Spencer Kimball & Peter Mattis",
"Spencer Kimball & Peter Mattis",
"1997",
PDB_INTERNAL,
/* Input arguments */
6,
channel_ops_offset_args,
/* Output arguments */
0,
NULL,
/* Exec method */
{ { channel_ops_offset_invoker } },
};
static Argument *
channel_ops_offset_invoker (Argument *args)
{
int success = TRUE;
int int_value;
GImage *gimage;
GimpDrawable *drawable;
int wrap_around;
int fill_type;
int offset_x;
int offset_y;
/* the gimage */
if (success)
{
int_value = args[0].value.pdb_int;
if (! (gimage = gimage_get_ID (int_value)))
success = FALSE;
}
if (success)
{
int_value = args[1].value.pdb_int;
drawable = drawable_get_ID (int_value);
if (drawable == NULL || gimage != drawable_gimage (drawable))
success = FALSE;
}
if (success)
{
wrap_around = (args[2].value.pdb_int) ? TRUE : FALSE;
}
if (success)
{
fill_type = args[3].value.pdb_int;
if (fill_type < OFFSET_BACKGROUND || fill_type > OFFSET_TRANSPARENT)
success = FALSE;
}
if (success)
{
offset_x = args[4].value.pdb_int;
offset_y = args[5].value.pdb_int;
}
if (success)
offset (gimage, drawable, wrap_around, fill_type, offset_x, offset_y);
return procedural_db_return_args (&channel_ops_offset_proc, success);
}