gimpimage-resize.c 97 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

23
#include <glib-object.h>
Sven Neumann's avatar
Sven Neumann committed
24

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

Michael Natterer's avatar
Michael Natterer committed
29 30
#include "core-types.h"

Michael Natterer's avatar
Michael Natterer committed
31 32 33
#include "base/pixel-region.h"
#include "base/temp-buf.h"
#include "base/tile-manager.h"
Sven Neumann's avatar
Sven Neumann committed
34

35 36
#include "paint-funcs/paint-funcs.h"

37
#include "gimp.h"
38
#include "gimpcontext.h"
Michael Natterer's avatar
Michael Natterer committed
39
#include "gimpcoreconfig.h"
40
#include "gimpimage.h"
Michael Natterer's avatar
Michael Natterer committed
41
#include "gimpimage-colorhash.h"
42
#include "gimpimage-mask.h"
43
#include "gimpimage-projection.h"
44
#include "gimpimage-undo.h"
45
#include "gimplayer.h"
46
#include "gimplayermask.h"
47
#include "gimplist.h"
48
#include "gimpmarshal.h"
Manish Singh's avatar
Manish Singh committed
49
#include "gimpparasite.h"
50
#include "gimpparasitelist.h"
51
#include "gimpundostack.h"
52 53

#include "floating_sel.h"
54
#include "path.h"
55
#include "undo.h"
56

57 58
#include "libgimp/gimpintl.h"

59

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

66 67
#define GUIDE_EPSILON 5

68

69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95
enum
{
  MODE_CHANGED,
  ALPHA_CHANGED,
  FLOATING_SELECTION_CHANGED,
  ACTIVE_LAYER_CHANGED,
  ACTIVE_CHANNEL_CHANGED,
  COMPONENT_VISIBILITY_CHANGED,
  COMPONENT_ACTIVE_CHANGED,
  MASK_CHANGED,
  RESOLUTION_CHANGED,
  UNIT_CHANGED,
  QMASK_CHANGED,
  SELECTION_CONTROL,

  CLEAN,
  DIRTY,
  UPDATE,
  UPDATE_GUIDE,
  COLORMAP_CHANGED,
  UNDO_EVENT,
  LAST_SIGNAL
};


/*  local function prototypes  */

96 97
static void     gimp_image_class_init            (GimpImageClass *klass);
static void     gimp_image_init                  (GimpImage      *gimage);
98 99 100 101

static void     gimp_image_dispose               (GObject        *object);
static void     gimp_image_finalize              (GObject        *object);

102 103
static void     gimp_image_name_changed          (GimpObject     *object);
static void     gimp_image_invalidate_preview    (GimpViewable   *viewable);
104
static void     gimp_image_size_changed          (GimpViewable   *viewable);
Michael Natterer's avatar
Michael Natterer committed
105 106
static void     gimp_image_real_colormap_changed (GimpImage      *gimage,
						  gint            ncol);
107
static TempBuf *gimp_image_get_preview           (GimpViewable   *gimage,
108 109
						  gint            width,
						  gint            height);
110
static TempBuf *gimp_image_get_new_preview       (GimpViewable   *viewable,
111 112 113 114 115 116
						  gint            width, 
						  gint            height);
static void     gimp_image_allocate_shadow       (GimpImage      *gimage,
						  gint            width,
						  gint            height,
						  gint            bpp);
117 118

/*
119
 *  Static variables
120
 */
121
static 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 138
static guint gimp_image_signals[LAST_SIGNAL] = { 0 };

139
static GimpViewableClass *parent_class = NULL;
140

141

142
GType 
143 144
gimp_image_get_type (void) 
{
145
  static GType image_type = 0;
146 147 148

  if (! image_type)
    {
149
      static const GTypeInfo image_info =
150 151
      {
        sizeof (GimpImageClass),
152 153 154 155 156 157 158 159
        NULL,           /* base_init */
        NULL,           /* base_finalize */
        (GClassInitFunc) gimp_image_class_init,
        NULL,           /* class_finalize */
        NULL,           /* class_data */
        sizeof (GimpImage),
        0,              /* n_preallocs */
        (GInstanceInitFunc) gimp_image_init,
160 161
      };

162 163 164
      image_type = g_type_register_static (GIMP_TYPE_VIEWABLE,
					   "GimpImage",
					   &image_info, 0);
165 166 167 168 169
    }

  return image_type;
}

