gimpimage-resize.c 104 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
/* 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.
 */
18

19 20
#include "config.h"

21
#include <string.h>
22

Sven Neumann's avatar
Sven Neumann committed
23 24 25 26
#include <gtk/gtk.h>

#include "apptypes.h"

27
#include "channel.h"
28
#include "cursorutil.h"
29 30
#include "drawable.h"
#include "floating_sel.h"
31
#include "gdisplay.h"
32
#include "gimage_mask.h"
33
#include "gimpcontext.h"
34
#include "gimpimage.h"
35
#include "gimpmarshal.h"
36
#include "gimprc.h"
Manish Singh's avatar
Manish Singh committed
37
#include "gimpparasite.h"
38 39 40
#include "layer.h"
#include "paint_funcs.h"
#include "parasitelist.h"
41
#include "path.h"
42 43
#include "pixel_region.h"
#include "temp_buf.h"
scott's avatar
scott committed
44 45
#include "tile_manager.h"
#include "tile.h"
46
#include "undo.h"
47

48 49
#include "libgimp/gimpcolorspace.h"
#include "libgimp/gimplimits.h"
50
#include "libgimp/gimpmath.h"
51
#include "libgimp/gimpparasite.h"
52 53 54

#include "libgimp/gimpintl.h"

55

56 57 58 59 60 61 62
#ifdef DEBUG
#define TRC(x) printf x
#else
#define TRC(x)
#endif


63
/*  Local function declarations  */
64
static void     gimp_image_destroy               (GtkObject    *object);
65
static void     gimp_image_name_changed          (GimpObject   *object);
66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
static void     gimp_image_free_projection       (GimpImage    *gimage);
static void     gimp_image_allocate_shadow       (GimpImage    *gimage,
						  gint          width,
						  gint          height,
						  gint          bpp);
static void     gimp_image_allocate_projection   (GimpImage    *gimage);
static void     gimp_image_free_layers           (GimpImage    *gimage);
static void     gimp_image_free_channels         (GimpImage    *gimage);
static void     gimp_image_construct_layers      (GimpImage    *gimage,
						  gint          x,
						  gint          y,
						  gint          w,
						  gint          h);
static void     gimp_image_construct_channels    (GimpImage    *gimage,
						  gint          x,
						  gint          y,
						  gint          w,
						  gint          h);
static void     gimp_image_initialize_projection (GimpImage    *gimage,
						  gint          x,
						  gint          y,
						  gint          w,
						  gint          h);
static void     gimp_image_get_active_channels   (GimpImage    *gimage,
						  GimpDrawable *drawable,
						  gint         *active);
92 93

/*  projection functions  */
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116
static void     project_intensity                (GimpImage    *gimage,
						  Layer        *layer,
						  PixelRegion  *src,
						  PixelRegion  *dest,
						  PixelRegion  *mask);
static void     project_intensity_alpha          (GimpImage    *gimage,
						  Layer        *layer,
						  PixelRegion  *src,
						  PixelRegion  *dest,
						  PixelRegion  *mask);
static void     project_indexed                  (GimpImage    *gimage,
						  Layer        *layer,
						  PixelRegion  *src,
						  PixelRegion  *dest);
static void     project_indexed_alpha            (GimpImage    *gimage, 
						  Layer        *layer,
						  PixelRegion  *src, 
						  PixelRegion  *dest,
						  PixelRegion  *mask);
static void     project_channel                  (GimpImage    *gimage,
						  Channel      *layer,
						  PixelRegion  *src,
						  PixelRegion  *src2);
117 118 119 120

/*
 *  Global variables
 */
121
gint valid_combinations[][MAX_CHANNELS + 1] =
122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
{
  /* RGB GIMAGE */
  { -1, -1, -1, COMBINE_INTEN_INTEN, COMBINE_INTEN_INTEN_A },
  /* RGBA GIMAGE */
  { -1, -1, -1, COMBINE_INTEN_A_INTEN, COMBINE_INTEN_A_INTEN_A },
  /* GRAY GIMAGE */
  { -1, COMBINE_INTEN_INTEN, COMBINE_INTEN_INTEN_A, -1, -1 },
  /* GRAYA GIMAGE */
  { -1, COMBINE_INTEN_A_INTEN, COMBINE_INTEN_A_INTEN_A, -1, -1 },
  /* INDEXED GIMAGE */
  { -1, COMBINE_INDEXED_INDEXED, COMBINE_INDEXED_INDEXED_A, -1, -1 },
  /* INDEXEDA GIMAGE */
  { -1, -1, COMBINE_INDEXED_A_INDEXED_A, -1, -1 },
};

