gimpimage-resize.c 118 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
34
#include "base/pixel-region.h"
#include "base/temp-buf.h"
#include "base/tile-manager.h"
#include "base/tile.h"
Sven Neumann's avatar
Sven Neumann committed
35

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

38
#include "gimp.h"
39
#include "gimpcontext.h"
Michael Natterer's avatar
Michael Natterer committed
40
#include "gimpcoreconfig.h"
41
#include "gimpimage.h"
Michael Natterer's avatar
Michael Natterer committed
42
#include "gimpimage-colorhash.h"
43
#include "gimpimage-mask.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
/*  Local function declarations  */
70
71
static void     gimp_image_class_init            (GimpImageClass *klass);
static void     gimp_image_init                  (GimpImage      *gimage);
72
73
74
75

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

76
77
static void     gimp_image_name_changed          (GimpObject     *object);
static void     gimp_image_invalidate_preview    (GimpViewable   *viewable);
78
static void     gimp_image_size_changed          (GimpViewable   *viewable);
Michael Natterer's avatar
Michael Natterer committed
79
80
static void     gimp_image_real_colormap_changed (GimpImage      *gimage,
						  gint            ncol);
81
static TempBuf *gimp_image_get_preview           (GimpViewable   *gimage,
82
83
						  gint            width,
						  gint            height);
84
static TempBuf *gimp_image_get_new_preview       (GimpViewable   *viewable,
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
						  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);
111
112
113
114
115
static void     gimp_image_construct             (GimpImage      *gimage,
						  gint            x,
						  gint            y,
						  gint            w,
						  gint            h);
116
117

/*  projection functions  */
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
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);
141
142
143
144

/*
 *  Global variables
 */
145
static gint valid_combinations[][MAX_CHANNELS + 1] =
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
{
  /* 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 },
};


/*
 *  Static variables
 */

166
167
enum
{
168
  MODE_CHANGED,
169
  ALPHA_CHANGED,
170
  FLOATING_SELECTION_CHANGED,
171
172
  ACTIVE_LAYER_CHANGED,
  ACTIVE_CHANNEL_CHANGED,
173
174
  COMPONENT_VISIBILITY_CHANGED,
  COMPONENT_ACTIVE_CHANGED,
175
  MASK_CHANGED,
176
177
  RESOLUTION_CHANGED,
  UNIT_CHANGED,
Michael Natterer's avatar
Michael Natterer committed
178
  QMASK_CHANGED,
179
  SELECTION_CONTROL,
180

181
  CLEAN,
182
  DIRTY,
Michael Natterer's avatar
Michael Natterer committed
183
  UPDATE,
184
  UPDATE_GUIDE,
185
  COLORMAP_CHANGED,
186
  UNDO_EVENT,
187
188
  LAST_SIGNAL
};
189

190

191
192
static guint gimp_image_signals[LAST_SIGNAL] = { 0 };

193
static GimpViewableClass *parent_class = NULL;
194

195

196
GType 
197
198
gimp_image_get_type (void) 
{
199
  static GType image_type = 0;
200
201
202

  if (! image_type)
    {
203
      static const GTypeInfo image_info =
204
205
      {
        sizeof (GimpImageClass),
206
207
208
209
210
211
212
213
        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,
214
215
      };

216
217
218
      image_type = g_type_register_static (GIMP_TYPE_VIEWABLE,
					   "GimpImage",
					   &image_info, 0);
219
220
221
222
223
    }

  return image_type;
}

224
225
226
static void
gimp_image_class_init (GimpImageClass *klass)
{
227
  GObjectClass      *object_class;
228
229
  GimpObjectClass   *gimp_object_class;
  GimpViewableClass *viewable_class;
230

231
232
233
  object_class      = G_OBJECT_CLASS (klass);
  gimp_object_class = GIMP_OBJECT_CLASS (klass);
  viewable_class    = GIMP_VIEWABLE_CLASS (klass);
234

235
  parent_class = g_type_class_peek_parent (klass);
236

237
  gimp_image_signals[MODE_CHANGED] =
238
239
240
241
242
    g_signal_new ("mode_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, mode_changed),
		  NULL, NULL,
243
		  gimp_marshal_VOID__VOID,
244
		  G_TYPE_NONE, 0);
245

246
  gimp_image_signals[ALPHA_CHANGED] =
247
248
249
250
251
    g_signal_new ("alpha_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, alpha_changed),
		  NULL, NULL,
252
		  gimp_marshal_VOID__VOID,
253
		  G_TYPE_NONE, 0);
254

255
  gimp_image_signals[FLOATING_SELECTION_CHANGED] =
256
257
258
259
260
    g_signal_new ("floating_selection_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, floating_selection_changed),
		  NULL, NULL,
261
		  gimp_marshal_VOID__VOID,
262
		  G_TYPE_NONE, 0);
263

264
  gimp_image_signals[ACTIVE_LAYER_CHANGED] =
265
266
267
268
269
    g_signal_new ("active_layer_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, active_layer_changed),
		  NULL, NULL,
270
		  gimp_marshal_VOID__VOID,
271
		  G_TYPE_NONE, 0);
272
273

  gimp_image_signals[ACTIVE_CHANNEL_CHANGED] =
274
275
276
277
278
    g_signal_new ("active_channel_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, active_channel_changed),
		  NULL, NULL,
279
		  gimp_marshal_VOID__VOID,
280
		  G_TYPE_NONE, 0);
281

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

  gimp_image_signals[COMPONENT_ACTIVE_CHANGED] =
293
294
295
296
297
    g_signal_new ("component_active_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, component_active_changed),
		  NULL, NULL,
298
		  gimp_marshal_VOID__INT,
299
300
		  G_TYPE_NONE, 1,
		  G_TYPE_INT);
301

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

311
312
313
314
315
316
  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,
317
		  gimp_marshal_VOID__VOID,
318
319
320
321
322
323
324
325
		  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,
326
		  gimp_marshal_VOID__VOID,
327
328
		  G_TYPE_NONE, 0);

Michael Natterer's avatar
Michael Natterer committed
329
330
331
332
333
334
  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,
335
		  gimp_marshal_VOID__VOID,
Michael Natterer's avatar
Michael Natterer committed
336
337
		  G_TYPE_NONE, 0);