170 171 172
static void
gimp_image_class_init (GimpImageClass *klass)
{
173
  GObjectClass      *object_class;
174 175
  GimpObjectClass   *gimp_object_class;
  GimpViewableClass *viewable_class;
176

177 178 179
  object_class      = G_OBJECT_CLASS (klass);
  gimp_object_class = GIMP_OBJECT_CLASS (klass);
  viewable_class    = GIMP_VIEWABLE_CLASS (klass);
180

181
  parent_class = g_type_class_peek_parent (klass);
182

183
  gimp_image_signals[MODE_CHANGED] =
184 185 186 187 188
    g_signal_new ("mode_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, mode_changed),
		  NULL, NULL,
189
		  gimp_marshal_VOID__VOID,
190
		  G_TYPE_NONE, 0);
191

192
  gimp_image_signals[ALPHA_CHANGED] =
193 194 195 196 197
    g_signal_new ("alpha_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, alpha_changed),
		  NULL, NULL,
198
		  gimp_marshal_VOID__VOID,
199
		  G_TYPE_NONE, 0);
200

201
  gimp_image_signals[FLOATING_SELECTION_CHANGED] =
202 203 204 205 206
    g_signal_new ("floating_selection_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, floating_selection_changed),
		  NULL, NULL,
207
		  gimp_marshal_VOID__VOID,
208
		  G_TYPE_NONE, 0);
209

210
  gimp_image_signals[ACTIVE_LAYER_CHANGED] =
211 212 213 214 215
    g_signal_new ("active_layer_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, active_layer_changed),
		  NULL, NULL,
216
		  gimp_marshal_VOID__VOID,
217
		  G_TYPE_NONE, 0);
218 219

  gimp_image_signals[ACTIVE_CHANNEL_CHANGED] =
220 221 222 223 224
    g_signal_new ("active_channel_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, active_channel_changed),
		  NULL, NULL,
225
		  gimp_marshal_VOID__VOID,
226
		  G_TYPE_NONE, 0);
227

228
  gimp_image_signals[COMPONENT_VISIBILITY_CHANGED] =
229 230 231 232 233
    g_signal_new ("component_visibility_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, component_visibility_changed),
		  NULL, NULL,
234
		  gimp_marshal_VOID__INT,
235 236
		  G_TYPE_NONE, 1,
		  G_TYPE_INT);
237 238

  gimp_image_signals[COMPONENT_ACTIVE_CHANGED] =
239 240 241 242 243
    g_signal_new ("component_active_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, component_active_changed),
		  NULL, NULL,
244
		  gimp_marshal_VOID__INT,
245 246
		  G_TYPE_NONE, 1,
		  G_TYPE_INT);
247

248
  gimp_image_signals[MASK_CHANGED] =
249 250 251 252 253
    g_signal_new ("mask_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, mask_changed),
		  NULL, NULL,
254
		  gimp_marshal_VOID__VOID,
255
		  G_TYPE_NONE, 0);
256

257 258 259 260 261 262
  gimp_image_signals[RESOLUTION_CHANGED] =
    g_signal_new ("resolution_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, resolution_changed),
		  NULL, NULL,
263
		  gimp_marshal_VOID__VOID,
264 265 266 267 268 269 270 271
		  G_TYPE_NONE, 0);

  gimp_image_signals[UNIT_CHANGED] =
    g_signal_new ("unit_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, unit_changed),
		  NULL, NULL,
272
		  gimp_marshal_VOID__VOID,
273 274
		  G_TYPE_NONE, 0);

Michael Natterer's avatar
Michael Natterer committed
275 276 277 278 279 280
  gimp_image_signals[QMASK_CHANGED] =
    g_signal_new ("qmask_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, qmask_changed),
		  NULL, NULL,
281
		  gimp_marshal_VOID__VOID,
Michael Natterer's avatar
Michael Natterer committed
282 283
		  G_TYPE_NONE, 0);

284 285 286 287 288 289
  gimp_image_signals[SELECTION_CONTROL] =
    g_signal_new ("selection_control",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, selection_control),
		  NULL, NULL,
290
		  gimp_marshal_VOID__INT,
291 292 293
		  G_TYPE_NONE, 1,
                  G_TYPE_INT);