137
guint32 next_guide_id = 1;  /* For generating guide_ID handles for PDB stuff */
138

139 140 141 142 143

/*
 *  Static variables
 */

144 145
enum
{
146
  CLEAN,
147 148 149 150
  DIRTY,
  REPAINT,
  RESIZE,
  RESTRUCTURE,
151
  COLORMAP_CHANGED,
152
  UNDO_EVENT,
153 154
  LAST_SIGNAL
};
155

156 157 158
static guint gimp_image_signals[LAST_SIGNAL] = { 0 };

static GimpObjectClass *parent_class = NULL;
159

160

161 162 163
static void
gimp_image_class_init (GimpImageClass *klass)
{
164 165
  GtkObjectClass  *object_class;
  GimpObjectClass *gimp_object_class;
166

167 168
  object_class      = (GtkObjectClass *) klass;
  gimp_object_class = (GimpObjectClass *) klass;
169

170
  parent_class = gtk_type_class (GIMP_TYPE_OBJECT);
171

172
  gimp_image_signals[CLEAN] =
173 174 175 176 177 178 179 180
    gtk_signal_new ("clean",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       clean),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

181
  gimp_image_signals[DIRTY] =
182 183 184 185 186 187 188 189
    gtk_signal_new ("dirty",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       dirty),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

190
  gimp_image_signals[REPAINT] =
191 192 193 194 195 196 197 198 199 200 201 202
    gtk_signal_new ("repaint",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       repaint),
                    gimp_marshal_NONE__INT_INT_INT_INT,
                    GTK_TYPE_NONE, 4,
		    GTK_TYPE_INT,
		    GTK_TYPE_INT,
		    GTK_TYPE_INT,
		    GTK_TYPE_INT);

203
  gimp_image_signals[RESIZE] =
204 205 206 207 208 209 210 211
    gtk_signal_new ("resize",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       resize),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

212
  gimp_image_signals[RESTRUCTURE] =
213 214 215 216 217 218 219 220
    gtk_signal_new ("restructure",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       restructure),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

221
  gimp_image_signals[COLORMAP_CHANGED] =
222 223 224 225 226 227 228 229 230
    gtk_signal_new ("colormap_changed",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       colormap_changed),
                    gtk_marshal_NONE__INT,
                    GTK_TYPE_NONE, 1,
		    GTK_TYPE_INT);

231
  gimp_image_signals[UNDO_EVENT] = 
232 233 234 235 236 237 238 239
    gtk_signal_new ("undo_event",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       undo_event),
                    gtk_marshal_NONE__INT,
                    GTK_TYPE_NONE, 1,
		    GTK_TYPE_INT);
240 241
  
  gtk_object_class_add_signals (object_class, gimp_image_signals, LAST_SIGNAL);
242 243 244

  object_class->destroy = gimp_image_destroy;

245 246
  gimp_object_class->name_changed = gimp_image_name_changed;

247 248 249 250 251 252 253
  klass->clean            = NULL;
  klass->dirty            = NULL;
  klass->repaint          = NULL;
  klass->resize           = NULL;
  klass->restructure      = NULL;
  klass->colormap_changed = NULL;
  klass->undo_event       = NULL;
254 255 256 257 258
}


/* static functions */