338
339
340
341
342
343
  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,
344
		  gimp_marshal_VOID__INT,
345
346
347
		  G_TYPE_NONE, 1,
                  G_TYPE_INT);

348
  gimp_image_signals[CLEAN] =
349
350
351
352
353
    g_signal_new ("clean",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, clean),
		  NULL, NULL,
354
		  gimp_marshal_VOID__VOID,
355
		  G_TYPE_NONE, 0);
356

357
  gimp_image_signals[DIRTY] =
358
359
360
361
362
    g_signal_new ("dirty",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, dirty),
		  NULL, NULL,
363
		  gimp_marshal_VOID__VOID,
364
		  G_TYPE_NONE, 0);
365

Michael Natterer's avatar
Michael Natterer committed
366
  gimp_image_signals[UPDATE] =
367
368
369
370
371
    g_signal_new ("update",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, update),
		  NULL, NULL,
372
		  gimp_marshal_VOID__INT_INT_INT_INT,
373
374
375
376
377
		  G_TYPE_NONE, 4,
		  G_TYPE_INT,
		  G_TYPE_INT,
		  G_TYPE_INT,
		  G_TYPE_INT);
378

379
380
381
382
383
384
  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,
385
		  gimp_marshal_VOID__POINTER,
386
387
388
		  G_TYPE_NONE, 1,
		  G_TYPE_POINTER);

389
  gimp_image_signals[COLORMAP_CHANGED] =
390
391
392
393
394
    g_signal_new ("colormap_changed",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, colormap_changed),
		  NULL, NULL,
395
		  gimp_marshal_VOID__INT,
396
397
		  G_TYPE_NONE, 1,
		  G_TYPE_INT);
398

399
  gimp_image_signals[UNDO_EVENT] = 
400
401
402
403
404
    g_signal_new ("undo_event",
		  G_TYPE_FROM_CLASS (klass),
		  G_SIGNAL_RUN_FIRST,
		  G_STRUCT_OFFSET (GimpImageClass, undo_event),
		  NULL, NULL,
405
		  gimp_marshal_VOID__INT,
406
407
		  G_TYPE_NONE, 1,
		  G_TYPE_INT);
408

409
410
  object_class->dispose               = gimp_image_dispose;
  object_class->finalize              = gimp_image_finalize;
411

412
  gimp_object_class->name_changed     = gimp_image_name_changed;
413

414
  viewable_class->invalidate_preview  = gimp_image_invalidate_preview;
415
  viewable_class->size_changed        = gimp_image_size_changed;
416
417
  viewable_class->get_preview         = gimp_image_get_preview;
  viewable_class->get_new_preview     = gimp_image_get_new_preview;
418

419
  klass->mode_changed                 = NULL;
420
  klass->alpha_changed                = NULL;
421
  klass->floating_selection_changed   = NULL;