294
  gimp_image_signals[CLEAN] =
295 296 297 298 299
    g_signal_new ("clean",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, clean),
		  NULL, NULL,
300
		  gimp_marshal_VOID__VOID,
301
		  G_TYPE_NONE, 0);
302

303
  gimp_image_signals[DIRTY] =
304 305 306 307 308
    g_signal_new ("dirty",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, dirty),
		  NULL, NULL,
309
		  gimp_marshal_VOID__VOID,
310
		  G_TYPE_NONE, 0);
311

Michael Natterer's avatar
Michael Natterer committed
312
  gimp_image_signals[UPDATE] =
313 314 315 316 317
    g_signal_new ("update",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, update),
		  NULL, NULL,
318
		  gimp_marshal_VOID__INT_INT_INT_INT,
319 320 321 322 323
		  G_TYPE_NONE, 4,
		  G_TYPE_INT,
		  G_TYPE_INT,
		  G_TYPE_INT,
		  G_TYPE_INT);
324

325 326 327 328 329 330
  gimp_image_signals[UPDATE_GUIDE] =
    g_signal_new ("update_guide",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, update_guide),
		  NULL, NULL,
331
		  gimp_marshal_VOID__POINTER,
332 333 334
		  G_TYPE_NONE, 1,
		  G_TYPE_POINTER);

335
  gimp_image_signals[COLORMAP_CHANGED] =
336 337 338 339 340
    g_signal_new ("colormap_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, colormap_changed),
		  NULL, NULL,
341
		  gimp_marshal_VOID__INT,
342 343
		  G_TYPE_NONE, 1,
		  G_TYPE_INT);
344

345
  gimp_image_signals[UNDO_EVENT] = 
346 347 348 349 350
    g_signal_new ("undo_event",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, undo_event),
		  NULL, NULL,
351
		  gimp_marshal_VOID__INT,
352 353
		  G_TYPE_NONE, 1,
		  G_TYPE_INT);
354

355 356
  object_class->dispose               = gimp_image_dispose;
  object_class->finalize              = gimp_image_finalize;
357

358
  gimp_object_class->name_changed     = gimp_image_name_changed;
359

360
  viewable_class->invalidate_preview  = gimp_image_invalidate_preview;
361
  viewable_class->size_changed        = gimp_image_size_changed;
362 363
  viewable_class->get_preview         = gimp_image_get_preview;
  viewable_class->get_new_preview     = gimp_image_get_new_preview;
364

365
  klass->mode_changed                 = NULL;
366
  klass->alpha_changed                = NULL;
367
  klass->floating_selection_changed   = NULL;
368 369 370 371
  klass->active_layer_changed         = NULL;
  klass->active_channel_changed       = NULL;
  klass->component_visibility_changed = NULL;
  klass->component_active_changed     = NULL;
372
  klass->mask_changed                 = NULL;
373

374 375
  klass->clean                        = NULL;
  klass->dirty                        = NULL;
Michael Natterer's avatar
Michael Natterer committed
376
  klass->update                       = NULL;
377
  klass->update_guide                 = NULL;
Michael Natterer's avatar
Michael Natterer committed
378
  klass->colormap_changed             = gimp_image_real_colormap_changed;
379 380 381
  klass->undo_event                   = NULL;
  klass->undo                         = gimp_image_undo;
  klass->redo                         = gimp_image_redo;
Michael Natterer's avatar
Michael Natterer committed
382 383

  gimp_image_color_hash_init ();
384 385 386 387 388
}


/* static functions */