Sven Neumann's avatar
Sven Neumann committed
259 260
static void 
gimp_image_init (GimpImage *gimage)
261
{
262 263 264 265 266 267 268 269 270
  gimage->save_proc             = NULL;

  gimage->width                 = 0;
  gimage->height                = 0;
  gimage->xresolution           = default_xresolution;
  gimage->yresolution           = default_yresolution;
  gimage->unit                  = default_units;
  gimage->base_type             = RGB;

271
  gimage->cmap                  = NULL;
272 273
  gimage->num_cols              = 0;

274
  gimage->dirty                 = 1;
275
  gimage->undo_on               = TRUE;
276 277 278 279

  gimage->instance_count        = 0;
  gimage->disp_count            = 0;

280
  gimage->tattoo_state          = 0;
281 282 283 284 285

  gimage->shadow                = NULL;

  gimage->construct_flag        = -1;
  gimage->proj_type             = RGBA_GIMAGE;
286
  gimage->projection            = NULL;
287

288
  gimage->guides                = NULL;
289

290 291 292
  gimage->layers                = NULL;
  gimage->channels              = NULL;
  gimage->layer_stack           = NULL;
293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310

  gimage->active_layer          = NULL;
  gimage->active_channel        = NULL;
  gimage->floating_sel          = NULL;
  gimage->selection_mask        = NULL;

  gimage->parasites             = parasite_list_new ();

  gimage->paths                 = NULL;

  gimage->by_color_select       = FALSE;

  gimage->qmask_state           = FALSE;
  gimage->qmask_color.r         = 1.0;
  gimage->qmask_color.g         = 0.0;
  gimage->qmask_color.b         = 0.0;
  gimage->qmask_color.a         = 0.5;

311 312 313 314
  gimage->undo_stack            = NULL;
  gimage->redo_stack            = NULL;
  gimage->undo_bytes            = 0;
  gimage->undo_levels           = 0;
315
  gimage->group_count           = 0;
316
  gimage->pushing_undo_group    = UNDO_NULL;
317 318 319
  gimage->undo_history          = NULL;

  gimage->comp_preview          = NULL;
320 321 322 323 324
  gimage->comp_preview_valid[0] = FALSE;
  gimage->comp_preview_valid[1] = FALSE;
  gimage->comp_preview_valid[2] = FALSE;
}

Sven Neumann's avatar
Sven Neumann committed
325 326 327
GtkType 
gimp_image_get_type (void) 
{
328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347
  static GtkType image_type = 0;

  if (! image_type)
    {
      GtkTypeInfo image_info =
      {
        "GimpImage",
        sizeof (GimpImage),
        sizeof (GimpImageClass),
        (GtkClassInitFunc) gimp_image_class_init,
        (GtkObjectInitFunc) gimp_image_init,
        /* reserved_1 */ NULL,
        /* reserved_2 */ NULL,
        (GtkClassInitFunc) NULL,
      };

      image_type = gtk_type_unique (GIMP_TYPE_OBJECT, &image_info);
    }

  return image_type;
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371
}

static void
gimp_image_allocate_projection (GimpImage *gimage)
{
  if (gimage->projection)
    gimp_image_free_projection (gimage);

  /*  Find the number of bytes required for the projection.
   *  This includes the intensity channels and an alpha channel
   *  if one doesn't exist.
   */
  switch (gimp_image_base_type (gimage))
    {
    case RGB:
    case INDEXED:
      gimage->proj_bytes = 4;
      gimage->proj_type = RGBA_GIMAGE;
      break;
    case GRAY:
      gimage->proj_bytes = 2;
      gimage->proj_type = GRAYA_GIMAGE;
      break;
    default:
372
      g_assert_not_reached ();
373 374 375
    }

  /*  allocate the new projection  */
376 377
  gimage->projection = tile_manager_new (gimage->width, gimage->height,
					 gimage->proj_bytes);
378
  tile_manager_set_user_data (gimage->projection, (void *) gimage);
379 380 381 382 383 384 385 386 387 388 389 390 391
  tile_manager_set_validate_proc (gimage->projection, gimp_image_validate);
}

static void
gimp_image_free_projection (GimpImage *gimage)
{
  if (gimage->projection)
    tile_manager_destroy (gimage->projection);

  gimage->projection = NULL;
}

static void
Sven Neumann's avatar
Sven Neumann committed
392
gimp_image_allocate_shadow (GimpImage *gimage, 
393 394 395
			    gint       width, 
			    gint       height, 
			    gint       bpp)
396 397 398 399 400 401 402 403 404
{
  /*  allocate the new projection  */
  gimage->shadow = tile_manager_new (width, height, bpp);
}


/* function definitions */

GimpImage *
405 406 407
gimp_image_new (gint               width,
		gint               height,
		GimpImageBaseType  base_type)
408
{
409
  GimpImage *gimage = GIMP_IMAGE (gtk_type_new (GIMP_TYPE_IMAGE));
410
  gint i;
411

412 413
  gimage->width     = width;
  gimage->height    = height;
414 415 416 417 418 419 420 421 422 423
  gimage->base_type = base_type;

  switch (base_type)
    {
    case RGB:
    case GRAY:
      break;
    case INDEXED:
      /* always allocate 256 colors for the colormap */
      gimage->num_cols = 0;
424
      gimage->cmap     = (guchar *) g_malloc0 (COLORMAP_SIZE);
425 426 427 428 429 430 431 432
      break;
    default:
      break;
    }

  /*  set all color channels visible and active  */
  for (i = 0; i < MAX_CHANNELS; i++)
    {
433 434
      gimage->visible[i] = TRUE;
      gimage->active[i]  = TRUE;
435 436 437
    }

  /* create the selection mask */
438 439
  gimage->selection_mask = channel_new_mask (gimage,
					     gimage->width, gimage->height);
440 441 442 443 444 445


  return gimage;
}