422
423
424
425
  klass->active_layer_changed         = NULL;
  klass->active_channel_changed       = NULL;
  klass->component_visibility_changed = NULL;
  klass->component_active_changed     = NULL;
426
  klass->mask_changed                 = NULL;
427

428
429
  klass->clean                        = NULL;
  klass->dirty                        = NULL;
Michael Natterer's avatar
Michael Natterer committed
430
  klass->update                       = NULL;
431
  klass->update_guide                 = NULL;
Michael Natterer's avatar
Michael Natterer committed
432
  klass->colormap_changed             = gimp_image_real_colormap_changed;
433
434
435
  klass->undo_event                   = NULL;
  klass->undo                         = gimp_image_undo;
  klass->redo                         = gimp_image_redo;
Michael Natterer's avatar
Michael Natterer committed
436
437

  gimp_image_color_hash_init ();
438
439
440
441
442
}


/* static functions */

Sven Neumann's avatar
Sven Neumann committed
443
444
static void 
gimp_image_init (GimpImage *gimage)
445
{
446
  gimage->ID                    = 0;
447

448
449
450
451
  gimage->save_proc             = NULL;

  gimage->width                 = 0;
  gimage->height                = 0;
452
453
454
  gimage->xresolution           = 1.0;
  gimage->yresolution           = 1.0;
  gimage->unit                  = GIMP_UNIT_INCH;
455
456
  gimage->base_type             = RGB;

457
  gimage->cmap                  = NULL;
458
459
  gimage->num_cols              = 0;

460
  gimage->dirty                 = 1;
461
  gimage->undo_on               = TRUE;
462
463
464
465

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

466
  gimage->tattoo_state          = 0;
467
468
469
470
471

  gimage->shadow                = NULL;

  gimage->construct_flag        = -1;
  gimage->proj_type             = RGBA_GIMAGE;
472
  gimage->projection            = NULL;
473

474
  gimage->guides                = NULL;
475

476
477
478
479
  gimage->layers                = gimp_list_new (GIMP_TYPE_LAYER, 
						 GIMP_CONTAINER_POLICY_STRONG);
  gimage->channels              = gimp_list_new (GIMP_TYPE_CHANNEL, 
						 GIMP_CONTAINER_POLICY_STRONG);
480
  gimage->layer_stack           = NULL;
481
482
483
484
485
486

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

487
  gimage->parasites             = gimp_parasite_list_new ();
488
489
490
491
492
493
494
495
496

  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;

497
498
499
500
  gimage->undo_stack            = NULL;
  gimage->redo_stack            = NULL;
  gimage->undo_bytes            = 0;
  gimage->undo_levels           = 0;
501
  gimage->group_count           = 0;
502
  gimage->pushing_undo_group    = UNDO_NULL;
503

504
505
506
  gimage->new_undo_stack        = gimp_undo_stack_new (gimage);
  gimage->new_redo_stack        = gimp_undo_stack_new (gimage);

507
  gimage->comp_preview          = NULL;
508
  gimage->comp_preview_valid    = FALSE;
509
510
}

511
static void
512
513
514
515
516
517
518
519
520
521
522
523
524
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
525
{
526
527
528
529
  GimpImage *gimage;

  gimage = GIMP_IMAGE (object);

530
531
532
533
534
535
536
537
538
539
540
541
  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)
    gimp_image_free_projection (gimage);

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

543
  if (gimage->cmap)
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
    {
      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;
    }
564

565
566
567
568
569
  if (gimage->selection_mask)
    {
      g_object_unref (G_OBJECT (gimage->selection_mask));
      gimage->selection_mask = NULL;
    }
570

571
  if (gimage->comp_preview)
572
573
574
575
    {
      temp_buf_free (gimage->comp_preview);
      gimage->comp_preview = NULL;
    }
576
577

  if (gimage->parasites)
578
579
580
581
    {
      g_object_unref (G_OBJECT (gimage->parasites));
      gimage->parasites = NULL;
    }
582

583
584
585
586
587
588
  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
589

590
591
592
593
594
595
596
597
598
599
600
  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;
    }

601
  G_OBJECT_CLASS (parent_class)->finalize (object);
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
}

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;
620
    }
621
}
622

623
624
625
626
627
static void
gimp_image_invalidate_preview (GimpViewable *viewable)
{
  GimpImage *gimage;

628
629
630
  if (GIMP_VIEWABLE_CLASS (parent_class)->invalidate_preview)
    GIMP_VIEWABLE_CLASS (parent_class)->invalidate_preview (viewable);

631
632
633
  gimage = GIMP_IMAGE (viewable);

  gimage->comp_preview_valid = FALSE;
634
635
}