Sven Neumann's avatar
Sven Neumann committed
389 390
static void 
gimp_image_init (GimpImage *gimage)
391
{
392
  gimage->ID                    = 0;
393

394 395 396 397
  gimage->save_proc             = NULL;

  gimage->width                 = 0;
  gimage->height                = 0;
398 399 400
  gimage->xresolution           = 1.0;
  gimage->yresolution           = 1.0;
  gimage->unit                  = GIMP_UNIT_INCH;
401 402
  gimage->base_type             = RGB;

403
  gimage->cmap                  = NULL;
404 405
  gimage->num_cols              = 0;

406
  gimage->dirty                 = 1;
407
  gimage->undo_on               = TRUE;
408 409 410 411

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

412
  gimage->tattoo_state          = 0;
413 414 415

  gimage->shadow                = NULL;

416
  gimage->construct_flag        = FALSE;
417
  gimage->proj_type             = RGBA_GIMAGE;
418
  gimage->projection            = NULL;
419

420
  gimage->guides                = NULL;
421

422 423 424 425
  gimage->layers                = gimp_list_new (GIMP_TYPE_LAYER, 
						 GIMP_CONTAINER_POLICY_STRONG);
  gimage->channels              = gimp_list_new (GIMP_TYPE_CHANNEL, 
						 GIMP_CONTAINER_POLICY_STRONG);
426
  gimage->layer_stack           = NULL;
427 428 429 430 431 432

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

433
  gimage->parasites             = gimp_parasite_list_new ();
434 435 436 437 438 439 440 441 442

  gimage->paths                 = NULL;

  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;

443 444 445 446
  gimage->undo_stack            = NULL;
  gimage->redo_stack            = NULL;
  gimage->undo_bytes            = 0;
  gimage->undo_levels           = 0;
447
  gimage->group_count           = 0;
448
  gimage->pushing_undo_group    = UNDO_NULL;
449

450 451 452
  gimage->new_undo_stack        = gimp_undo_stack_new (gimage);
  gimage->new_redo_stack        = gimp_undo_stack_new (gimage);

453
  gimage->comp_preview          = NULL;
454
  gimage->comp_preview_valid    = FALSE;
455 456
}

457
static void
458 459 460 461 462 463 464 465 466 467 468 469 470
gimp_image_dispose (GObject *object)
{
  GimpImage *gimage;

  gimage = GIMP_IMAGE (object);

  undo_free (gimage);

  G_OBJECT_CLASS (parent_class)->dispose (object);
}

static void
gimp_image_finalize (GObject *object)
Sven Neumann's avatar
Sven Neumann committed
471
{
472 473 474 475
  GimpImage *gimage;

  gimage = GIMP_IMAGE (object);

476 477 478 479 480 481 482 483
  if (gimage->gimp && gimage->gimp->image_table)
    {
      g_hash_table_remove (gimage->gimp->image_table,
			   GINT_TO_POINTER (gimage->ID));
      gimage->gimp = NULL;
    }

  if (gimage->projection)
484
    gimp_image_projection_free (gimage);
485 486 487

  if (gimage->shadow)
    gimp_image_free_shadow (gimage);
488

489
  if (gimage->cmap)
490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509
    {
      g_free (gimage->cmap);
      gimage->cmap = NULL;
    }

  if (gimage->layers)
    {
      g_object_unref (G_OBJECT (gimage->layers));
      gimage->layers = NULL;
    }
  if (gimage->channels)
    {
      g_object_unref (G_OBJECT (gimage->channels));
      gimage->channels = NULL;
    }
  if (gimage->layer_stack)
    {
      g_slist_free (gimage->layer_stack);
      gimage->layer_stack = NULL;
    }
510

511 512 513 514 515
  if (gimage->selection_mask)
    {
      g_object_unref (G_OBJECT (gimage->selection_mask));
      gimage->selection_mask = NULL;
    }
516

517
  if (gimage->comp_preview)
518 519 520 521
    {
      temp_buf_free (gimage->comp_preview);
      gimage->comp_preview = NULL;
    }
522 523

  if (gimage->parasites)
524 525 526 527
    {
      g_object_unref (G_OBJECT (gimage->parasites));
      gimage->parasites = NULL;
    }
528

529 530 531 532 533 534
  if (gimage->guides)
    {
      g_list_foreach (gimage->guides, (GFunc) g_free, NULL);
      g_list_free (gimage->guides);
      gimage->guides = NULL;
    }
Michael Natterer's avatar
Michael Natterer committed
535

536 537 538 539 540 541 542 543 544 545 546
  if (gimage->new_undo_stack)
    {
      g_object_unref (G_OBJECT (gimage->new_undo_stack));
      gimage->new_undo_stack = NULL;
    }
  if (gimage->new_redo_stack)
    {
      g_object_unref (G_OBJECT (gimage->new_redo_stack));
      gimage->new_redo_stack = NULL;
    }

547
  G_OBJECT_CLASS (parent_class)->finalize (object);
548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565
}

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;
566
    }