void
446 447
gimp_image_set_filename (GimpImage   *gimage,
			 const gchar *filename)
448
{
449 450
  gimp_object_set_name (GIMP_OBJECT (gimage), filename);
}
451

452 453 454 455 456
static void
gimp_image_name_changed (GimpObject *object)
{
  GimpImage   *gimage;
  const gchar *name;
457

458 459 460
  if (GIMP_OBJECT_CLASS (parent_class)->name_changed)
    GIMP_OBJECT_CLASS (parent_class)->name_changed (object);

461 462
  gimage = GIMP_IMAGE (object);
  name   = gimp_object_get_name (object);
463

464
  if (! (name && strlen (name)))
465
    {
466 467
      g_free (object->name);
      object->name = NULL;
468 469 470
    }
}

471
void
472
gimp_image_set_resolution (GimpImage *gimage,
473 474
			   gdouble    xresolution,
			   gdouble    yresolution)
475
{
476 477 478 479 480
  /* nothing to do if setting res to the same as before */
  if ((ABS (gimage->xresolution - xresolution) < 1e-5) &&
      (ABS (gimage->yresolution - yresolution) < 1e-5))
      return;

481 482 483
  /* don't allow to set the resolution out of bounds */
  if (xresolution < GIMP_MIN_RESOLUTION || xresolution > GIMP_MAX_RESOLUTION ||
      yresolution < GIMP_MIN_RESOLUTION || yresolution > GIMP_MAX_RESOLUTION)
484 485
    return;

486
  undo_push_resolution (gimage);
487

488 489
  gimage->xresolution = xresolution;
  gimage->yresolution = yresolution;
490 491 492

  /* really just want to recalc size and repaint */
  gdisplays_shrink_wrap (gimage);
493 494
}

495
void
496 497 498
gimp_image_get_resolution (const GimpImage *gimage,
			   gdouble         *xresolution,
			   gdouble         *yresolution)
499
{
500 501
  g_return_if_fail (xresolution && yresolution);

502 503
  *xresolution = gimage->xresolution;
  *yresolution = gimage->yresolution;
504 505
}

506 507
void
gimp_image_set_unit (GimpImage *gimage,
508
		     GimpUnit   unit)
509
{
510 511
  undo_push_resolution (gimage);

512 513 514
  gimage->unit = unit;
}

515
GimpUnit
516
gimp_image_get_unit (const GimpImage *gimage)
517 518 519 520
{
  return gimage->unit;
}

521
void
Sven Neumann's avatar
Sven Neumann committed
522 523
gimp_image_set_save_proc (GimpImage     *gimage, 
			  PlugInProcDef *proc)
524 525 526 527 528
{
  gimage->save_proc = proc;
}

PlugInProcDef *
529
gimp_image_get_save_proc (const GimpImage *gimage)
530 531 532
{
  return gimage->save_proc;
}
533

534 535 536 537 538 539 540 541 542 543 544 545
gint
gimp_image_get_width (const GimpImage *gimage)
{
  return gimage->width;
}

gint
gimp_image_get_height (const GimpImage *gimage)
{
  return gimage->height;
}

546
void
547
gimp_image_resize (GimpImage *gimage, 
548 549 550 551
		   gint       new_width, 
		   gint       new_height,
		   gint       offset_x, 
		   gint       offset_y)