636
637
638
639
640
641
642
643
644
645
646
647
648
649
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
650
651
652
653
654
655
656
657
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);
}

658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
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:
680
      g_assert_not_reached ();
681
682
683
    }

  /*  allocate the new projection  */
684
685
  gimage->projection = tile_manager_new (gimage->width, gimage->height,
					 gimage->proj_bytes);
686
  tile_manager_set_user_data (gimage->projection, (void *) gimage);
687
688
689
690
691
692
  tile_manager_set_validate_proc (gimage->projection, gimp_image_validate);
}

static void
gimp_image_free_projection (GimpImage *gimage)
{
693
  g_return_if_fail (GIMP_IS_IMAGE (gimage));
694

695
696
697
698
699
  if (gimage->projection)
    {
      tile_manager_destroy (gimage->projection);
      gimage->projection = NULL;
    }
700
701
702
}

static void
Sven Neumann's avatar
Sven Neumann committed
703
gimp_image_allocate_shadow (GimpImage *gimage, 
704
705
706
			    gint       width, 
			    gint       height, 
			    gint       bpp)
707
708
709
710
711
712
713
714
715
{
  /*  allocate the new projection  */
  gimage->shadow = tile_manager_new (width, height, bpp);
}


/* function definitions */

GimpImage *
716
717
gimp_image_new (Gimp              *gimp,
		gint               width,
718
719
		gint               height,
		GimpImageBaseType  base_type)
720
{
721
722
723
724
725
  GimpImage *gimage;
  gint       i;

  g_return_val_if_fail (GIMP_IS_GIMP (gimp), NULL);

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

728
729
730
731
732
733
734
735
736
737
738
739
740
741
  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;
742
743
744
745
746
747
748
749
750

  switch (base_type)
    {
    case RGB:
    case GRAY:
      break;
    case INDEXED:
      /* always allocate 256 colors for the colormap */
      gimage->num_cols = 0;
751
      gimage->cmap     = (guchar *) g_malloc0 (COLORMAP_SIZE);
752
753
754
755
756
757
758
759
      break;
    default:
      break;
    }

  /*  set all color channels visible and active  */
  for (i = 0; i < MAX_CHANNELS; i++)
    {
760
761
      gimage->visible[i] = TRUE;
      gimage->active[i]  = TRUE;
762
763
764
    }

  /* create the selection mask */
Michael Natterer's avatar
Michael Natterer committed
765
766
767
  gimage->selection_mask = gimp_channel_new_mask (gimage,
						  gimage->width,
						  gimage->height);
768
769
770
771
772


  return gimage;
}

773
774
775
776
777
778
779
780
781
gint
gimp_image_get_ID (GimpImage *gimage)
{
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), -1);

  return gimage->ID;
}

GimpImage *
782
783
gimp_image_get_by_ID (Gimp *gimp,
		      gint  image_id)
784
{
785
786
787
  g_return_val_if_fail (GIMP_IS_GIMP (gimp), NULL);

  if (gimp->image_table == NULL)
788
789
    return NULL;

790
  return (GimpImage *) g_hash_table_lookup (gimp->image_table, 
791
792
793
					    GINT_TO_POINTER (image_id));
}

794
void
795
796
gimp_image_set_filename (GimpImage   *gimage,
			 const gchar *filename)
797
{
798
799
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

800
801
  gimp_object_set_name (GIMP_OBJECT (gimage), filename);
}
802

803
void
804
gimp_image_set_resolution (GimpImage *gimage,
805
806
			   gdouble    xresolution,
			   gdouble    yresolution)