567
}
568

569 570 571 572 573
static void
gimp_image_invalidate_preview (GimpViewable *viewable)
{
  GimpImage *gimage;

574 575 576
  if (GIMP_VIEWABLE_CLASS (parent_class)->invalidate_preview)
    GIMP_VIEWABLE_CLASS (parent_class)->invalidate_preview (viewable);

577 578 579
  gimage = GIMP_IMAGE (viewable);

  gimage->comp_preview_valid = FALSE;
580 581
}

582 583 584 585 586 587 588 589 590 591 592 593 594 595
static void
gimp_image_size_changed (GimpViewable *viewable)
{
  GimpImage *gimage;

  if (GIMP_VIEWABLE_CLASS (parent_class)->size_changed)
    GIMP_VIEWABLE_CLASS (parent_class)->size_changed (viewable);

  gimage = GIMP_IMAGE (viewable);

  gimp_image_invalidate_layer_previews (gimage);
  gimp_image_invalidate_channel_previews (gimage);
}

Michael Natterer's avatar
Michael Natterer committed
596 597 598 599 600 601 602 603
static void 
gimp_image_real_colormap_changed (GimpImage *gimage,
				  gint       ncol)
{
  if (gimp_image_base_type (gimage) == INDEXED)
    gimp_image_color_hash_invalidate (gimage, ncol);
}

604
static void
Sven Neumann's avatar
Sven Neumann committed
605
gimp_image_allocate_shadow (GimpImage *gimage, 
606 607 608
			    gint       width, 
			    gint       height, 
			    gint       bpp)
609 610 611 612 613 614 615 616 617
{
  /*  allocate the new projection  */
  gimage->shadow = tile_manager_new (width, height, bpp);
}


/* function definitions */

GimpImage *
618 619
gimp_image_new (Gimp              *gimp,
		gint               width,
620 621
		gint               height,
		GimpImageBaseType  base_type)
622
{
623 624 625 626 627
  GimpImage *gimage;
  gint       i;

  g_return_val_if_fail (GIMP_IS_GIMP (gimp), NULL);

Michael Natterer's avatar
Michael Natterer committed
628
  gimage = GIMP_IMAGE (g_object_new (GIMP_TYPE_IMAGE, NULL));
629

630 631 632 633 634 635 636 637 638 639 640 641 642 643
  gimage->gimp        = gimp;
  gimage->ID          = gimp->next_image_ID++;

  g_hash_table_insert (gimp->image_table,
		       GINT_TO_POINTER (gimage->ID),
		       (gpointer) gimage);

  gimage->width       = width;
  gimage->height      = height;
  gimage->base_type   = base_type;

  gimage->xresolution = gimp->config->default_xresolution;
  gimage->yresolution = gimp->config->default_yresolution;
  gimage->unit        = gimp->config->default_units;
644 645 646 647 648 649 650 651 652

  switch (base_type)
    {
    case RGB:
    case GRAY:
      break;
    case INDEXED:
      /* always allocate 256 colors for the colormap */
      gimage->num_cols = 0;
653
      gimage->cmap     = (guchar *) g_malloc0 (COLORMAP_SIZE);
654 655 656 657 658 659 660 661
      break;
    default:
      break;
    }

  /*  set all color channels visible and active  */
  for (i = 0; i < MAX_CHANNELS; i++)
    {
662 663
      gimage->visible[i] = TRUE;
      gimage->active[i]  = TRUE;
664 665 666
    }

  /* create the selection mask */
Michael Natterer's avatar
Michael Natterer committed
667 668 669
  gimage->selection_mask = gimp_channel_new_mask (gimage,
						  gimage->width,
						  gimage->height);
670 671 672 673 674


  return gimage;
}

