mirror of https://github.com/GNOME/gimp.git
272 lines
6.2 KiB
C
272 lines
6.2 KiB
C
#include "gimpimageP.h"
|
|
#include "gimage.h"
|
|
#include "gimpimage.h"
|
|
#include "lc_dialog.h"
|
|
|
|
#include "drawable.h"
|
|
#include "gdisplay.h"
|
|
#include "procedural_db.h"
|
|
|
|
#include "paletteP.h"
|
|
#include "undo.h"
|
|
|
|
#include "layer.h"
|
|
#include "layer_pvt.h"
|
|
#include "channel.h"
|
|
#include "tools.h"
|
|
#include "appenv.h"
|
|
#include "gimpset.h"
|
|
#include "dialog_handler.h"
|
|
|
|
/* gimage.c: Junk (ugly dependencies) from gimpimage.c on its way
|
|
to proper places. That is, the handlers should be moved to
|
|
layers_dialog, gdisplay, tools, etc.. */
|
|
|
|
static void gimage_dirty_handler (GimpImage* gimage);
|
|
static void gimage_destroy_handler (GimpImage* gimage);
|
|
static void gimage_cmap_change_handler (GimpImage* gimage,
|
|
gint ncol,
|
|
gpointer user_data);
|
|
static void gimage_rename_handler (GimpImage* gimage);
|
|
static void gimage_resize_handler (GimpImage* gimage);
|
|
static void gimage_restructure_handler (GimpImage* gimage);
|
|
static void gimage_repaint_handler (GimpImage* gimage,
|
|
gint, gint, gint, gint);
|
|
|
|
|
|
GImage*
|
|
gimage_new (int width,
|
|
int height,
|
|
GimpImageBaseType base_type)
|
|
{
|
|
GimpImage* gimage = gimp_image_new (width, height, base_type);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (gimage), "dirty",
|
|
GTK_SIGNAL_FUNC(gimage_dirty_handler), NULL);
|
|
gtk_signal_connect (GTK_OBJECT (gimage), "destroy",
|
|
GTK_SIGNAL_FUNC(gimage_destroy_handler), NULL);
|
|
gtk_signal_connect (GTK_OBJECT (gimage), "rename",
|
|
GTK_SIGNAL_FUNC(gimage_rename_handler), NULL);
|
|
gtk_signal_connect (GTK_OBJECT (gimage), "resize",
|
|
GTK_SIGNAL_FUNC(gimage_resize_handler), NULL);
|
|
gtk_signal_connect (GTK_OBJECT (gimage), "restructure",
|
|
GTK_SIGNAL_FUNC(gimage_restructure_handler), NULL);
|
|
gtk_signal_connect (GTK_OBJECT (gimage), "repaint",
|
|
GTK_SIGNAL_FUNC(gimage_repaint_handler), NULL);
|
|
gtk_signal_connect (GTK_OBJECT (gimage), "colormap_changed",
|
|
GTK_SIGNAL_FUNC(gimage_cmap_change_handler), NULL);
|
|
|
|
gimp_set_add (image_context, gimage);
|
|
|
|
return gimage;
|
|
}
|
|
|
|
GImage*
|
|
gimage_get_ID (gint ID)
|
|
{
|
|
return pdb_id_to_image (ID);
|
|
}
|
|
|
|
|
|
/* Ack, GImages have their own ref counts! This is going to cause
|
|
trouble.. It should be pretty easy to convert to proper GtkObject
|
|
ref counting, though. */
|
|
|
|
/* This caused trouble indeed. The ref_count was only used by the
|
|
displays showing the image, so I renamed it to disp_count to
|
|
make clear that it should only be used for display references.
|
|
(Sven, 23.01.2000) */
|
|
|
|
void
|
|
gimage_delete (GImage *gimage)
|
|
{
|
|
if (gimage->disp_count <= 0)
|
|
gtk_object_unref (GTK_OBJECT (gimage));
|
|
}
|
|
|
|
static void
|
|
invalidate_cb (gpointer image,
|
|
gpointer user_data)
|
|
{
|
|
gimp_image_invalidate_preview (GIMP_IMAGE(image));
|
|
}
|
|
|
|
void
|
|
gimage_invalidate_previews (void)
|
|
{
|
|
gimp_set_foreach (image_context, invalidate_cb, NULL);
|
|
}
|
|
|
|
static void
|
|
gimage_dirty_handler (GimpImage *gimage)
|
|
{
|
|
if (active_tool && !active_tool->preserve)
|
|
{
|
|
GDisplay* gdisp = active_tool->gdisp_ptr;
|
|
|
|
if (gdisp)
|
|
{
|
|
if (gdisp->gimage == gimage)
|
|
tools_initialize (active_tool->type, gdisp);
|
|
else
|
|
tools_initialize (active_tool->type, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimlist_cb (gpointer im,
|
|
gpointer data)
|
|
{
|
|
GSList** l=(GSList**)data;
|
|
*l=g_slist_prepend(*l, im);
|
|
}
|
|
|
|
gint
|
|
gimage_image_count (void)
|
|
{
|
|
GSList *list=NULL;
|
|
gint num_images = 0;
|
|
|
|
gimage_foreach (gimlist_cb, &list);
|
|
num_images = g_slist_length (list);
|
|
|
|
g_slist_free (list);
|
|
|
|
return (num_images);
|
|
}
|
|
|
|
static void
|
|
gimage_destroy_handler (GimpImage *gimage)
|
|
{
|
|
GList *list;
|
|
|
|
/* free the undo list */
|
|
undo_free (gimage);
|
|
|
|
/* free all guides */
|
|
list = gimage->guides;
|
|
while (list)
|
|
{
|
|
g_free ((Guide*) list->data);
|
|
list = g_list_next (list);
|
|
}
|
|
g_list_free (gimage->guides);
|
|
|
|
if (gimage_image_count () == 1) /* This is the last image */
|
|
{
|
|
dialog_show_toolbox ();
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimage_cmap_change_handler (GimpImage *gimage,
|
|
gint ncol,
|
|
gpointer user_data)
|
|
{
|
|
gdisplays_update_full (gimage);
|
|
|
|
if (gimp_image_base_type (gimage) == INDEXED)
|
|
paint_funcs_invalidate_color_hash_table (gimage, ncol);
|
|
}
|
|
|
|
static void
|
|
gimage_rename_handler (GimpImage *gimage)
|
|
{
|
|
gdisplays_update_title (gimage);
|
|
lc_dialog_update_image_list ();
|
|
|
|
palette_import_image_renamed (gimage);
|
|
}
|
|
|
|
static void
|
|
gimage_resize_handler (GimpImage *gimage)
|
|
{
|
|
undo_push_group_end (gimage);
|
|
|
|
/* shrink wrap and update all views */
|
|
channel_invalidate_previews (gimage);
|
|
layer_invalidate_previews (gimage);
|
|
gimp_image_invalidate_preview (gimage);
|
|
gdisplays_resize_cursor_label (gimage);
|
|
gdisplays_update_full (gimage);
|
|
gdisplays_shrink_wrap (gimage);
|
|
}
|
|
|
|
static void
|
|
gimage_restructure_handler (GimpImage* gimage)
|
|
{
|
|
gdisplays_update_title (gimage);
|
|
}
|
|
|
|
static void
|
|
gimage_repaint_handler (GimpImage *gimage,
|
|
gint x,
|
|
gint y,
|
|
gint w,
|
|
gint h)
|
|
{
|
|
gdisplays_update_area (gimage, x, y, w, h);
|
|
}
|
|
|
|
|
|
/* These really belong in the layer class */
|
|
|
|
void
|
|
gimage_set_layer_mask_apply (GImage *gimage,
|
|
GimpLayer *layer)
|
|
{
|
|
int off_x, off_y;
|
|
|
|
g_return_if_fail (gimage);
|
|
g_return_if_fail (layer);
|
|
|
|
if (! layer->mask)
|
|
return;
|
|
|
|
layer->apply_mask = ! layer->apply_mask;
|
|
drawable_offsets (GIMP_DRAWABLE (layer), &off_x, &off_y);
|
|
gdisplays_update_area (gimage, off_x, off_y,
|
|
drawable_width (GIMP_DRAWABLE (layer)),
|
|
drawable_height (GIMP_DRAWABLE (layer)));
|
|
}
|
|
|
|
void
|
|
gimage_set_layer_mask_edit (GImage *gimage,
|
|
Layer *layer,
|
|
gboolean edit)
|
|
{
|
|
/* find the layer */
|
|
if (!layer)
|
|
return;
|
|
|
|
if (layer->mask)
|
|
layer->edit_mask = edit;
|
|
}
|
|
|
|
void
|
|
gimage_set_layer_mask_show (GImage *gimage,
|
|
GimpLayer *layer)
|
|
{
|
|
int off_x, off_y;
|
|
|
|
g_return_if_fail (gimage);
|
|
g_return_if_fail (layer);
|
|
|
|
if (! layer->mask)
|
|
return;
|
|
|
|
layer->show_mask = ! layer->show_mask;
|
|
drawable_offsets (GIMP_DRAWABLE (layer), &off_x, &off_y);
|
|
gdisplays_update_area (gimage, off_x, off_y,
|
|
drawable_width (GIMP_DRAWABLE (layer)),
|
|
drawable_height (GIMP_DRAWABLE (layer)));
|
|
}
|
|
|
|
void
|
|
gimage_foreach (GFunc func,
|
|
gpointer user_data)
|
|
{
|
|
gimp_set_foreach (image_context, func, user_data);
|
|
}
|