gimpimage-resize.c 109 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
#include <gtk/gtk.h>

25
#include "libgimpcolor/gimpcolor.h"
26
#include "libgimpmath/gimpmath.h"
27

Sven Neumann's avatar
Sven Neumann committed
28 29
#include "apptypes.h"

30 31
#include "paint-funcs/paint-funcs.h"

32
#include "cursorutil.h"
33 34
#include "drawable.h"
#include "floating_sel.h"
35
#include "gdisplay.h"
36
#include "gimage_mask.h"
37
#include "gimpcontext.h"
38
#include "gimpimage.h"
39
#include "gimpimage-undo.h"
40
#include "gimplayer.h"
41
#include "gimplayermask.h"
42
#include "gimplist.h"
43
#include "gimpmarshal.h"
44
#include "gimprc.h"
Manish Singh's avatar
Manish Singh committed
45
#include "gimpparasite.h"
46
#include "gimpundostack.h"
47
#include "parasitelist.h"
48
#include "path.h"
49 50
#include "pixel_region.h"
#include "temp_buf.h"
scott's avatar
scott committed
51 52
#include "tile_manager.h"
#include "tile.h"
53
#include "undo.h"
54

55
#include "libgimp/gimplimits.h"
56
#include "libgimp/gimpparasite.h"
57 58 59

#include "libgimp/gimpintl.h"

60

61 62 63 64 65 66 67
#ifdef DEBUG
#define TRC(x) printf x
#else
#define TRC(x)
#endif


68
/*  Local function declarations  */
69 70 71 72 73
static void     gimp_image_class_init            (GimpImageClass *klass);
static void     gimp_image_init                  (GimpImage      *gimage);
static void     gimp_image_destroy               (GtkObject      *object);
static void     gimp_image_name_changed          (GimpObject     *object);
static void     gimp_image_invalidate_preview    (GimpViewable   *viewable);
74
static TempBuf *gimp_image_get_preview           (GimpViewable   *gimage,
75 76
						  gint            width,
						  gint            height);
77
static TempBuf *gimp_image_get_new_preview       (GimpViewable   *viewable,
78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
						  gint            width, 
						  gint            height);
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_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);
104 105

/*  projection functions  */
106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
static void     project_intensity                (GimpImage      *gimage,
						  GimpLayer      *layer,
						  PixelRegion    *src,
						  PixelRegion    *dest,
						  PixelRegion    *mask);
static void     project_intensity_alpha          (GimpImage      *gimage,
						  GimpLayer      *layer,
						  PixelRegion    *src,
						  PixelRegion    *dest,
						  PixelRegion    *mask);
static void     project_indexed                  (GimpImage      *gimage,
						  GimpLayer      *layer,
						  PixelRegion    *src,
						  PixelRegion    *dest);
static void     project_indexed_alpha            (GimpImage      *gimage, 
						  GimpLayer      *layer,
						  PixelRegion    *src, 
						  PixelRegion    *dest,
						  PixelRegion    *mask);
static void     project_channel                  (GimpImage      *gimage,
						  GimpChannel    *channel,
						  PixelRegion    *src,
						  PixelRegion    *src2);
129 130 131 132

/*
 *  Global variables
 */
133
gint valid_combinations[][MAX_CHANNELS + 1] =
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148
{
  /* 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 },
};

149
guint32 next_guide_id = 1;  /* For generating guide_ID handles for PDB stuff */
150

151 152 153 154 155

/*
 *  Static variables
 */

156 157
enum
{
158
  MODE_CHANGED,
159
  SIZE_CHANGED,
160 161
  ACTIVE_LAYER_CHANGED,
  ACTIVE_CHANNEL_CHANGED,
162 163
  COMPONENT_VISIBILITY_CHANGED,
  COMPONENT_ACTIVE_CHANGED,
164
  CLEAN,
165 166 167
  DIRTY,
  REPAINT,
  RESTRUCTURE,
168
  COLORMAP_CHANGED,
169
  UNDO_EVENT,
170 171
  LAST_SIGNAL
};
172

173 174
static guint gimp_image_signals[LAST_SIGNAL] = { 0 };

175
static GimpViewableClass *parent_class = NULL;
176

177 178 179
static gint        global_image_ID  = 1;
static GHashTable *gimp_image_table = NULL;

180

181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205
GtkType 
gimp_image_get_type (void) 
{
  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_VIEWABLE, &image_info);
    }

  return image_type;
}