675 676 677 678 679 680 681 682 683
gint
gimp_image_get_ID (GimpImage *gimage)
{
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), -1);

  return gimage->ID;
}

GimpImage *
684 685
gimp_image_get_by_ID (Gimp *gimp,
		      gint  image_id)
686
{
687 688 689
  g_return_val_if_fail (GIMP_IS_GIMP (gimp), NULL);

  if (gimp->image_table == NULL)
690 691
    return NULL;

692
  return (GimpImage *) g_hash_table_lookup (gimp->image_table, 
693 694 695
					    GINT_TO_POINTER (image_id));
}

696
void
697 698
gimp_image_set_filename (GimpImage   *gimage,
			 const gchar *filename)
699
{
700 701
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

702 703
  gimp_object_set_name (GIMP_OBJECT (gimage), filename);
}
704

705
void
706
gimp_image_set_resolution (GimpImage *gimage,
707 708
			   gdouble    xresolution,
			   gdouble    yresolution)
709
{
710 711
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

712 713 714
  /* 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)
715 716
    return;

717 718 719 720
  if ((ABS (gimage->xresolution - xresolution) >= 1e-5) ||
      (ABS (gimage->yresolution - yresolution) >= 1e-5))
    {
      undo_push_resolution (gimage);
721

722 723
      gimage->xresolution = xresolution;
      gimage->yresolution = yresolution;
724

725 726 727
      gimp_image_resolution_changed (gimage);
      gimp_viewable_size_changed (GIMP_VIEWABLE (gimage));
    }
728 729
}

730
void
731 732 733
gimp_image_get_resolution (const GimpImage *gimage,
			   gdouble         *xresolution,
			   gdouble         *yresolution)
734
{
735
  g_return_if_fail (GIMP_IS_IMAGE (gimage));
736 737
  g_return_if_fail (xresolution && yresolution);

738 739
  *xresolution = gimage->xresolution;
  *yresolution = gimage->yresolution;
740 741
}

742 743
void
gimp_image_set_unit (GimpImage *gimage,
744
		     GimpUnit   unit)
745
{
746 747
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

748 749 750 751 752
  if (gimage->unit != unit)
    {
      undo_push_resolution (gimage);

      gimage->unit = unit;
753

754 755
      gimp_image_unit_changed (gimage);
    }
756 757
}

758
GimpUnit
759
gimp_image_get_unit (const GimpImage *gimage)
760
{
761 762
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), GIMP_UNIT_INCH);

763 764 765
  return gimage->unit;
}

Michael Natterer's avatar
Michael Natterer committed
766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787
void
gimp_image_set_qmask_state (GimpImage *gimage,
                            gboolean   qmask_state)
{
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

  if (qmask_state != gimage->qmask_state)
    {
      gimage->qmask_state = qmask_state ? TRUE : FALSE;

      gimp_image_qmask_changed (gimage);
    }
}

gboolean
gimp_image_get_qmask_state (const GimpImage *gimage)
{
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), FALSE);

  return gimage->qmask_state;
}

788
void
Sven Neumann's avatar
Sven Neumann committed
789 790
gimp_image_set_save_proc (GimpImage     *gimage, 
			  PlugInProcDef *proc)
791
{
792 793
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

794 795 796 797
  gimage->save_proc = proc;
}

PlugInProcDef *
798
gimp_image_get_save_proc (const GimpImage *gimage)
799
{
800 801
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), NULL);

802 803
  return gimage->save_proc;
}
804

805 806 807
gint
gimp_image_get_width (const GimpImage *gimage)
{
808 809
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), 0);

810 811 812 813 814 815
  return gimage->width;
}

gint
gimp_image_get_height (const GimpImage *gimage)
{
816 817
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), 0);

818 819 820
  return gimage->height;
}

821
void
822
gimp_image_resize (GimpImage *gimage, 
823 824 825 826
		   gint       new_width, 
		   gint       new_height,
		   gint       offset_x, 
		   gint       offset_y)
827
{
Michael Natterer's avatar
Michael Natterer committed
828 829 830
  GimpChannel *channel;
  GimpLayer   *layer;
  GimpLayer   *floating_layer;
831
  GList       *list;
Michael Natterer's avatar
Michael Natterer committed
832
  GList       *guide_list;
833

834 835
  g_return_if_fail (GIMP_IS_IMAGE (gimage));
  g_return_if_fail (new_width > 0 && new_height > 0);
836

837
  gimp_set_busy (gimage->gimp);
838 839 840 841

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

Sven Neumann's avatar
Sven Neumann committed
842
  undo_push_group_start (gimage, IMAGE_RESIZE_UNDO);
843 844 845 846 847 848 849 850 851

  /*  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  */
852
  gimage->width  = new_width;
853 854 855
  gimage->height = new_height;

  /*  Resize all channels  */
856 857 858
  for (list = GIMP_LIST (gimage->channels)->list; 
       list; 
       list = g_list_next (list))
859
    {
Michael Natterer's avatar
Michael Natterer committed
860
      channel = (GimpChannel *) list->data;
861

Michael Natterer's avatar
Michael Natterer committed
862
      gimp_channel_resize (channel, new_width, new_height, offset_x, offset_y);
863 864 865 866 867 868
    }

  /*  Reposition or remove any guides  */
  guide_list = gimage->guides;
  while (guide_list)
    {
Michael Natterer's avatar
Michael Natterer committed
869
      GimpGuide *guide;
870

Michael Natterer's avatar
Michael Natterer committed
871
      guide = (GimpGuide *) guide_list->data;
Sven Neumann's avatar
Sven Neumann committed
872
      guide_list = g_list_next (guide_list);
873 874 875

      switch (guide->orientation)
	{
876
	case ORIENTATION_HORIZONTAL:
Sven Neumann's avatar
Sven Neumann committed
877
	  undo_push_guide (gimage, guide);
878 879 880 881
	  guide->position += offset_y;
	  if (guide->position < 0 || guide->position > new_height)
	    gimp_image_delete_guide (gimage, guide);
	  break;
882

883
	case ORIENTATION_VERTICAL:
Sven Neumann's avatar
Sven Neumann committed
884
	  undo_push_guide (gimage, guide);
885 886 887 888
	  guide->position += offset_x;
	  if (guide->position < 0 || guide->position > new_width)
	    gimp_image_delete_guide (gimage, guide);
	  break;
889

890
	default:
891
	  g_error ("Unknown guide orientation\n");
892
	}
893 894 895
    }

  /*  Don't forget the selection mask!  */
Michael Natterer's avatar
Michael Natterer committed
896 897
  gimp_channel_resize (gimage->selection_mask,
		       new_width, new_height, offset_x, offset_y);
898 899 900
  gimage_mask_invalidate (gimage);

  /*  Reposition all layers  */
901 902 903
  for (list = GIMP_LIST (gimage->layers)->list; 
       list; 
       list = g_list_next (list))
904
    {
905
      layer = (GimpLayer *) list->data;
906

907
      gimp_layer_translate (layer, offset_x, offset_y);
908 909 910
    }

  /*  Make sure the projection matches the gimage size  */
911
  gimp_image_projection_allocate (gimage);
912 913 914 915 916

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

917 918
  undo_push_group_end (gimage);

919
  gimp_viewable_size_changed (GIMP_VIEWABLE (gimage));
920

921
  gimp_unset_busy (gimage->gimp);
922 923 924
}

void
Michael Natterer's avatar
Michael Natterer committed
925 926 927 928 929
gimp_image_scale (GimpImage        *gimage, 
		  gint              new_width, 
		  gint              new_height,
                  GimpProgressFunc  progress_func,
                  gpointer          progress_data)
930
{
Michael Natterer's avatar
Michael Natterer committed
931 932 933
  GimpChannel *channel;
  GimpLayer   *layer;
  GimpLayer   *floating_layer;
934
  GList       *list;