552 553
{
  Channel *channel;
554 555 556 557
  Layer   *layer;
  Layer   *floating_layer;
  GSList  *list;
  GList   *guide_list;
558

559
  gimp_add_busy_cursors ();
560

561
  g_assert (new_width > 0 && new_height > 0);
562 563 564 565

  /*  Get the floating layer if one exists  */
  floating_layer = gimp_image_floating_sel (gimage);

Sven Neumann's avatar
Sven Neumann committed
566
  undo_push_group_start (gimage, IMAGE_RESIZE_UNDO);
567 568 569 570 571 572 573 574 575

  /*  Relax the floating selection  */
  if (floating_layer)
    floating_sel_relax (floating_layer, TRUE);

  /*  Push the image size to the stack  */
  undo_push_gimage_mod (gimage);

  /*  Set the new width and height  */
576
  gimage->width  = new_width;
577 578 579
  gimage->height = new_height;

  /*  Resize all channels  */
580
  for (list = gimage->channels; list; list = g_slist_next (list))
581 582
    {
      channel = (Channel *) list->data;
583

584
      channel_resize (channel, new_width, new_height, offset_x, offset_y);
585 586 587 588 589 590 591 592 593
    }

  /*  Reposition or remove any guides  */
  guide_list = gimage->guides;
  while (guide_list)
    {
      Guide *guide;

      guide = (Guide*) guide_list->data;
Sven Neumann's avatar
Sven Neumann committed
594
      guide_list = g_list_next (guide_list);
595 596 597

      switch (guide->orientation)
	{
598
	case ORIENTATION_HORIZONTAL:
Sven Neumann's avatar
Sven Neumann committed
599
	  undo_push_guide (gimage, guide);
600 601 602 603
	  guide->position += offset_y;
	  if (guide->position < 0 || guide->position > new_height)
	    gimp_image_delete_guide (gimage, guide);
	  break;
604

605
	case ORIENTATION_VERTICAL:
Sven Neumann's avatar
Sven Neumann committed
606
	  undo_push_guide (gimage, guide);
607 608 609 610
	  guide->position += offset_x;
	  if (guide->position < 0 || guide->position > new_width)
	    gimp_image_delete_guide (gimage, guide);
	  break;
611

612
	default:
613
	  g_error ("Unknown guide orientation\n");
614
	}
615 616 617
    }

  /*  Don't forget the selection mask!  */
618 619
  channel_resize (gimage->selection_mask,
		  new_width, new_height, offset_x, offset_y);
620 621 622
  gimage_mask_invalidate (gimage);

  /*  Reposition all layers  */
623
  for (list = gimage->layers; list; list = g_slist_next (list))
624 625
    {
      layer = (Layer *) list->data;
626

627 628 629 630 631 632 633 634 635 636 637
      layer_translate (layer, offset_x, offset_y);
    }

  /*  Make sure the projection matches the gimage size  */
  gimp_image_projection_realloc (gimage);

  /*  Rigor the floating selection  */
  if (floating_layer)
    floating_sel_rigor (floating_layer, TRUE);

  gtk_signal_emit (GTK_OBJECT (gimage), gimp_image_signals[RESIZE]);
638

639
  gimp_remove_busy_cursors (NULL);
640 641 642
}

void
643
gimp_image_scale (GimpImage *gimage, 
644 645
		  gint       new_width, 
		  gint       new_height)