206 207 208
static void
gimp_image_class_init (GimpImageClass *klass)
{
209 210 211
  GtkObjectClass    *object_class;
  GimpObjectClass   *gimp_object_class;
  GimpViewableClass *viewable_class;
212

213 214
  object_class      = (GtkObjectClass *) klass;
  gimp_object_class = (GimpObjectClass *) klass;
215
  viewable_class    = (GimpViewableClass *) klass;
216

217
  parent_class = gtk_type_class (GIMP_TYPE_VIEWABLE);
218

219 220 221 222 223 224 225 226 227
  gimp_image_signals[MODE_CHANGED] =
    gtk_signal_new ("mode_changed",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       mode_changed),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

228 229 230 231 232 233 234 235 236
  gimp_image_signals[SIZE_CHANGED] =
    gtk_signal_new ("size_changed",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       size_changed),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254
  gimp_image_signals[ACTIVE_LAYER_CHANGED] =
    gtk_signal_new ("active_layer_changed",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       active_layer_changed),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

  gimp_image_signals[ACTIVE_CHANNEL_CHANGED] =
    gtk_signal_new ("active_channel_changed",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       active_channel_changed),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274
  gimp_image_signals[COMPONENT_VISIBILITY_CHANGED] =
    gtk_signal_new ("component_visibility_changed",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       component_visibility_changed),
                    gtk_marshal_NONE__INT,
                    GTK_TYPE_NONE, 1,
		    GTK_TYPE_INT);

  gimp_image_signals[COMPONENT_ACTIVE_CHANGED] =
    gtk_signal_new ("component_active_changed",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       component_active_changed),
                    gtk_marshal_NONE__INT,
                    GTK_TYPE_NONE, 1,
		    GTK_TYPE_INT);

275
  gimp_image_signals[CLEAN] =
276 277 278 279 280 281 282 283
    gtk_signal_new ("clean",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       clean),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

284
  gimp_image_signals[DIRTY] =
285 286 287 288 289 290 291 292
    gtk_signal_new ("dirty",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       dirty),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

293
  gimp_image_signals[REPAINT] =
294 295 296 297 298 299 300 301 302 303 304 305
    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);

306
  gimp_image_signals[RESTRUCTURE] =
307 308 309 310 311 312 313 314
    gtk_signal_new ("restructure",
                    GTK_RUN_FIRST,
                    object_class->type,
                    GTK_SIGNAL_OFFSET (GimpImageClass,
				       restructure),
                    gtk_signal_default_marshaller,
                    GTK_TYPE_NONE, 0);

315
  gimp_image_signals[COLORMAP_CHANGED] =
316 317 318 319 320 321 322 323 324
    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);

325
  gimp_image_signals[UNDO_EVENT] = 
326 327 328 329 330 331 332 333
    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);
334

335
  gtk_object_class_add_signals (object_class, gimp_image_signals, LAST_SIGNAL);
336

337
  object_class->destroy               = gimp_image_destroy;
338

339
  gimp_object_class->name_changed     = gimp_image_name_changed;
340

341 342 343
  viewable_class->invalidate_preview  = gimp_image_invalidate_preview;
  viewable_class->get_preview         = gimp_image_get_preview;
  viewable_class->get_new_preview     = gimp_image_get_new_preview;
344

345 346
  klass->mode_changed                 = NULL;
  klass->size_changed                 = NULL;
347 348 349 350
  klass->active_layer_changed         = NULL;
  klass->active_channel_changed       = NULL;
  klass->component_visibility_changed = NULL;
  klass->component_active_changed     = NULL;
351

352 353 354 355 356 357 358 359
  klass->clean                        = NULL;
  klass->dirty                        = NULL;
  klass->repaint                      = NULL;
  klass->restructure                  = NULL;
  klass->colormap_changed             = NULL;
  klass->undo_event                   = NULL;
  klass->undo                         = gimp_image_undo;
  klass->redo                         = gimp_image_redo;
360 361 362 363 364
}


/* static functions */

