gimp/app/flip_tool.c

460 lines
11 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <stdlib.h>
#include "appenv.h"
#include "cursorutil.h"
#include "drawable.h"
#include "flip_tool.h"
#include "gdisplay.h"
#include "temp_buf.h"
#include "transform_core.h"
#include "undo.h"
#define FLIP 0
typedef struct _FlipOptions FlipOptions;
struct _FlipOptions
{
ToolType type;
};
/* forward function declarations */
static Tool * tools_new_flip_horz (void);
static Tool * tools_new_flip_vert (void);
static TileManager * flip_tool_flip_horz (GImage *, int, TileManager *, int);
static TileManager * flip_tool_flip_vert (GImage *, int, TileManager *, int);
static void flip_change_type (int);
static Argument * flip_invoker (Argument *);
/* Static variables */
static FlipOptions *flip_options = NULL;
static void
flip_type_callback (GtkWidget *w,
gpointer client_data)
{
flip_change_type ((long) client_data);
}
static FlipOptions *
create_flip_options (void)
{
FlipOptions *options;
GtkWidget *vbox;
GtkWidget *label;
GtkWidget *radio_box;
GtkWidget *radio_button;
GSList *group = NULL;
int i;
char *button_names[2] =
{
"Horizontal",
"Vertical",
};
/* the new options structure */
options = (FlipOptions *) g_malloc (sizeof (FlipOptions));
options->type = FLIP_HORZ;
/* the main vbox */
vbox = gtk_vbox_new (FALSE, 1);
/* the main label */
label = gtk_label_new ("Flip Tool Options");
gtk_box_pack_start (GTK_BOX (vbox), label, FALSE, FALSE, 0);
gtk_widget_show (label);
radio_box = gtk_vbox_new (FALSE, 2);
gtk_box_pack_start (GTK_BOX (vbox), radio_box, FALSE, FALSE, 0);
/* the radio buttons */
for (i = 0; i < 2; i++)
{
radio_button = gtk_radio_button_new_with_label (group, button_names[i]);
group = gtk_radio_button_group (GTK_RADIO_BUTTON (radio_button));
gtk_box_pack_start (GTK_BOX (radio_box), radio_button, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (radio_button), "toggled",
(GtkSignalFunc) flip_type_callback,
(gpointer) ((long) (FLIP_HORZ + i)));
gtk_widget_show (radio_button);
}
gtk_widget_show (radio_box);
/* Register this selection options widget with the main tools options dialog */
tools_register_options (FLIP_HORZ, vbox);
tools_register_options (FLIP_VERT, vbox);
return options;
}
static void *
flip_tool_transform_horz (Tool *tool,
gpointer gdisp_ptr,
int state)
{
TransformCore * transform_core;
GDisplay *gdisp;
transform_core = (TransformCore *) tool->private;
gdisp = (GDisplay *) gdisp_ptr;
switch (state)
{
case INIT :
transform_info = NULL;
break;
case MOTION :
break;
case RECALC :
break;
case FINISH :
transform_core->trans_info[FLIP] *= -1.0;
return flip_tool_flip_horz (gdisp->gimage, gimage_active_drawable (gdisp->gimage),
transform_core->original, transform_core->trans_info[FLIP]);
break;
}
return NULL;
}
static void *
flip_tool_transform_vert (Tool *tool,
gpointer gdisp_ptr,
int state)
{
TransformCore * transform_core;
GDisplay *gdisp;
transform_core = (TransformCore *) tool->private;
gdisp = (GDisplay *) gdisp_ptr;
switch (state)
{
case INIT :
transform_info = NULL;
break;
case MOTION :
break;
case RECALC :
break;
case FINISH :
transform_core->trans_info[FLIP] *= -1.0;
return flip_tool_flip_vert (gdisp->gimage, gimage_active_drawable (gdisp->gimage),
transform_core->original, transform_core->trans_info[FLIP]);
break;
}
return NULL;
}
Tool *
tools_new_flip ()
{
if (! flip_options)
flip_options = create_flip_options ();
switch (flip_options->type)
{
case FLIP_HORZ:
return tools_new_flip_horz ();
break;
case FLIP_VERT:
return tools_new_flip_vert ();
break;
default:
return NULL;
break;
}
}
void
tools_free_flip_tool (Tool *tool)
{
transform_core_free (tool);
}
static Tool *
tools_new_flip_horz ()
{
Tool * tool;
TransformCore * private;
tool = transform_core_new (FLIP_HORZ, NON_INTERACTIVE);
private = tool->private;
/* set the rotation specific transformation attributes */
private->trans_func = flip_tool_transform_horz;
private->trans_info[FLIP] = 1.0;
return tool;
}
static Tool *
tools_new_flip_vert ()
{
Tool * tool;
TransformCore * private;
tool = transform_core_new (FLIP_VERT, NON_INTERACTIVE);
private = tool->private;
/* set the rotation specific transformation attributes */
private->trans_func = flip_tool_transform_vert;
private->trans_info[FLIP] = 1.0;
return tool;
}
static TileManager *
flip_tool_flip_horz (GImage *gimage,
int drawable_id,
TileManager *orig,
int flip)
{
TileManager *new;
PixelRegion srcPR, destPR;
int i;
if (!orig)
return NULL;
if (flip > 0)
{
new = tile_manager_new (orig->levels[0].width, orig->levels[0].height, orig->levels[0].bpp);
pixel_region_init (&srcPR, orig, 0, 0, orig->levels[0].width, orig->levels[0].height, FALSE);
pixel_region_init (&destPR, new, 0, 0, orig->levels[0].width, orig->levels[0].height, TRUE);
copy_region (&srcPR, &destPR);
new->x = orig->x;
new->y = orig->y;
}
else
{
new = tile_manager_new (orig->levels[0].width, orig->levels[0].height, orig->levels[0].bpp);
new->x = orig->x;
new->y = orig->y;
for (i = 0; i < orig->levels[0].width; i++)
{
pixel_region_init (&srcPR, orig, i, 0, 1, orig->levels[0].height, FALSE);
pixel_region_init (&destPR, new, (orig->levels[0].width - i - 1), 0, 1, orig->levels[0].height, TRUE);
copy_region (&srcPR, &destPR);
}
}
return new;
}
static TileManager *
flip_tool_flip_vert (GImage *gimage,
int drawable_id,
TileManager *orig,
int flip)
{
TileManager *new;
PixelRegion srcPR, destPR;
int i;
if (!orig)
return NULL;
if (flip > 0)
{
new = tile_manager_new (orig->levels[0].width, orig->levels[0].height, orig->levels[0].bpp);
pixel_region_init (&srcPR, orig, 0, 0, orig->levels[0].width, orig->levels[0].height, FALSE);
pixel_region_init (&destPR, new, 0, 0, orig->levels[0].width, orig->levels[0].height, TRUE);
copy_region (&srcPR, &destPR);
new->x = orig->x;
new->y = orig->y;
}
else
{
new = tile_manager_new (orig->levels[0].width, orig->levels[0].height, orig->levels[0].bpp);
new->x = orig->x;
new->y = orig->y;
for (i = 0; i < orig->levels[0].height; i++)
{
pixel_region_init (&srcPR, orig, 0, i, orig->levels[0].width, 1, FALSE);
pixel_region_init (&destPR, new, 0, (orig->levels[0].height - i - 1), orig->levels[0].width, 1, TRUE);
copy_region (&srcPR, &destPR);
}
}
return new;
}
static void
flip_change_type (int new_type)
{
if (flip_options->type != new_type)
{
/* change the type, free the old tool, create the new tool */
flip_options->type = new_type;
tools_select (flip_options->type);
}
}
/* The flip procedure definition */
ProcArg flip_args[] =
{
{ PDB_IMAGE,
"image",
"the image"
},
{ PDB_DRAWABLE,
"drawable",
"the affected drawable"
},
{ PDB_INT32,
"flip_type",
"Type of flip: { HORIZONTAL (0), VERTICAL (1) }"
}
};
ProcArg flip_out_args[] =
{
{ PDB_DRAWABLE,
"drawable",
"the flipped drawable"
}
};
ProcRecord flip_proc =
{
"gimp_flip",
"Flip the specified drawable about its center either vertically or horizontally",
"This tool flips the specified drawable if no selection exists. If a selection exists, the portion of the drawable which lies under the selection is cut from the drawable and made into a floating selection which is then flipd by the specified amount. The return value is the ID of the flipped drawable. If there was no selection, this will be equal to the drawable ID supplied as input. Otherwise, this will be the newly created and flipd drawable. The flip type parameter indicates whether the flip will be applied horizontally or vertically.",
"Spencer Kimball & Peter Mattis",
"Spencer Kimball & Peter Mattis",
"1995-1996",
PDB_INTERNAL,
/* Input arguments */
3,
flip_args,
/* Output arguments */
1,
flip_out_args,
/* Exec method */
{ { flip_invoker } },
};
static Argument *
flip_invoker (Argument *args)
{
int success = TRUE;
GImage *gimage;
int drawable_id;
int flip_type;
int int_value;
TileManager *float_tiles;
TileManager *new_tiles;
int new_layer;
Layer *layer;
Argument *return_args;
drawable_id = -1;
flip_type = 0;
new_tiles = NULL;
layer = NULL;
/* the gimage */
if (success)
{
int_value = args[0].value.pdb_int;
if (! (gimage = gimage_get_ID (int_value)))
success = FALSE;
}
/* the drawable */
if (success)
{
int_value = args[1].value.pdb_int;
if (gimage == drawable_gimage (int_value))
drawable_id = int_value;
else
success = FALSE;
}
/* flip type */
if (success)
{
int_value = args[2].value.pdb_int;
switch (int_value)
{
case 0: flip_type = 0; break;
case 1: flip_type = 1; break;
default: success = FALSE;
}
}
/* call the flip procedure */
if (success)
{
/* Start a transform undo group */
undo_push_group_start (gimage, TRANSFORM_CORE_UNDO);
/* Cut/Copy from the specified drawable */
float_tiles = transform_core_cut (gimage, drawable_id, &new_layer);
/* flip the buffer */
switch (flip_type)
{
case 0: /* horz */
new_tiles = flip_tool_flip_horz (gimage, drawable_id, float_tiles, -1);
break;
case 1: /* vert */
new_tiles = flip_tool_flip_vert (gimage, drawable_id, float_tiles, -1);
break;
}
/* free the cut/copied buffer */
tile_manager_destroy (float_tiles);
if (new_tiles)
success = (layer = transform_core_paste (gimage, drawable_id, new_tiles, new_layer)) != NULL;
else
success = FALSE;
/* push the undo group end */
undo_push_group_end (gimage);
}
return_args = procedural_db_return_args (&flip_proc, success);
if (success)
return_args[1].value.pdb_int = layer->ID;
return return_args;
}