646 647
{
  Channel *channel;
Sven Neumann's avatar
Sven Neumann committed
648 649 650
  Layer   *layer;
  Layer   *floating_layer;
  GSList  *list;
651
  GSList  *remove = NULL;
Sven Neumann's avatar
Sven Neumann committed
652 653
  GList   *glist;
  Guide   *guide;
654 655 656 657
  gint     old_width;
  gint     old_height;
  gdouble  img_scale_w = 1.0;
  gdouble  img_scale_h = 1.0;
658

659
  if ((new_width == 0) || (new_height == 0))
660
    {
661
      g_message (("gimp_image_scale: Scaling to zero width or height has been rejected."));
662 663
      return;
    }
664

665
  gimp_add_busy_cursors ();
666

667 668 669
  /*  Get the floating layer if one exists  */
  floating_layer = gimp_image_floating_sel (gimage);

Sven Neumann's avatar
Sven Neumann committed
670
  undo_push_group_start (gimage, IMAGE_SCALE_UNDO);
671 672 673 674 675 676 677 678 679

  /*  Relax the floating selection  */
  if (floating_layer)
    floating_sel_relax (floating_layer, TRUE);

  /*  Push the image size to the stack  */
  undo_push_gimage_mod (gimage);

  /*  Set the new width and height  */
680 681 682

  old_width      = gimage->width;
  old_height     = gimage->height;
683
  gimage->width  = new_width;
684
  gimage->height = new_height;
685 686
  img_scale_w    = (gdouble) new_width  / (gdouble) old_width;
  img_scale_h    = (gdouble) new_height / (gdouble) old_height;
687
 
688
  /*  Scale all channels  */
689
  for (list = gimage->channels; list; list = g_slist_next (list))
690 691 692 693 694 695
    {
      channel = (Channel *) list->data;
      channel_scale (channel, new_width, new_height);
    }

  /*  Don't forget the selection mask!  */
696 697 698 699 700
  /*  if (channel_is_empty(gimage->selection_mask))
        channel_resize(gimage->selection_mask, new_width, new_height, 0, 0)
      else
  */
        
701 702 703 704 705
  channel_scale (gimage->selection_mask, new_width, new_height);
  gimage_mask_invalidate (gimage);

  /*  Scale all layers  */
  list = gimage->layers;
706
  for (list = gimage->layers; list; list = g_slist_next (list))
707
    {
708 709
      layer = (Layer *) list->data;
      if (layer_scale_by_factors (layer, img_scale_w, img_scale_h) == FALSE)
710
	{
711 712 713 714 715
	  /* Since 0 < img_scale_w, img_scale_h, failure due to one or more
	   * vanishing scaled layer dimensions. Implicit delete implemented
	   * here. Upstream warning implemented in resize_check_layer_scaling()
	   * [resize.c line 1295], which offers the user the chance to bail out.
	   */
716

717
          remove = g_slist_append (remove, layer);
718 719 720 721 722
        }
    }
  /* We defer removing layers lost to scaling until now            */
  /* so as not to mix the operations of iterating over and removal */
  /* from gimage->layers.                                          */  
723

724
  for (list = remove; list; list = g_slist_next (list))
725
    {
726
      layer = list->data;
727
      gimp_image_remove_layer (gimage, layer);
728
    }
729
  g_slist_free (remove);
730

731
  /*  Scale any Guides  */
732
  for (glist = gimage->guides; glist; glist = g_list_next (glist))
733 734 735 736 737
    {
      guide = (Guide*) glist->data;

      switch (guide->orientation)
	{
738
	case ORIENTATION_HORIZONTAL:
Sven Neumann's avatar
Sven Neumann committed
739
	  undo_push_guide (gimage, guide);
740 741
	  guide->position = (guide->position * new_height) / old_height;
	  break;
742
	case ORIENTATION_VERTICAL:
Sven Neumann's avatar
Sven Neumann committed
743
	  undo_push_guide (gimage, guide);
744 745 746 747 748 749
	  guide->position = (guide->position * new_width) / old_width;
	  break;
	default:
	  g_error("Unknown guide orientation II.\n");
	}
    }
750

751 752 753 754 755 756 757 758
  /*  Make sure the projection matches the gimage size  */
  gimp_image_projection_realloc (gimage);

  /*  Rigor the floating selection  */
  if (floating_layer)
    floating_sel_rigor (floating_layer, TRUE);

  gtk_signal_emit (GTK_OBJECT (gimage), gimp_image_signals[RESIZE]);
759

760
  gimp_remove_busy_cursors (NULL);
761 762 763
}

TileManager *
764
gimp_image_shadow (GimpImage *gimage, 
765 766 767
		   gint       width, 
		   gint       height, 
		   gint       bpp)
768 769
{
  if (gimage->shadow &&
770 771 772
      ((width != tile_manager_width (gimage->shadow)) ||
       (height != tile_manager_height (gimage->shadow)) ||
       (bpp != tile_manager_bpp (gimage->shadow))))
773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794
    gimp_image_free_shadow (gimage);
  else if (gimage->shadow)
    return gimage->shadow;

  gimp_image_allocate_shadow (gimage, width, height, bpp);

  return gimage->shadow;
}

void
gimp_image_free_shadow (GimpImage *gimage)
{
  /*  Free the shadow buffer from the specified gimage if it exists  */
  if (gimage->shadow)
    tile_manager_destroy (gimage->shadow);

  gimage->shadow = NULL;
}

static void
gimp_image_destroy (GtkObject *object)
{
795
  GimpImage* gimage = GIMP_IMAGE (object);
Sven Neumann's avatar
Sven Neumann committed
796 797 798 799 800 801

  gimp_image_free_projection (gimage);
  gimp_image_free_shadow (gimage);
  
  if (gimage->cmap)
    g_free (gimage->cmap);
802

Sven Neumann's avatar
Sven Neumann committed
803 804
  gimp_image_free_layers (gimage);
  gimp_image_free_channels (gimage);
805 806

  gtk_object_unref (GTK_OBJECT (gimage->selection_mask));
807

BST 1999 Andy Thomas's avatar
BST 1999 Andy Thomas committed
808 809 810
  if (gimage->comp_preview)
    temp_buf_free (gimage->comp_preview);

Sven Neumann's avatar
Sven Neumann committed
811
  if (gimage->parasites)
812
    gtk_object_unref (GTK_OBJECT (gimage->parasites));
813 814 815
}