Sven Neumann's avatar
Sven Neumann committed
365 366
static void 
gimp_image_init (GimpImage *gimage)
367
{
368 369
  gimage->ID                    = global_image_ID++;

370 371 372 373 374 375 376 377 378
  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;

379
  gimage->cmap                  = NULL;
380 381
  gimage->num_cols              = 0;

382
  gimage->dirty                 = 1;
383
  gimage->undo_on               = TRUE;
384 385 386 387

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

388
  gimage->tattoo_state          = 0;
389 390 391 392 393

  gimage->shadow                = NULL;

  gimage->construct_flag        = -1;
  gimage->proj_type             = RGBA_GIMAGE;
394
  gimage->projection            = NULL;
395

396
  gimage->guides                = NULL;
397

398 399 400 401
  gimage->layers                = gimp_list_new (GIMP_TYPE_LAYER, 
						 GIMP_CONTAINER_POLICY_STRONG);
  gimage->channels              = gimp_list_new (GIMP_TYPE_CHANNEL, 
						 GIMP_CONTAINER_POLICY_STRONG);
402
  gimage->layer_stack           = NULL;
403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420

  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;

421 422 423 424
  gimage->undo_stack            = NULL;
  gimage->redo_stack            = NULL;
  gimage->undo_bytes            = 0;
  gimage->undo_levels           = 0;
425
  gimage->group_count           = 0;
426
  gimage->pushing_undo_group    = UNDO_NULL;
427 428
  gimage->undo_history          = NULL;

429 430 431
  gimage->new_undo_stack        = gimp_undo_stack_new (gimage);
  gimage->new_redo_stack        = gimp_undo_stack_new (gimage);

432
  gimage->comp_preview          = NULL;
433
  gimage->comp_preview_valid    = FALSE;
434 435 436 437 438 439 440

  if (gimp_image_table == NULL)
    gimp_image_table = g_hash_table_new (g_direct_hash, NULL);

  g_hash_table_insert (gimp_image_table,
		       GINT_TO_POINTER (gimage->ID),
		       (gpointer) gimage);
441 442
}

443 444
static void
gimp_image_destroy (GtkObject *object)
Sven Neumann's avatar
Sven Neumann committed
445
{
446 447 448 449 450
  GimpImage *gimage;

  gimage = GIMP_IMAGE (object);

  g_hash_table_remove (gimp_image_table, GINT_TO_POINTER (gimage->ID));
451

452 453 454 455 456
  gimp_image_free_projection (gimage);
  gimp_image_free_shadow (gimage);
  
  if (gimage->cmap)
    g_free (gimage->cmap);
457 458 459 460
  
  gtk_object_unref (GTK_OBJECT (gimage->layers));
  gtk_object_unref (GTK_OBJECT (gimage->channels));
  g_slist_free (gimage->layer_stack);
461 462

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

464 465 466 467 468
  if (gimage->comp_preview)
    temp_buf_free (gimage->comp_preview);

  if (gimage->parasites)
    gtk_object_unref (GTK_OBJECT (gimage->parasites));
469 470 471

  gtk_object_unref (GTK_OBJECT (gimage->new_undo_stack));
  gtk_object_unref (GTK_OBJECT (gimage->new_redo_stack));
472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489
}

static void
gimp_image_name_changed (GimpObject *object)
{
  GimpImage   *gimage;
  const gchar *name;

  if (GIMP_OBJECT_CLASS (parent_class)->name_changed)
    GIMP_OBJECT_CLASS (parent_class)->name_changed (object);

  gimage = GIMP_IMAGE (object);
  name   = gimp_object_get_name (object);

  if (! (name && strlen (name)))
    {
      g_free (object->name);
      object->name = NULL;
490
    }
491
}
492

493 494 495 496 497 498
static void
gimp_image_invalidate_preview (GimpViewable *viewable)
{
  GimpImage *gimage;
  GimpLayer *layer;

499 500 501
  if (GIMP_VIEWABLE_CLASS (parent_class)->invalidate_preview)
    GIMP_VIEWABLE_CLASS (parent_class)->invalidate_preview (viewable);

502 503 504 505 506 507 508
  gimage = GIMP_IMAGE (viewable);

  /*  Invalidate the floating sel if it exists  */
  if ((layer = gimp_image_floating_sel (gimage)))
    floating_sel_invalidate (layer);

  gimage->comp_preview_valid = FALSE;
509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532
}

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:
533
      g_assert_not_reached ();
534 535 536
    }

  /*  allocate the new projection  */
537 538
  gimage->projection = tile_manager_new (gimage->width, gimage->height,
					 gimage->proj_bytes);
539
  tile_manager_set_user_data (gimage->projection, (void *) gimage);