807
{
808
809
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

810
811
812
  /* 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)
813
814
    return;

815
816
817
818
  if ((ABS (gimage->xresolution - xresolution) >= 1e-5) ||
      (ABS (gimage->yresolution - yresolution) >= 1e-5))
    {
      undo_push_resolution (gimage);
819

820
821
      gimage->xresolution = xresolution;
      gimage->yresolution = yresolution;
822

823
824
825
      gimp_image_resolution_changed (gimage);
      gimp_viewable_size_changed (GIMP_VIEWABLE (gimage));
    }
826
827
}

828
void
829
830
831
gimp_image_get_resolution (const GimpImage *gimage,
			   gdouble         *xresolution,
			   gdouble         *yresolution)
832
{
833
  g_return_if_fail (GIMP_IS_IMAGE (gimage));
834
835
  g_return_if_fail (xresolution && yresolution);

836
837
  *xresolution = gimage->xresolution;
  *yresolution = gimage->yresolution;
838
839
}

840
841
void
gimp_image_set_unit (GimpImage *gimage,
842
		     GimpUnit   unit)
843
{
844
845
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

846
847
848
849
850
  if (gimage->unit != unit)
    {
      undo_push_resolution (gimage);

      gimage->unit = unit;
851

852
853
      gimp_image_unit_changed (gimage);
    }
854
855
}

856
GimpUnit
857
gimp_image_get_unit (const GimpImage *gimage)
858
{
859
860
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), GIMP_UNIT_INCH);

861
862
863
  return gimage->unit;
}

Michael Natterer's avatar
Michael Natterer committed
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
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;
}

886
void
Sven Neumann's avatar
Sven Neumann committed
887
888
gimp_image_set_save_proc (GimpImage     *gimage, 
			  PlugInProcDef *proc)
889
{
890
891
  g_return_if_fail (GIMP_IS_IMAGE (gimage));

892
893
894
895
  gimage->save_proc = proc;
}

PlugInProcDef *
896
gimp_image_get_save_proc (const GimpImage *gimage)
897
{
898
899
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), NULL);

900
901
  return gimage->save_proc;
}
902

903
904
905
gint
gimp_image_get_width (const GimpImage *gimage)
{
906
907
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), 0);

908
909
910
911
912
913
  return gimage->width;
}

gint
gimp_image_get_height (const GimpImage *gimage)
{
914
915
  g_return_val_if_fail (GIMP_IS_IMAGE (gimage), 0);

916
917
918
  return gimage->height;
}

919
void
920
gimp_image_resize (GimpImage *gimage, 
921
922
923
924
		   gint       new_width, 
		   gint       new_height,
		   gint       offset_x, 
		   gint       offset_y)
925
{
Michael Natterer's avatar
Michael Natterer committed
926
927
928
  GimpChannel *channel;
  GimpLayer   *layer;
  GimpLayer   *floating_layer;
929
  GList       *list;
Michael Natterer's avatar
Michael Natterer committed
930
  GList       *guide_list;
931

932
933
  g_return_if_fail (GIMP_IS_IMAGE (gimage));
  g_return_if_fail (new_width > 0 && new_height > 0);
934

935
  gimp_set_busy (gimage->gimp);
936
937
938
939

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

Sven Neumann's avatar
Sven Neumann committed
940
  undo_push_group_start (gimage, IMAGE_RESIZE_UNDO);
941
942
943
944
945
946
947
948
949

  /*  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  */
950
  gimage->width  = new_width;
951
952
953
  gimage->height = new_height;

  /*  Resize all channels  */
954
955
956
  for (list = GIMP_LIST (gimage->channels)->list; 
       list; 
       list = g_list_next (list))
957
    {
Michael Natterer's avatar
Michael Natterer committed
958
      channel = (GimpChannel *) list->data;
959

Michael Natterer's avatar
Michael Natterer committed
960
      gimp_channel_resize (channel, new_width, new_height, offset_x, offset_y);
961
962
963
964
965
966
    }

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

Michael Natterer's avatar
Michael Natterer committed
969
      guide = (GimpGuide *) guide_list->data;
Sven Neumann's avatar
Sven Neumann committed
970
      guide_list = g_list_next (guide_list);
971
972
973

      switch (guide->orientation)
	{
974
	case ORIENTATION_HORIZONTAL:
Sven Neumann's avatar
Sven Neumann committed
975
	  undo_push_guide (gimage, guide);
976
977
978
979
	  guide->position += offset_y;
	  if (guide->position < 0 || guide->position > new_height)
	    gimp_image_delete_guide (gimage, guide);
	  break;
980

981
	case ORIENTATION_VERTICAL:
Sven Neumann's avatar
Sven Neumann committed
982
	  undo_push_guide (gimage, guide);
983
984
985
986
	  guide->position += offset_x;
	  if (guide->position < 0 || guide->position > new_width)
	    gimp_image_delete_guide (gimage, guide);
	  break;
987

988
	default:
989
	  g_error ("Unknown guide orientation\n");
990
	}
991
992
993
    }

  /*  Don't forget the selection mask!  */
Michael Natterer's avatar
Michael Natterer committed
994
995
  gimp_channel_resize (gimage->selection_mask,
		       new_width, new_height, offset_x, offset_y);
996
997
998
  gimage_mask_invalidate (gimage);

  /*  Reposition all layers  */
999
1000
  for (list = GIMP_LIST (gimage->layers)->list; 
       list; 
For faster browsing, not all history is shown. View entire blame