void
816 817 818 819 820 821
gimp_image_apply_image (GimpImage	 *gimage,
			GimpDrawable	 *drawable,
			PixelRegion	 *src2PR,
			gint              undo,
			gint              opacity,
			LayerModeEffects  mode,
822
			/*  alternative to using drawable tiles as src1: */
823 824 825
			TileManager	 *src1_tiles,
			gint              x,
			gint              y)
826
{
827 828 829 830 831 832
  Channel     *mask;
  gint         x1, y1, x2, y2;
  gint         offset_x, offset_y;
  PixelRegion  src1PR, destPR, maskPR;
  gint         operation;
  gint         active [MAX_CHANNELS];
833 834

  /*  get the selection mask if one exists  */
835
  mask = (gimage_mask_is_empty (gimage)) ? NULL : gimp_image_get_mask (gimage);
836 837 838 839 840 841 842

  /*  configure the active channel array  */
  gimp_image_get_active_channels (gimage, drawable, active);

  /*  determine what sort of operation is being attempted and
   *  if it's actually legal...
   */
843
  operation = valid_combinations[gimp_drawable_type (drawable)][src2PR->bytes];
844 845
  if (operation == -1)
    {
Sven Neumann's avatar
Sven Neumann committed
846
      g_message ("gimp_image_apply_image sent illegal parameters");
847 848 849 850
      return;
    }

  /*  get the layer offsets  */
851
  gimp_drawable_offsets (drawable, &offset_x, &offset_y);
852 853

  /*  make sure the image application coordinates are within gimage bounds  */
854 855 856 857
  x1 = CLAMP (x, 0, gimp_drawable_width  (drawable));
  y1 = CLAMP (y, 0, gimp_drawable_height (drawable));
  x2 = CLAMP (x + src2PR->w, 0, gimp_drawable_width  (drawable));
  y2 = CLAMP (y + src2PR->h, 0, gimp_drawable_height (drawable));
858 859 860 861 862 863 864

  if (mask)
    {
      /*  make sure coordinates are in mask bounds ...
       *  we need to add the layer offset to transform coords
       *  into the mask coordinate system
       */
865 866 867 868
      x1 = CLAMP (x1, -offset_x, gimp_drawable_width (GIMP_DRAWABLE (mask))-offset_x);
      y1 = CLAMP (y1, -offset_y, gimp_drawable_height(GIMP_DRAWABLE (mask))-offset_y);
      x2 = CLAMP (x2, -offset_x, gimp_drawable_width (GIMP_DRAWABLE (mask))-offset_x);
      y2 = CLAMP (y2, -offset_y, gimp_drawable_height(GIMP_DRAWABLE (mask))-offset_y);
869 870 871 872
    }

  /*  If the calling procedure specified an undo step...  */
  if (undo)
873
    undo_push_image (gimp_drawable_gimage (drawable), drawable, x1, y1, x2, y2);
874 875 876 877 878

  /* configure the pixel regions
   *  If an alternative to using the drawable's data as src1 was provided...
   */
  if (src1_tiles)
879 880
    pixel_region_init (&src1PR, src1_tiles, 
		       x1, y1, (x2 - x1), (y2 - y1), FALSE);
881
  else
882
    pixel_region_init (&src1PR, gimp_drawable_data (drawable), 
883
		       x1, y1, (x2 - x1), (y2 - y1), FALSE);
884
  pixel_region_init (&destPR, gimp_drawable_data (drawable), 
885 886 887 888
		     x1, y1, (x2 - x1), (y2 - y1), TRUE);
  pixel_region_resize (src2PR, 
		       src2PR->x + (x1 - x), src2PR->y + (y1 - y), 
		       (x2 - x1), (y2 - y1));
889 890 891

  if (mask)
    {
892
      gint mx, my;
893 894 895 896 897 898 899 900

      /*  configure the mask pixel region
       *  don't use x1 and y1 because they are in layer
       *  coordinate system.  Need mask coordinate system
       */
      mx = x1 + offset_x;
      my = y1 + offset_y;

901
      pixel_region_init (&maskPR, 
902
			 gimp_drawable_data (GIMP_DRAWABLE (mask)), 
903 904 905
			 mx, my, 
			 (x2 - x1), (y2 - y1), 
			 FALSE);
906 907 908 909
      combine_regions (&src1PR, src2PR, &destPR, &maskPR, NULL,
		       opacity, mode, active, operation);
    }
  else
910 911 912 913
    {
      combine_regions (&src1PR, src2PR, &destPR, NULL, NULL,
		       opacity, mode, active, operation);
    }
914 915 916 917 918 919 920 921 922
}