540 541 542 543 544 545 546 547 548 549 550 551 552
  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
553
gimp_image_allocate_shadow (GimpImage *gimage, 
554 555 556
			    gint       width, 
			    gint       height, 
			    gint       bpp)
557 558 559 560 561 562 563 564 565
{
  /*  allocate the new projection  */
  gimage->shadow = tile_manager_new (width, height, bpp);
}


/* function definitions */

GimpImage *
566 567 568
gimp_image_new (gint               width,
		gint               height,
		GimpImageBaseType  base_type)
569
{
570
  GimpImage *gimage = GIMP_IMAGE (gtk_type_new (GIMP_TYPE_IMAGE));
571
  gint i;
572

573 574
  gimage->width     = width;
  gimage->height    = height;
575 576 577 578 579 580 581 582 583 584
  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;
585
      gimage->cmap     = (guchar *) g_malloc0 (COLORMAP_SIZE);
586 587 588 589 590 591 592 593
      break;
    default:
      break;
    }

  /*  set all color channels visible and active  */
  for (i = 0; i < MAX_CHANNELS; i++)
    {
594 595
      gimage->visible[i] = TRUE;
      gimage->active[i]  = TRUE;
596 597 598
    }

  /* create the selection mask */
Michael Natterer's avatar
Michael Natterer committed
599 600 601
  gimage->selection_mask = gimp_channel_new_mask (gimage,
						  gimage->width,
						  gimage->height);
602 603 604 605 606


  return gimage;
}

607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625
gint
gimp_image_get_ID (GimpImage *gimage)
{
  g_return_val_if_fail (gimage != NULL, -1);
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), -1);

  return gimage->ID;
}

GimpImage *
gimp_image_get_by_ID (gint image_id)
{
  if (gimp_image_table == NULL)
    return NULL;

  return (GimpImage *) g_hash_table_lookup (gimp_image_table, 
					    GINT_TO_POINTER (image_id));
}

626
void
627 628
gimp_image_set_filename (GimpImage   *gimage,
			 const gchar *filename)
629
{
630 631
  gimp_object_set_name (GIMP_OBJECT (gimage), filename);
}
632

633
void
634
gimp_image_set_resolution (GimpImage *gimage,
635 636
			   gdouble    xresolution,
			   gdouble    yresolution)
637
{
638 639 640 641 642
  /* 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;

643 644 645
  /* 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)
646 647
    return;

648
  undo_push_resolution (gimage);
649

650 651
  gimage->xresolution = xresolution;
  gimage->yresolution = yresolution;
652 653 654

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

657
void
658 659 660
gimp_image_get_resolution (const GimpImage *gimage,
			   gdouble         *xresolution,
			   gdouble         *yresolution)
661
{
662 663
  g_return_if_fail (xresolution && yresolution);

664 665
  *xresolution = gimage->xresolution;
  *yresolution = gimage->yresolution;
666 667
}

668 669
void
gimp_image_set_unit (GimpImage *gimage,
670
		     GimpUnit   unit)
671
{
672 673
  undo_push_resolution (gimage);

674 675 676
  gimage->unit = unit;
}

677
GimpUnit
678
gimp_image_get_unit (const GimpImage *gimage)
679 680 681 682
{
  return gimage->unit;
}

683
void
Sven Neumann's avatar
Sven Neumann committed
684 685
gimp_image_set_save_proc (GimpImage     *gimage, 
			  PlugInProcDef *proc)
686 687 688 689 690
{
  gimage->save_proc = proc;
}

PlugInProcDef *
691
gimp_image_get_save_proc (const GimpImage *gimage)
692 693 694
{
  return gimage->save_proc;
}
695

696 697 698 699 700 701 702 703 704 705 706 707
gint
gimp_image_get_width (const GimpImage *gimage)
{
  return gimage->width;
}

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

708
void
709
gimp_image_resize (GimpImage *gimage, 
710 711 712 713
		   gint       new_width, 
		   gint       new_height,
		   gint       offset_x, 
		   gint       offset_y)
714
{
Michael Natterer's avatar
Michael Natterer committed
715 716 717
  GimpChannel *channel;
  GimpLayer   *layer;
  GimpLayer   *floating_layer;
718
  GList       *list;
Michael Natterer's avatar
Michael Natterer committed
719
  GList       *guide_list;
720

721
  gimp_add_busy_cursors ();
722

723
  g_assert (new_width > 0 && new_height > 0);
724 725 726 727

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

Sven Neumann's avatar
Sven Neumann committed
728
  undo_push_group_start (gimage, IMAGE_RESIZE_UNDO);
729 730 731 732 733 734 735 736 737

  /*  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  */
738
  gimage->width  = new_width;
739 740 741
  gimage->height = new_height;

  /*  Resize all channels  */
742 743 744
  for (list = GIMP_LIST (gimage->channels)->list; 
       list; 
       list = g_list_next (list))
745
    {
Michael Natterer's avatar
Michael Natterer committed
746
      channel = (GimpChannel *) list->data;
747

Michael Natterer's avatar
Michael Natterer committed
748
      gimp_channel_resize (channel, new_width, new_height, offset_x, offset_y);
749 750 751 752 753 754 755 756 757
    }

  /*  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
758
      guide_list = g_list_next (guide_list);
759 760 761

      switch (guide->orientation)
	{
762
	case ORIENTATION_HORIZONTAL:
Sven Neumann's avatar
Sven Neumann committed
763
	  undo_push_guide (gimage, guide);
764 765 766 767
	  guide->position += offset_y;
	  if (guide->position < 0 || guide->position > new_height)
	    gimp_image_delete_guide (gimage, guide);
	  break;
768

769
	case ORIENTATION_VERTICAL:
Sven Neumann's avatar
Sven Neumann committed
770
	  undo_push_guide (gimage, guide);
771 772 773 774
	  guide->position += offset_x;
	  if (guide->position < 0 || guide->position > new_width)
	    gimp_image_delete_guide (gimage, guide);
	  break;
775

776
	default:
777
	  g_error ("Unknown guide orientation\n");
778
	}
779 780 781
    }

  /*  Don't forget the selection mask!  */
Michael Natterer's avatar
Michael Natterer committed
782 783
  gimp_channel_resize (gimage->selection_mask,
		       new_width, new_height, offset_x, offset_y);
784 785 786
  gimage_mask_invalidate (gimage);

  /*  Reposition all layers  */
787 788 789
  for (list = GIMP_LIST (gimage->layers)->list; 
       list; 
       list = g_list_next (list))
790
    {
791
      layer = (GimpLayer *) list->data;
792

793
      gimp_layer_translate (layer, offset_x, offset_y);
794 795 796 797 798 799 800 801 802
    }

  /*  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);

803 804 805
  undo_push_group_end (gimage);

  gimp_image_size_changed (gimage);
806

807
  gimp_remove_busy_cursors (NULL);
808 809 810
}

void
811
gimp_image_scale (GimpImage *gimage, 
812 813
		  gint       new_width, 
		  gint       new_height)
814
{
Michael Natterer's avatar
Michael Natterer committed
815 816 817
  GimpChannel *channel;
  GimpLayer   *layer;
  GimpLayer   *floating_layer;
818
  GList       *list;
Michael Natterer's avatar
Michael Natterer committed
819
  GSList      *remove = NULL;
820
  GSList      *slist;
Michael Natterer's avatar
Michael Natterer committed
821 822 823 824 825
  Guide       *guide;
  gint         old_width;
  gint         old_height;
  gdouble      img_scale_w = 1.0;
  gdouble      img_scale_h = 1.0;
826

827
  if ((new_width == 0) || (new_height == 0))
828
    {
829 830
      g_message ("%s(): Scaling to zero width or height has been rejected.",
		 G_GNUC_FUNCTION);
831 832
      return;
    }
833

834
  gimp_add_busy_cursors ();
835

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

Sven Neumann's avatar
Sven Neumann committed
839
  undo_push_group_start (gimage, IMAGE_SCALE_UNDO);
840 841 842 843 844 845 846 847 848

  /*  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  */
849 850 851

  old_width      = gimage->width;
  old_height     = gimage->height;
852
  gimage->width  = new_width;
853
  gimage->height = new_height;
854 855
  img_scale_w    = (gdouble) new_width  / (gdouble) old_width;
  img_scale_h    = (gdouble) new_height / (gdouble) old_height;
856
 
857
  /*  Scale all channels  */
858 859 860
  for (list = GIMP_LIST (gimage->channels)->list; 
       list; 
       list = g_list_next (list))
861
    {
Michael Natterer's avatar
Michael Natterer committed
862 863 864
      channel = (GimpChannel *) list->data;

      gimp_channel_scale (channel, new_width, new_height);
865 866 867
    }

  /*  Don't forget the selection mask!  */
868
  /*  if (channel_is_empty(gimage->selection_mask))
Michael Natterer's avatar
Michael Natterer committed
869
        gimp_channel_resize(gimage->selection_mask, new_width, new_height, 0, 0)
870 871 872
      else
  */
        
Michael Natterer's avatar
Michael Natterer committed
873
  gimp_channel_scale (gimage->selection_mask, new_width, new_height);
874 875 876
  gimage_mask_invalidate (gimage);

  /*  Scale all layers  */
877 878 879
  for (list = GIMP_LIST (gimage->layers)->list; 
       list; 
       list = g_list_next (list))
880
    {
881 882 883
      layer = (GimpLayer *) list->data;

      if (gimp_layer_scale_by_factors (layer, img_scale_w, img_scale_h) == FALSE)
884
	{
885 886 887 888 889
	  /* 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.
	   */
890

891
          remove = g_slist_append (remove, layer);
892 893 894 895 896
        }
    }
  /* We defer removing layers lost to scaling until now            */
  /* so as not to mix the operations of iterating over and removal */
  /* from gimage->layers.                                          */  
897

898
  for (slist = remove; slist; slist = g_slist_next (slist))
899
    {
900
      layer = slist->data;
901
      gimp_image_remove_layer (gimage, layer);
902
    }
903
  g_slist_free (remove);
904

905
  /*  Scale any Guides  */
906
  for (list = gimage->guides; list; list = g_list_next (list))
907
    {
908
      guide = (Guide*) list->data;
909 910 911

      switch (guide->orientation)
	{
912
	case ORIENTATION_HORIZONTAL:
Sven Neumann's avatar
Sven Neumann committed
913
	  undo_push_guide (gimage, guide);
914 915
	  guide->position = (guide->position * new_height) / old_height;
	  break;
916
	case ORIENTATION_VERTICAL:
Sven Neumann's avatar
Sven Neumann committed
917
	  undo_push_guide (gimage, guide);
918 919 920 921 922 923
	  guide->position = (guide->position * new_width) / old_width;
	  break;
	default:
	  g_error("Unknown guide orientation II.\n");
	}
    }
924

925 926 927 928 929 930 931
  /*  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);

932 933 934
  undo_push_group_end (gimage);

  gimp_image_size_changed (gimage);
935

936
  gimp_remove_busy_cursors (NULL);
937 938 939
}

TileManager *
940
gimp_image_shadow (GimpImage *gimage, 
941 942 943
		   gint       width, 
		   gint       height, 
		   gint       bpp)
944 945
{
  if (gimage->shadow &&
946 947 948
      ((width != tile_manager_width (gimage->shadow)) ||
       (height != tile_manager_height (gimage->shadow)) ||
       (bpp != tile_manager_bpp (gimage->shadow))))
949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968
    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;
}

void
969 970 971 972 973 974
gimp_image_apply_image (GimpImage	 *gimage,
			GimpDrawable	 *drawable,
			PixelRegion	 *src2PR,
			gint              undo,
			gint              opacity,
			LayerModeEffects  mode,
975
			/*  alternative to using drawable tiles as src1: */
976 977 978
			TileManager	 *src1_tiles,
			gint              x,
			gint              y)
979
{
Michael Natterer's avatar
Michael Natterer committed
980
  GimpChannel *mask;
981 982 983 984 985
  gint         x1, y1, x2, y2;
  gint         offset_x, offset_y;
  PixelRegion  src1PR, destPR, maskPR;
  gint         operation;
  gint         active [MAX_CHANNELS];
986 987

  /*  get the selection mask if one exists  */
988
  mask = (gimage_mask_is_empty (gimage)) ? NULL : gimp_image_get_mask (gimage);
989 990 991 992 993 994 995

  /*  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...
   */
996
  operation = valid_combinations[gimp_drawable_type (drawable)][src2PR->bytes];
997 998
  if (operation == -1)
    {
999
      g_message ("%s(): illegal parameters", G_GNUC_FUNCTION);
1000 1001 1002 1003
      return;
    }

  /*  get the layer offsets  */
1004
  gimp_drawable_offsets (drawable, &offset_x, &offset_y);
1005 1006

  /*  make sure the image application coordinates are within gimage bounds  */
1007