/* Similar to gimp_image_apply_image but works in "replace" mode (i.e.
   transparent pixels in src2 make the result transparent rather
   than opaque.

   Takes an additional mask pixel region as well.
*/
void
923 924 925
gimp_image_replace_image (GimpImage    *gimage, 
			  GimpDrawable *drawable, 
			  PixelRegion  *src2PR,
926 927
			  gint          undo, 
			  gint          opacity,
928
			  PixelRegion  *maskPR,
929 930
			  gint          x, 
			  gint          y)
931
{
932 933 934 935 936 937 938 939
  Channel     *mask;
  gint         x1, y1, x2, y2;
  gint         offset_x, offset_y;
  PixelRegion  src1PR, destPR;
  PixelRegion  mask2PR, tempPR;
  guchar      *temp_data;
  gint         operation;
  gint         active [MAX_CHANNELS];
940 941

  /*  get the selection mask if one exists  */
942
  mask = (gimage_mask_is_empty (gimage)) ? NULL : gimp_image_get_mask (gimage);
943 944 945 946 947 948 949

  /*  configure the active channel array  */
  gimp_image_get_active_channels (gimage, drawable, active);

  /*  determine what sort of operation is being attempted and
   *  if it's actually legal...
   */
950
  operation = valid_combinations [gimp_drawable_type (drawable)][src2PR->bytes];
951 952
  if (operation == -1)
    {
Sven Neumann's avatar
Sven Neumann committed
953
      g_message ("gimp_image_apply_image sent illegal parameters");
954 955 956 957
      return;
    }

  /*  get the layer offsets  */
958
  gimp_drawable_offsets (drawable, &offset_x, &offset_y);
959 960

  /*  make sure the image application coordinates are within gimage bounds  */
961 962 963 964
  x1 = CLAMP (x, 0, gimp_drawable_width (drawable));
  y1 = CLAMP (y, 0, gimp_drawable_height (drawable));
  x2 = CLAMP (x + src2PR->w, 0, gimp_drawable_width (drawable));
  y2 = CLAMP (y + src2PR->h, 0, gimp_drawable_height (drawable));
965 966 967 968 969 970 971

  if (mask)
    {
      /*  make sure coordinates are in mask bounds ...
       *  we need to add the layer offset to transform coords
       *  into the mask coordinate system
       */
972 973 974 975
      x1 = CLAMP (x1, -offset_x, gimp_drawable_width (GIMP_DRAWABLE (mask))-offset_x);
      y1 = CLAMP (y1, -offset_y, gimp_drawable_height(GIMP_DRAWABLE (mask))-offset_y);
      x2 = CLAMP (x2, -offset_x, gimp_drawable_width (GIMP_DRAWABLE (mask))-offset_x);
      y2 = CLAMP (y2, -offset_y, gimp_drawable_height(GIMP_DRAWABLE (mask))-offset_y);
976 977 978 979 980 981 982 983 984
    }

  /*  If the calling procedure specified an undo step...  */
  if (undo)
    drawable_apply_image (drawable, x1, y1, x2, y2, NULL, FALSE);

  /* configure the pixel regions
   *  If an alternative to using the drawable's data as src1 was provided...
   */
985
  pixel_region_init (&src1PR, gimp_drawable_data (drawable),
986
		     x1, y1, (x2 - x1), (y2 - y1), FALSE);
987
  pixel_region_init (&destPR, gimp_drawable_data (drawable),
988 989 990 991
		     x1, y1, (x2 - x1), (y2 - y1), TRUE);
  pixel_region_resize (src2PR,
		       src2PR->x + (x1 - x), src2PR->y + (y1 - y),
		       (x2 - x1), (y2 - y1));
992 993 994 995 996 997 998 999 1000 1001 1002 1003

  if (mask)
    {
      int mx, my;

      /*  configure the mask pixel region
       *  don't use x1 and y1 because they are in layer
       *  coordinate system.  Need mask coordinate system
       */
      mx = x1 + offset_x;
      my = y1 + offset_y;

Sven Neumann's avatar