blob: a22bd0dd732f6bb98e758aae28866d3a9d014bec [file] [log] [blame]
cristy3ed852e2009-09-05 21:47:34 +00001// This may look like C code, but it is really -*- C++ -*-
2//
3// Copyright Bob Friesenhahn, 1999, 2000, 2001, 2002, 2003
4//
5// Implementation of Image
6//
7
8#define MAGICKCORE_IMPLEMENTATION 1
9#define MAGICK_PLUSPLUS_IMPLEMENTATION 1
10
cristy3ed852e2009-09-05 21:47:34 +000011#include <cstdlib>
12#include <string>
13#include <string.h>
14#include <errno.h>
15#include <math.h>
cristy4c08aed2011-07-01 19:47:50 +000016#include "Magick++/Include.h"
cristy3ed852e2009-09-05 21:47:34 +000017
18using namespace std;
19
20#include "Magick++/Image.h"
21#include "Magick++/Functions.h"
22#include "Magick++/Pixels.h"
23#include "Magick++/Options.h"
24#include "Magick++/ImageRef.h"
25
26#define AbsoluteValue(x) ((x) < 0 ? -(x) : (x))
cristy3ed852e2009-09-05 21:47:34 +000027#define DegreesToRadians(x) (MagickPI*(x)/180.0)
28
cristyaf1dd252011-09-07 19:04:02 +000029MagickPPExport const char *Magick::borderGeometryDefault = "6x6+0+0";
30MagickPPExport const char *Magick::frameGeometryDefault = "25x25+6+6";
31MagickPPExport const char *Magick::raiseGeometryDefault = "6x6+0+0";
cristy3ed852e2009-09-05 21:47:34 +000032
33static bool magick_initialized=false;
34
35//
36// Explicit template instantiations
37//
38
39//
40// Friend functions to compare Image objects
41//
42
cristyaf1dd252011-09-07 19:04:02 +000043MagickPPExport int Magick::operator == ( const Magick::Image& left_,
cristy3ed852e2009-09-05 21:47:34 +000044 const Magick::Image& right_ )
45{
46 // If image pixels and signature are the same, then the image is identical
47 return ( ( left_.rows() == right_.rows() ) &&
48 ( left_.columns() == right_.columns() ) &&
49 ( left_.signature() == right_.signature() )
50 );
51}
cristyaf1dd252011-09-07 19:04:02 +000052MagickPPExport int Magick::operator != ( const Magick::Image& left_,
cristy3ed852e2009-09-05 21:47:34 +000053 const Magick::Image& right_ )
54{
55 return ( ! (left_ == right_) );
56}
cristyaf1dd252011-09-07 19:04:02 +000057MagickPPExport int Magick::operator > ( const Magick::Image& left_,
cristy3ed852e2009-09-05 21:47:34 +000058 const Magick::Image& right_ )
59{
60 return ( !( left_ < right_ ) && ( left_ != right_ ) );
61}
cristyaf1dd252011-09-07 19:04:02 +000062MagickPPExport int Magick::operator < ( const Magick::Image& left_,
cristy3ed852e2009-09-05 21:47:34 +000063 const Magick::Image& right_ )
64{
65 // If image pixels are less, then image is smaller
66 return ( ( left_.rows() * left_.columns() ) <
67 ( right_.rows() * right_.columns() )
68 );
69}
cristyaf1dd252011-09-07 19:04:02 +000070MagickPPExport int Magick::operator >= ( const Magick::Image& left_,
cristy3ed852e2009-09-05 21:47:34 +000071 const Magick::Image& right_ )
72{
73 return ( ( left_ > right_ ) || ( left_ == right_ ) );
74}
cristyaf1dd252011-09-07 19:04:02 +000075MagickPPExport int Magick::operator <= ( const Magick::Image& left_,
cristy3ed852e2009-09-05 21:47:34 +000076 const Magick::Image& right_ )
77{
78 return ( ( left_ < right_ ) || ( left_ == right_ ) );
79}
80
81//
82// Image object implementation
83//
84
85// Construct from image file or image specification
86Magick::Image::Image( const std::string &imageSpec_ )
87 : _imgRef(new ImageRef)
88{
89 try
90 {
91 // Initialize, Allocate and Read images
92 read( imageSpec_ );
93 }
94 catch ( const Warning & /*warning_*/ )
95 {
96 // FIXME: need a way to report warnings in constructor
97 }
98 catch ( const Error & /*error_*/ )
99 {
100 // Release resources
101 delete _imgRef;
102 throw;
103 }
104}
105
106// Construct a blank image canvas of specified size and color
107Magick::Image::Image( const Geometry &size_,
108 const Color &color_ )
109 : _imgRef(new ImageRef)
110{
111 // xc: prefix specifies an X11 color string
112 std::string imageSpec("xc:");
113 imageSpec += color_;
114
115 try
116 {
117 // Set image size
118 size( size_ );
119
120 // Initialize, Allocate and Read images
121 read( imageSpec );
122 }
123 catch ( const Warning & /*warning_*/ )
124 {
125 // FIXME: need a way to report warnings in constructor
126 }
127 catch ( const Error & /*error_*/ )
128 {
129 // Release resources
130 delete _imgRef;
131 throw;
132 }
133}
134
135// Construct Image from in-memory BLOB
136Magick::Image::Image ( const Blob &blob_ )
137 : _imgRef(new ImageRef)
138{
139 try
140 {
141 // Initialize, Allocate and Read images
142 read( blob_ );
143 }
144 catch ( const Warning & /*warning_*/ )
145 {
146 // FIXME: need a way to report warnings in constructor
147 }
148 catch ( const Error & /*error_*/ )
149 {
150 // Release resources
151 delete _imgRef;
152 throw;
153 }
154}
155
156// Construct Image of specified size from in-memory BLOB
157Magick::Image::Image ( const Blob &blob_,
158 const Geometry &size_ )
159 : _imgRef(new ImageRef)
160{
161 try
162 {
163 // Read from Blob
164 read( blob_, size_ );
165 }
166 catch ( const Warning & /*warning_*/ )
167 {
168 // FIXME: need a way to report warnings in constructor
169 }
170 catch ( const Error & /*error_*/ )
171 {
172 // Release resources
173 delete _imgRef;
174 throw;
175 }
176}
177
178// Construct Image of specified size and depth from in-memory BLOB
179Magick::Image::Image ( const Blob &blob_,
180 const Geometry &size_,
cristyeaedf062010-05-29 22:36:02 +0000181 const size_t depth_ )
cristy3ed852e2009-09-05 21:47:34 +0000182 : _imgRef(new ImageRef)
183{
184 try
185 {
186 // Read from Blob
187 read( blob_, size_, depth_ );
188 }
189 catch ( const Warning & /*warning_*/ )
190 {
191 // FIXME: need a way to report warnings in constructor
192 }
193 catch ( const Error & /*error_*/ )
194 {
195 // Release resources
196 delete _imgRef;
197 throw;
198 }
199}
200
201// Construct Image of specified size, depth, and format from in-memory BLOB
202Magick::Image::Image ( const Blob &blob_,
203 const Geometry &size_,
cristyeaedf062010-05-29 22:36:02 +0000204 const size_t depth_,
cristy3ed852e2009-09-05 21:47:34 +0000205 const std::string &magick_ )
206 : _imgRef(new ImageRef)
207{
208 try
209 {
210 // Read from Blob
211 read( blob_, size_, depth_, magick_ );
212 }
213 catch ( const Warning & /*warning_*/ )
214 {
215 // FIXME: need a way to report warnings in constructor
216 }
217 catch ( const Error & /*error_*/ )
218 {
219 // Release resources
220 delete _imgRef;
221 throw;
222 }
223}
224
225// Construct Image of specified size, and format from in-memory BLOB
226Magick::Image::Image ( const Blob &blob_,
227 const Geometry &size_,
228 const std::string &magick_ )
229 : _imgRef(new ImageRef)
230{
231 try
232 {
233 // Read from Blob
234 read( blob_, size_, magick_ );
235 }
236 catch ( const Warning & /*warning_*/ )
237 {
238 // FIXME: need a way to report warnings in constructor
239 }
240 catch ( const Error & /*error_*/ )
241 {
242 // Release resources
243 delete _imgRef;
244 throw;
245 }
246}
247
248// Construct an image based on an array of raw pixels, of specified
249// type and mapping, in memory
cristyeaedf062010-05-29 22:36:02 +0000250Magick::Image::Image ( const size_t width_,
251 const size_t height_,
cristy3ed852e2009-09-05 21:47:34 +0000252 const std::string &map_,
253 const StorageType type_,
254 const void *pixels_ )
255 : _imgRef(new ImageRef)
256{
257 try
258 {
259 read( width_, height_, map_.c_str(), type_, pixels_ );
260 }
261 catch ( const Warning & /*warning_*/ )
262 {
263 // FIXME: need a way to report warnings in constructor
264 }
265 catch ( const Error & /*error_*/ )
266 {
267 // Release resources
268 delete _imgRef;
269 throw;
270 }
271}
272
273// Default constructor
274Magick::Image::Image( void )
275 : _imgRef(new ImageRef)
276{
277}
278
279// Destructor
280/* virtual */
281Magick::Image::~Image()
282{
283 bool doDelete = false;
284 {
285 Lock( &_imgRef->_mutexLock );
286 if ( --_imgRef->_refCount == 0 )
287 doDelete = true;
288 }
289
290 if ( doDelete )
291 {
292 delete _imgRef;
293 }
294 _imgRef = 0;
295}
296
297// Adaptive-blur image
298void Magick::Image::adaptiveBlur( const double radius_, const double sigma_ )
299{
300 ExceptionInfo exceptionInfo;
301 GetExceptionInfo( &exceptionInfo );
302 MagickCore::Image* newImage =
cristyaa2c16c2012-03-25 22:21:35 +0000303 AdaptiveBlurImage( image(), radius_, sigma_, &exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +0000304 replaceImage( newImage );
305 throwException( exceptionInfo );
306 (void) DestroyExceptionInfo( &exceptionInfo );
307}
308
309// Local adaptive threshold image
310// http://www.dai.ed.ac.uk/HIPR2/adpthrsh.htm
311// Width x height define the size of the pixel neighborhood
312// offset = constant to subtract from pixel neighborhood mean
cristyeaedf062010-05-29 22:36:02 +0000313void Magick::Image::adaptiveThreshold ( const size_t width_,
314 const size_t height_,
cristy4e0eef02010-05-30 21:52:21 +0000315 const ssize_t offset_ )
cristy3ed852e2009-09-05 21:47:34 +0000316{
317 ExceptionInfo exceptionInfo;
318 GetExceptionInfo( &exceptionInfo );
319 MagickCore::Image* newImage =
320 AdaptiveThresholdImage( constImage(), width_, height_, offset_, &exceptionInfo );
321 replaceImage( newImage );
322 throwException( exceptionInfo );
323 (void) DestroyExceptionInfo( &exceptionInfo );
324}
325
326// Add noise to image
327void Magick::Image::addNoise( const NoiseType noiseType_ )
328{
329 ExceptionInfo exceptionInfo;
330 GetExceptionInfo( &exceptionInfo );
331 MagickCore::Image* newImage =
332 AddNoiseImage ( image(),
cristy9ed1f812011-10-08 02:00:08 +0000333 noiseType_, 1.0,
cristy3ed852e2009-09-05 21:47:34 +0000334 &exceptionInfo );
335 replaceImage( newImage );
336 throwException( exceptionInfo );
337 (void) DestroyExceptionInfo( &exceptionInfo );
338}
339
340void Magick::Image::addNoiseChannel( const ChannelType channel_,
341 const NoiseType noiseType_ )
342{
343 ExceptionInfo exceptionInfo;
344 GetExceptionInfo( &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +0000345 ChannelType channel_mask = SetPixelChannelMask( image(), channel_);
cristy3ed852e2009-09-05 21:47:34 +0000346 MagickCore::Image* newImage =
cristy490408a2011-07-07 14:42:05 +0000347 AddNoiseImage ( image(),
cristy9ed1f812011-10-08 02:00:08 +0000348 noiseType_, 1.0,
cristy3ed852e2009-09-05 21:47:34 +0000349 &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +0000350 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +0000351 replaceImage( newImage );
352 throwException( exceptionInfo );
353 (void) DestroyExceptionInfo( &exceptionInfo );
354}
355
356// Affine Transform image
357void Magick::Image::affineTransform ( const DrawableAffine &affine_ )
358{
359 ExceptionInfo exceptionInfo;
360 GetExceptionInfo( &exceptionInfo );
361
362 AffineMatrix _affine;
363 _affine.sx = affine_.sx();
364 _affine.sy = affine_.sy();
365 _affine.rx = affine_.rx();
366 _affine.ry = affine_.ry();
367 _affine.tx = affine_.tx();
368 _affine.ty = affine_.ty();
369
370 MagickCore::Image* newImage =
371 AffineTransformImage( image(), &_affine, &exceptionInfo);
372 replaceImage( newImage );
373 throwException( exceptionInfo );
374 (void) DestroyExceptionInfo( &exceptionInfo );
375}
376
377// Annotate using specified text, and placement location
378void Magick::Image::annotate ( const std::string &text_,
379 const Geometry &location_ )
380{
381 annotate ( text_, location_, NorthWestGravity, 0.0 );
382}
383// Annotate using specified text, bounding area, and placement gravity
384void Magick::Image::annotate ( const std::string &text_,
385 const Geometry &boundingArea_,
386 const GravityType gravity_ )
387{
388 annotate ( text_, boundingArea_, gravity_, 0.0 );
389}
390// Annotate with text using specified text, bounding area, placement
391// gravity, and rotation.
392void Magick::Image::annotate ( const std::string &text_,
393 const Geometry &boundingArea_,
394 const GravityType gravity_,
395 const double degrees_ )
396{
397 modifyImage();
398
399 DrawInfo *drawInfo
400 = options()->drawInfo();
401
402 drawInfo->text = const_cast<char *>(text_.c_str());
403
404 char boundingArea[MaxTextExtent];
405
406 drawInfo->geometry = 0;
407 if ( boundingArea_.isValid() ){
408 if ( boundingArea_.width() == 0 || boundingArea_.height() == 0 )
409 {
cristyb51dff52011-05-19 16:55:47 +0000410 FormatLocaleString( boundingArea, MaxTextExtent, "%+.20g%+.20g",
cristyfe0019b2010-06-07 02:23:32 +0000411 (double) boundingArea_.xOff(), (double) boundingArea_.yOff() );
cristy3ed852e2009-09-05 21:47:34 +0000412 }
413 else
414 {
415 (void) CopyMagickString( boundingArea, string(boundingArea_).c_str(),
416 MaxTextExtent);
417 }
418 drawInfo->geometry = boundingArea;
419 }
420
421 drawInfo->gravity = gravity_;
422
423 AffineMatrix oaffine = drawInfo->affine;
424 if ( degrees_ != 0.0)
425 {
426 AffineMatrix affine;
427 affine.sx=1.0;
428 affine.rx=0.0;
429 affine.ry=0.0;
430 affine.sy=1.0;
431 affine.tx=0.0;
432 affine.ty=0.0;
433
434 AffineMatrix current = drawInfo->affine;
435 affine.sx=cos(DegreesToRadians(fmod(degrees_,360.0)));
436 affine.rx=sin(DegreesToRadians(fmod(degrees_,360.0)));
437 affine.ry=(-sin(DegreesToRadians(fmod(degrees_,360.0))));
438 affine.sy=cos(DegreesToRadians(fmod(degrees_,360.0)));
439
440 drawInfo->affine.sx=current.sx*affine.sx+current.ry*affine.rx;
441 drawInfo->affine.rx=current.rx*affine.sx+current.sy*affine.rx;
442 drawInfo->affine.ry=current.sx*affine.ry+current.ry*affine.sy;
443 drawInfo->affine.sy=current.rx*affine.ry+current.sy*affine.sy;
444 drawInfo->affine.tx=current.sx*affine.tx+current.ry*affine.ty
445 +current.tx;
446 }
447
cristy5cbc0162011-08-29 00:36:28 +0000448 ExceptionInfo exceptionInfo;
449 GetExceptionInfo( &exceptionInfo );
450 AnnotateImage( image(), drawInfo, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000451
452 // Restore original values
453 drawInfo->affine = oaffine;
454 drawInfo->text = 0;
455 drawInfo->geometry = 0;
456
cristy5cbc0162011-08-29 00:36:28 +0000457 throwException( exceptionInfo );
458 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000459}
460// Annotate with text (bounding area is entire image) and placement gravity.
461void Magick::Image::annotate ( const std::string &text_,
462 const GravityType gravity_ )
463{
464 modifyImage();
465
466 DrawInfo *drawInfo
467 = options()->drawInfo();
468
469 drawInfo->text = const_cast<char *>(text_.c_str());
470
471 drawInfo->gravity = gravity_;
472
cristy5cbc0162011-08-29 00:36:28 +0000473 ExceptionInfo exceptionInfo;
474 GetExceptionInfo( &exceptionInfo );
475 AnnotateImage( image(), drawInfo, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000476
477 drawInfo->gravity = NorthWestGravity;
478 drawInfo->text = 0;
479
cristy5cbc0162011-08-29 00:36:28 +0000480 throwException( exceptionInfo );
481 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000482}
483
484// Blur image
485void Magick::Image::blur( const double radius_, const double sigma_ )
486{
487 ExceptionInfo exceptionInfo;
488 GetExceptionInfo( &exceptionInfo );
489 MagickCore::Image* newImage =
cristyaa2c16c2012-03-25 22:21:35 +0000490 BlurImage( image(), radius_, sigma_, &exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +0000491 replaceImage( newImage );
492 throwException( exceptionInfo );
493 (void) DestroyExceptionInfo( &exceptionInfo );
494}
495
496void Magick::Image::blurChannel( const ChannelType channel_,
497 const double radius_, const double sigma_ )
498{
499 ExceptionInfo exceptionInfo;
500 GetExceptionInfo( &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +0000501 ChannelType channel_mask = SetPixelChannelMask( image(), channel_ );
cristy3ed852e2009-09-05 21:47:34 +0000502 MagickCore::Image* newImage =
cristyaa2c16c2012-03-25 22:21:35 +0000503 BlurImage( image(), radius_, sigma_, &exceptionInfo);
cristye2a912b2011-12-05 20:02:07 +0000504 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +0000505 replaceImage( newImage );
506 throwException( exceptionInfo );
507 (void) DestroyExceptionInfo( &exceptionInfo );
508}
509
510// Add border to image
511// Only uses width & height
512void Magick::Image::border( const Geometry &geometry_ )
513{
514 RectangleInfo borderInfo = geometry_;
515 ExceptionInfo exceptionInfo;
516 GetExceptionInfo( &exceptionInfo );
517 MagickCore::Image* newImage =
cristy633f0c62011-09-15 13:27:36 +0000518 BorderImage( image(), &borderInfo, image()->compose, &exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +0000519 replaceImage( newImage );
520 throwException( exceptionInfo );
521 (void) DestroyExceptionInfo( &exceptionInfo );
522}
523
524// Extract channel from image
525void Magick::Image::channel ( const ChannelType channel_ )
526{
527 modifyImage();
cristye941a752011-10-15 01:52:48 +0000528 ExceptionInfo exceptionInfo;
529 GetExceptionInfo( &exceptionInfo );
cristyc8d63672012-01-11 13:03:13 +0000530 MagickCore::Image* newImage =
531 SeparateImage( image(), channel_, &exceptionInfo);
532 replaceImage( newImage );
cristye941a752011-10-15 01:52:48 +0000533 throwException( exceptionInfo );
534 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000535}
536
537// Set or obtain modulus channel depth
cristyfefab1b2011-07-05 00:33:22 +0000538void Magick::Image::channelDepth ( const size_t depth_ )
cristy3ed852e2009-09-05 21:47:34 +0000539{
540 modifyImage();
cristy8a11cb12011-10-19 23:53:34 +0000541 ExceptionInfo exceptionInfo;
542 GetExceptionInfo( &exceptionInfo );
543 SetImageDepth( image(), depth_, &exceptionInfo);
544 throwException( exceptionInfo );
545 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000546}
cristyfefab1b2011-07-05 00:33:22 +0000547size_t Magick::Image::channelDepth ( )
cristy3ed852e2009-09-05 21:47:34 +0000548{
cristyeaedf062010-05-29 22:36:02 +0000549 size_t channel_depth;
cristy3ed852e2009-09-05 21:47:34 +0000550
551 ExceptionInfo exceptionInfo;
552 GetExceptionInfo( &exceptionInfo );
cristyfefab1b2011-07-05 00:33:22 +0000553 channel_depth=GetImageDepth( constImage(), &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000554 throwException( exceptionInfo );
555 (void) DestroyExceptionInfo( &exceptionInfo );
556 return channel_depth;
557}
558
559
560// Charcoal-effect image
561void Magick::Image::charcoal( const double radius_, const double sigma_ )
562{
563 ExceptionInfo exceptionInfo;
564 GetExceptionInfo( &exceptionInfo );
565 MagickCore::Image* newImage =
cristyaa2c16c2012-03-25 22:21:35 +0000566 CharcoalImage( image(), radius_, sigma_, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000567 replaceImage( newImage );
568 throwException( exceptionInfo );
569 (void) DestroyExceptionInfo( &exceptionInfo );
570}
571
572// Chop image
573void Magick::Image::chop( const Geometry &geometry_ )
574{
575 RectangleInfo chopInfo = geometry_;
576 ExceptionInfo exceptionInfo;
577 GetExceptionInfo( &exceptionInfo );
578 MagickCore::Image* newImage =
579 ChopImage( image(), &chopInfo, &exceptionInfo);
580 replaceImage( newImage );
581 throwException( exceptionInfo );
582 (void) DestroyExceptionInfo( &exceptionInfo );
583}
584
cristyb32b90a2009-09-07 21:45:48 +0000585// contains one or more color corrections and applies the correction to the
586// image.
587void Magick::Image::cdl ( const std::string &cdl_ )
588{
589 modifyImage();
cristy1bfa9f02011-08-11 02:35:43 +0000590 ExceptionInfo exceptionInfo;
591 GetExceptionInfo( &exceptionInfo );
592 (void) ColorDecisionListImage( image(), cdl_.c_str(), &exceptionInfo );
593 throwException( exceptionInfo );
594 (void) DestroyExceptionInfo( &exceptionInfo );
cristyb32b90a2009-09-07 21:45:48 +0000595}
596
cristy3ed852e2009-09-05 21:47:34 +0000597// Colorize
cristy4c08aed2011-07-01 19:47:50 +0000598void Magick::Image::colorize ( const unsigned int alphaRed_,
599 const unsigned int alphaGreen_,
600 const unsigned int alphaBlue_,
cristy3ed852e2009-09-05 21:47:34 +0000601 const Color &penColor_ )
602{
603 if ( !penColor_.isValid() )
604 {
605 throwExceptionExplicit( OptionError,
606 "Pen color argument is invalid");
607 }
608
cristyc7e6ff62011-10-03 13:46:11 +0000609 char blend[MaxTextExtent];
610 FormatLocaleString(blend,MaxTextExtent,"%u/%u/%u",alphaRed_,alphaGreen_,alphaBlue_);
cristy3ed852e2009-09-05 21:47:34 +0000611
612 ExceptionInfo exceptionInfo;
613 GetExceptionInfo( &exceptionInfo );
cristyc7e6ff62011-10-03 13:46:11 +0000614 PixelInfo target;
615 GetPixelInfo(image(),&target);
cristy101ab702011-10-13 13:06:32 +0000616 PixelInfo pixel=static_cast<PixelInfo>(penColor_);
cristyc7e6ff62011-10-03 13:46:11 +0000617 target.red=pixel.red;
618 target.green=pixel.green;
619 target.blue=pixel.blue;
620 target.alpha=pixel.alpha;
cristy3ed852e2009-09-05 21:47:34 +0000621 MagickCore::Image* newImage =
cristyc7e6ff62011-10-03 13:46:11 +0000622 ColorizeImage ( image(), blend, &target, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000623 replaceImage( newImage );
624 throwException( exceptionInfo );
625 (void) DestroyExceptionInfo( &exceptionInfo );
626}
cristy4c08aed2011-07-01 19:47:50 +0000627void Magick::Image::colorize ( const unsigned int alpha_,
cristy3ed852e2009-09-05 21:47:34 +0000628 const Color &penColor_ )
629{
cristy4c08aed2011-07-01 19:47:50 +0000630 colorize( alpha_, alpha_, alpha_, penColor_ );
cristy3ed852e2009-09-05 21:47:34 +0000631}
632
cristy735e8942010-04-02 20:32:57 +0000633// Apply a color matrix to the image channels. The user supplied
634// matrix may be of order 1 to 6 (1x1 through 6x6).
cristyeaedf062010-05-29 22:36:02 +0000635void Magick::Image::colorMatrix (const size_t order_,
cristyc8918bb2010-04-03 01:57:27 +0000636 const double *color_matrix_)
cristy735e8942010-04-02 20:32:57 +0000637{
cristyc8918bb2010-04-03 01:57:27 +0000638 KernelInfo
639 *kernel_info;
640
cristy735e8942010-04-02 20:32:57 +0000641 ExceptionInfo exceptionInfo;
642 GetExceptionInfo( &exceptionInfo );
cristy5e6be1e2011-07-16 01:23:39 +0000643 kernel_info=AcquireKernelInfo((const char *) NULL);
cristyc8918bb2010-04-03 01:57:27 +0000644 kernel_info->width=order_;
645 kernel_info->height=order_;
cristyd8c73242012-02-10 17:52:19 +0000646 kernel_info->values=(double *) color_matrix_;
cristy735e8942010-04-02 20:32:57 +0000647 MagickCore::Image* newImage =
cristyc8918bb2010-04-03 01:57:27 +0000648 ColorMatrixImage( image(), kernel_info, &exceptionInfo );
cristyd8c73242012-02-10 17:52:19 +0000649 kernel_info->values=(double *) NULL;
cristyc8918bb2010-04-03 01:57:27 +0000650 kernel_info=DestroyKernelInfo(kernel_info);
cristy735e8942010-04-02 20:32:57 +0000651 replaceImage( newImage );
652 throwException( exceptionInfo );
653 (void) DestroyExceptionInfo( &exceptionInfo );
654}
655
cristy3ed852e2009-09-05 21:47:34 +0000656// Compare current image with another image
657// Sets meanErrorPerPixel, normalizedMaxError, and normalizedMeanError
658// in the current image. False is returned if the images are identical.
659bool Magick::Image::compare ( const Image &reference_ )
660{
cristy018f07f2011-09-04 21:15:19 +0000661 ExceptionInfo exceptionInfo;
662 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000663 modifyImage();
664 Image ref = reference_;
665 ref.modifyImage();
cristy018f07f2011-09-04 21:15:19 +0000666 bool status =
667 static_cast<bool>(IsImagesEqual(image(), ref.image(), &exceptionInfo));
668 throwException( exceptionInfo );
669 (void) DestroyExceptionInfo( &exceptionInfo );
670 return status;
cristy3ed852e2009-09-05 21:47:34 +0000671}
672
673// Composite two images
674void Magick::Image::composite ( const Image &compositeImage_,
cristyd99b0962010-05-29 23:14:26 +0000675 const ssize_t xOffset_,
676 const ssize_t yOffset_,
cristy3ed852e2009-09-05 21:47:34 +0000677 const CompositeOperator compose_ )
678{
679 // Image supplied as compositeImage is composited with current image and
680 // results in updating current image.
681 modifyImage();
682
cristye941a752011-10-15 01:52:48 +0000683 ExceptionInfo exceptionInfo;
684 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000685 CompositeImage( image(),
cristy3ed852e2009-09-05 21:47:34 +0000686 compositeImage_.constImage(),
cristyfeb3e962012-03-29 17:25:55 +0000687 compose_, MagickFalse,
cristy3ed852e2009-09-05 21:47:34 +0000688 xOffset_,
cristye941a752011-10-15 01:52:48 +0000689 yOffset_, &exceptionInfo );
690 throwException( exceptionInfo );
691 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000692}
693void Magick::Image::composite ( const Image &compositeImage_,
694 const Geometry &offset_,
695 const CompositeOperator compose_ )
696{
697 modifyImage();
698
cristybb503372010-05-27 20:51:26 +0000699 ssize_t x = offset_.xOff();
700 ssize_t y = offset_.yOff();
701 size_t width = columns();
702 size_t height = rows();
cristy3ed852e2009-09-05 21:47:34 +0000703
704 ParseMetaGeometry (static_cast<std::string>(offset_).c_str(),
705 &x, &y,
706 &width, &height );
707
cristye941a752011-10-15 01:52:48 +0000708 ExceptionInfo exceptionInfo;
709 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000710 CompositeImage( image(),
cristy3ed852e2009-09-05 21:47:34 +0000711 compositeImage_.constImage(),
cristyfeb3e962012-03-29 17:25:55 +0000712 compose_, MagickFalse,
cristye941a752011-10-15 01:52:48 +0000713 x, y, &exceptionInfo );
714 throwException( exceptionInfo );
715 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000716}
717void Magick::Image::composite ( const Image &compositeImage_,
718 const GravityType gravity_,
719 const CompositeOperator compose_ )
720{
721 modifyImage();
722
723 RectangleInfo geometry;
724
725 SetGeometry(compositeImage_.constImage(), &geometry);
726 GravityAdjustGeometry(columns(), rows(), gravity_, &geometry);
727
cristye941a752011-10-15 01:52:48 +0000728 ExceptionInfo exceptionInfo;
729 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000730 CompositeImage( image(),
cristy3ed852e2009-09-05 21:47:34 +0000731 compositeImage_.constImage(),
cristyfeb3e962012-03-29 17:25:55 +0000732 compose_, MagickFalse,
cristye941a752011-10-15 01:52:48 +0000733 geometry.x, geometry.y, &exceptionInfo );
734 throwException( exceptionInfo );
735 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000736}
737
738// Contrast image
cristyeaedf062010-05-29 22:36:02 +0000739void Magick::Image::contrast ( const size_t sharpen_ )
cristy3ed852e2009-09-05 21:47:34 +0000740{
741 modifyImage();
cristye23ec9d2011-08-16 18:15:40 +0000742 ExceptionInfo exceptionInfo;
743 GetExceptionInfo( &exceptionInfo );
744 ContrastImage ( image(), (MagickBooleanType) sharpen_, &exceptionInfo );
745 throwException( exceptionInfo );
746 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000747}
748
749// Convolve image. Applies a general image convolution kernel to the image.
750// order_ represents the number of columns and rows in the filter kernel.
751// kernel_ is an array of doubles representing the convolution kernel.
cristyeaedf062010-05-29 22:36:02 +0000752void Magick::Image::convolve ( const size_t order_,
cristy3ed852e2009-09-05 21:47:34 +0000753 const double *kernel_ )
754{
cristy5e6be1e2011-07-16 01:23:39 +0000755 KernelInfo
756 *kernel_info;
757
cristy3ed852e2009-09-05 21:47:34 +0000758 ExceptionInfo exceptionInfo;
759 GetExceptionInfo( &exceptionInfo );
cristy5e6be1e2011-07-16 01:23:39 +0000760 kernel_info=AcquireKernelInfo((const char *) NULL);
761 kernel_info->width=order_;
762 kernel_info->height=order_;
cristya96f2492011-12-14 18:25:41 +0000763 kernel_info->values=(MagickRealType *) kernel_;
cristy3ed852e2009-09-05 21:47:34 +0000764 MagickCore::Image* newImage =
cristy5e6be1e2011-07-16 01:23:39 +0000765 ConvolveImage ( image(), kernel_info, &exceptionInfo );
cristya96f2492011-12-14 18:25:41 +0000766 kernel_info->values=(MagickRealType *) NULL;
cristy5e6be1e2011-07-16 01:23:39 +0000767 kernel_info=DestroyKernelInfo(kernel_info);
cristy3ed852e2009-09-05 21:47:34 +0000768 replaceImage( newImage );
769 throwException( exceptionInfo );
770 (void) DestroyExceptionInfo( &exceptionInfo );
771}
772
773// Crop image
774void Magick::Image::crop ( const Geometry &geometry_ )
775{
776 RectangleInfo cropInfo = geometry_;
777 ExceptionInfo exceptionInfo;
778 GetExceptionInfo( &exceptionInfo );
779 MagickCore::Image* newImage =
780 CropImage( image(),
781 &cropInfo,
782 &exceptionInfo);
783 replaceImage( newImage );
784 throwException( exceptionInfo );
785 (void) DestroyExceptionInfo( &exceptionInfo );
786}
787
788// Cycle Color Map
cristyd99b0962010-05-29 23:14:26 +0000789void Magick::Image::cycleColormap ( const ssize_t amount_ )
cristy3ed852e2009-09-05 21:47:34 +0000790{
cristy018f07f2011-09-04 21:15:19 +0000791 ExceptionInfo exceptionInfo;
792 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000793 modifyImage();
cristy018f07f2011-09-04 21:15:19 +0000794 CycleColormapImage( image(), amount_, &exceptionInfo );
795 throwException( exceptionInfo );
796 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000797}
798
799// Despeckle
800void Magick::Image::despeckle ( void )
801{
802 ExceptionInfo exceptionInfo;
803 GetExceptionInfo( &exceptionInfo );
804 MagickCore::Image* newImage =
805 DespeckleImage( image(), &exceptionInfo );
806 replaceImage( newImage );
807 throwException( exceptionInfo );
808 (void) DestroyExceptionInfo( &exceptionInfo );
809}
810
811// Display image
812void Magick::Image::display( void )
813{
cristy051718b2011-08-28 22:49:25 +0000814 ExceptionInfo exceptionInfo;
815 GetExceptionInfo( &exceptionInfo );
816 DisplayImages( imageInfo(), image(), &exceptionInfo );
817 throwException( exceptionInfo );
818 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000819}
820
821// Distort image. distorts an image using various distortion methods, by
822// mapping color lookups of the source image to a new destination image
823// usally of the same size as the source image, unless 'bestfit' is set to
824// true.
cristyb32b90a2009-09-07 21:45:48 +0000825void Magick::Image::distort ( const DistortImageMethod method_,
cristybb503372010-05-27 20:51:26 +0000826 const size_t number_arguments_,
cristyb32b90a2009-09-07 21:45:48 +0000827 const double *arguments_,
828 const bool bestfit_ )
cristy3ed852e2009-09-05 21:47:34 +0000829{
830 ExceptionInfo exceptionInfo;
831 GetExceptionInfo( &exceptionInfo );
cristyb32b90a2009-09-07 21:45:48 +0000832 MagickCore::Image* newImage = DistortImage ( image(), method_,
833 number_arguments_, arguments_, bestfit_ == true ? MagickTrue : MagickFalse,
834 &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000835 replaceImage( newImage );
836 throwException( exceptionInfo );
837 (void) DestroyExceptionInfo( &exceptionInfo );
838}
839
840// Draw on image using single drawable
841void Magick::Image::draw ( const Magick::Drawable &drawable_ )
842{
843 modifyImage();
844
845 DrawingWand *wand = DrawAllocateWand( options()->drawInfo(), image());
846
847 if(wand)
848 {
849 drawable_.operator()(wand);
850
cristyc82a27b2011-10-21 01:07:16 +0000851 DrawRender(wand);
cristy3ed852e2009-09-05 21:47:34 +0000852
853 wand=DestroyDrawingWand(wand);
854 }
855
856 throwImageException();
857}
858
859// Draw on image using a drawable list
860void Magick::Image::draw ( const std::list<Magick::Drawable> &drawable_ )
861{
862 modifyImage();
863
864 DrawingWand *wand = DrawAllocateWand( options()->drawInfo(), image());
865
866 if(wand)
867 {
868 for( std::list<Magick::Drawable>::const_iterator p = drawable_.begin();
869 p != drawable_.end(); p++ )
870 {
871 p->operator()(wand);
cristy3ed852e2009-09-05 21:47:34 +0000872 }
873
cristyc82a27b2011-10-21 01:07:16 +0000874 DrawRender(wand);
cristy3ed852e2009-09-05 21:47:34 +0000875
876 wand=DestroyDrawingWand(wand);
877 }
878
879 throwImageException();
880}
881
882// Hilight edges in image
cristy8ae632d2011-09-05 17:29:53 +0000883void Magick::Image::edge ( const double radius_, const double sigma_ )
884{
885 ExceptionInfo exceptionInfo;
886 GetExceptionInfo( &exceptionInfo );
887 MagickCore::Image* newImage =
888 EdgeImage( image(), radius_, sigma_, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000889 replaceImage( newImage );
890 throwException( exceptionInfo );
891 (void) DestroyExceptionInfo( &exceptionInfo );
892}
893
894// Emboss image (hilight edges)
895void Magick::Image::emboss ( const double radius_, const double sigma_ )
896{
897 ExceptionInfo exceptionInfo;
898 GetExceptionInfo( &exceptionInfo );
899 MagickCore::Image* newImage =
900 EmbossImage( image(), radius_, sigma_, &exceptionInfo );
901 replaceImage( newImage );
902 throwException( exceptionInfo );
903 (void) DestroyExceptionInfo( &exceptionInfo );
904}
905
906// Enhance image (minimize noise)
907void Magick::Image::enhance ( void )
908{
909 ExceptionInfo exceptionInfo;
910 GetExceptionInfo( &exceptionInfo );
911 MagickCore::Image* newImage =
912 EnhanceImage( image(), &exceptionInfo );
913 replaceImage( newImage );
914 throwException( exceptionInfo );
915 (void) DestroyExceptionInfo( &exceptionInfo );
916}
917
918// Equalize image (histogram equalization)
919void Magick::Image::equalize ( void )
920{
cristy6d8c3d72011-08-22 01:20:01 +0000921 ExceptionInfo exceptionInfo;
922 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000923 modifyImage();
cristy6d8c3d72011-08-22 01:20:01 +0000924 EqualizeImage( image(), &exceptionInfo );
925 throwException( exceptionInfo );
cristyea1a8aa2011-10-20 13:24:06 +0000926 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000927}
928
929// Erase image to current "background color"
930void Magick::Image::erase ( void )
931{
932 modifyImage();
cristyea1a8aa2011-10-20 13:24:06 +0000933 ExceptionInfo exceptionInfo;
934 GetExceptionInfo( &exceptionInfo );
935 SetImageBackgroundColor( image(), &exceptionInfo );
936 throwException( exceptionInfo );
937 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +0000938}
939
940// Extends image as defined by the geometry.
941//
942void Magick::Image::extent ( const Geometry &geometry_ )
943{
944 RectangleInfo extentInfo = geometry_;
945 modifyImage();
cristy8bf9e292010-02-21 17:48:01 +0000946 ExceptionInfo exceptionInfo;
947 GetExceptionInfo( &exceptionInfo );
cristy1ee42c42010-05-12 12:52:50 +0000948 MagickCore::Image* newImage =
949 ExtentImage ( image(), &extentInfo, &exceptionInfo );
950 replaceImage( newImage );
cristy8bf9e292010-02-21 17:48:01 +0000951 throwException( exceptionInfo );
952 (void) DestroyExceptionInfo( &exceptionInfo );
953}
954void Magick::Image::extent ( const Geometry &geometry_, const Color &backgroundColor_ )
955{
956 backgroundColor ( backgroundColor_ );
957 extent ( geometry_ );
958}
cristyff024b42010-02-21 22:55:09 +0000959void Magick::Image::extent ( const Geometry &geometry_, const GravityType gravity_ )
cristy8bf9e292010-02-21 17:48:01 +0000960{
961 image()->gravity = gravity_;
962 extent ( geometry_ );
963}
964void Magick::Image::extent ( const Geometry &geometry_, const Color &backgroundColor_, const GravityType gravity_ )
965{
966 image()->gravity = gravity_;
967 backgroundColor ( backgroundColor_ );
968 extent ( geometry_ );
cristy3ed852e2009-09-05 21:47:34 +0000969}
970
971// Flip image (reflect each scanline in the vertical direction)
972void Magick::Image::flip ( void )
973{
974 ExceptionInfo exceptionInfo;
975 GetExceptionInfo( &exceptionInfo );
976 MagickCore::Image* newImage =
977 FlipImage( image(), &exceptionInfo );
978 replaceImage( newImage );
979 throwException( exceptionInfo );
980 (void) DestroyExceptionInfo( &exceptionInfo );
981}
982
983// Flood-fill color across pixels that match the color of the
984// target pixel and are neighbors of the target pixel.
985// Uses current fuzz setting when determining color match.
cristy35ef8242010-06-03 16:24:13 +0000986void Magick::Image::floodFillColor( const ssize_t x_,
987 const ssize_t y_,
cristy3ed852e2009-09-05 21:47:34 +0000988 const Magick::Color &fillColor_ )
989{
990 floodFillTexture( x_, y_, Image( Geometry( 1, 1), fillColor_ ) );
991}
992void Magick::Image::floodFillColor( const Geometry &point_,
993 const Magick::Color &fillColor_ )
994{
995 floodFillTexture( point_, Image( Geometry( 1, 1), fillColor_) );
996}
997
998// Flood-fill color across pixels starting at target-pixel and
999// stopping at pixels matching specified border color.
1000// Uses current fuzz setting when determining color match.
cristy35ef8242010-06-03 16:24:13 +00001001void Magick::Image::floodFillColor( const ssize_t x_,
1002 const ssize_t y_,
cristy3ed852e2009-09-05 21:47:34 +00001003 const Magick::Color &fillColor_,
1004 const Magick::Color &borderColor_ )
1005{
1006 floodFillTexture( x_, y_, Image( Geometry( 1, 1), fillColor_),
1007 borderColor_ );
1008}
1009void Magick::Image::floodFillColor( const Geometry &point_,
1010 const Magick::Color &fillColor_,
1011 const Magick::Color &borderColor_ )
1012{
1013 floodFillTexture( point_, Image( Geometry( 1, 1), fillColor_),
1014 borderColor_ );
1015}
1016
1017// Floodfill pixels matching color (within fuzz factor) of target
cristy4c08aed2011-07-01 19:47:50 +00001018// pixel(x,y) with replacement alpha value using method.
cristyb6a294d2011-10-03 00:55:17 +00001019void Magick::Image::floodFillAlpha( const ssize_t x_,
cristy35ef8242010-06-03 16:24:13 +00001020 const ssize_t y_,
cristy4c08aed2011-07-01 19:47:50 +00001021 const unsigned int alpha_,
cristy3ed852e2009-09-05 21:47:34 +00001022 const PaintMethod method_ )
1023{
1024 modifyImage();
cristy4c08aed2011-07-01 19:47:50 +00001025 PixelInfo target;
1026 GetPixelInfo(image(),&target);
cristy101ab702011-10-13 13:06:32 +00001027 PixelInfo pixel=static_cast<PixelInfo>(pixelColor(x_,y_));
cristy3ed852e2009-09-05 21:47:34 +00001028 target.red=pixel.red;
1029 target.green=pixel.green;
1030 target.blue=pixel.blue;
cristy4c08aed2011-07-01 19:47:50 +00001031 target.alpha=alpha_;
cristy189e84c2011-08-27 18:08:53 +00001032 ExceptionInfo exceptionInfo;
1033 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001034 FloodfillPaintImage ( image(),
cristy3ed852e2009-09-05 21:47:34 +00001035 options()->drawInfo(), // const DrawInfo *draw_info
1036 &target,
cristybb503372010-05-27 20:51:26 +00001037 static_cast<ssize_t>(x_), static_cast<ssize_t>(y_),
cristy189e84c2011-08-27 18:08:53 +00001038 method_ == FloodfillMethod ? MagickFalse : MagickTrue,
1039 &exceptionInfo);
1040 throwException( exceptionInfo );
1041 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001042}
1043
1044// Flood-fill texture across pixels that match the color of the
1045// target pixel and are neighbors of the target pixel.
1046// Uses current fuzz setting when determining color match.
cristy35ef8242010-06-03 16:24:13 +00001047void Magick::Image::floodFillTexture( const ssize_t x_,
1048 const ssize_t y_,
cristy3ed852e2009-09-05 21:47:34 +00001049 const Magick::Image &texture_ )
1050{
1051 modifyImage();
1052
1053 // Set drawing pattern
1054 options()->fillPattern(texture_.constImage());
1055
1056 // Get pixel view
1057 Pixels pixels(*this);
1058 // Fill image
cristy4c08aed2011-07-01 19:47:50 +00001059 Quantum *p = pixels.get(x_, y_, 1, 1 );
1060 PixelInfo target;
1061 GetPixelInfo(constImage(),&target);
1062 target.red=GetPixelRed(constImage(),p);
1063 target.green=GetPixelGreen(constImage(),p);
1064 target.blue=GetPixelBlue(constImage(),p);
cristy189e84c2011-08-27 18:08:53 +00001065 ExceptionInfo exceptionInfo;
1066 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001067 if (p)
1068 FloodfillPaintImage ( image(), // Image *image
cristy3ed852e2009-09-05 21:47:34 +00001069 options()->drawInfo(), // const DrawInfo *draw_info
1070 &target, // const MagickPacket target
cristybb503372010-05-27 20:51:26 +00001071 static_cast<ssize_t>(x_), // const ssize_t x_offset
1072 static_cast<ssize_t>(y_), // const ssize_t y_offset
cristy189e84c2011-08-27 18:08:53 +00001073 MagickFalse, // const PaintMethod method
1074 &exceptionInfo );
1075 throwException( exceptionInfo );
1076 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001077
cristy3ed852e2009-09-05 21:47:34 +00001078}
1079void Magick::Image::floodFillTexture( const Magick::Geometry &point_,
1080 const Magick::Image &texture_ )
1081{
1082 floodFillTexture( point_.xOff(), point_.yOff(), texture_ );
1083}
1084
1085// Flood-fill texture across pixels starting at target-pixel and
1086// stopping at pixels matching specified border color.
1087// Uses current fuzz setting when determining color match.
cristy35ef8242010-06-03 16:24:13 +00001088void Magick::Image::floodFillTexture( const ssize_t x_,
1089 const ssize_t y_,
cristy3ed852e2009-09-05 21:47:34 +00001090 const Magick::Image &texture_,
1091 const Magick::Color &borderColor_ )
1092{
1093 modifyImage();
1094
1095 // Set drawing fill pattern
1096 options()->fillPattern(texture_.constImage());
1097
cristy4c08aed2011-07-01 19:47:50 +00001098 PixelInfo target;
1099 GetPixelInfo(constImage(),&target);
cristy101ab702011-10-13 13:06:32 +00001100 target.red=static_cast<PixelInfo>(borderColor_).red;
1101 target.green=static_cast<PixelInfo>(borderColor_).green;
1102 target.blue=static_cast<PixelInfo>(borderColor_).blue;
cristy189e84c2011-08-27 18:08:53 +00001103 ExceptionInfo exceptionInfo;
1104 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001105 FloodfillPaintImage ( image(),
cristy3ed852e2009-09-05 21:47:34 +00001106 options()->drawInfo(),
1107 &target,
cristybb503372010-05-27 20:51:26 +00001108 static_cast<ssize_t>(x_),
1109 static_cast<ssize_t>(y_),
cristy189e84c2011-08-27 18:08:53 +00001110 MagickTrue, &exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00001111
cristy189e84c2011-08-27 18:08:53 +00001112 throwException( exceptionInfo );
1113 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001114}
1115void Magick::Image::floodFillTexture( const Magick::Geometry &point_,
1116 const Magick::Image &texture_,
1117 const Magick::Color &borderColor_ )
1118{
1119 floodFillTexture( point_.xOff(), point_.yOff(), texture_, borderColor_ );
1120}
1121
1122// Flop image (reflect each scanline in the horizontal direction)
1123void Magick::Image::flop ( void )
1124{
1125 ExceptionInfo exceptionInfo;
1126 GetExceptionInfo( &exceptionInfo );
1127 MagickCore::Image* newImage =
1128 FlopImage( image(), &exceptionInfo );
1129 replaceImage( newImage );
1130 throwException( exceptionInfo );
1131 (void) DestroyExceptionInfo( &exceptionInfo );
1132}
1133
1134// Frame image
1135void Magick::Image::frame ( const Geometry &geometry_ )
1136{
1137 FrameInfo info;
1138
cristybb503372010-05-27 20:51:26 +00001139 info.x = static_cast<ssize_t>(geometry_.width());
1140 info.y = static_cast<ssize_t>(geometry_.height());
1141 info.width = columns() + ( static_cast<size_t>(info.x) << 1 );
1142 info.height = rows() + ( static_cast<size_t>(info.y) << 1 );
cristy3ed852e2009-09-05 21:47:34 +00001143 info.outer_bevel = geometry_.xOff();
1144 info.inner_bevel = geometry_.yOff();
1145
1146 ExceptionInfo exceptionInfo;
1147 GetExceptionInfo( &exceptionInfo );
1148 MagickCore::Image* newImage =
cristy633f0c62011-09-15 13:27:36 +00001149 FrameImage( image(), &info, image()->compose, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001150 replaceImage( newImage );
1151 throwException( exceptionInfo );
1152 (void) DestroyExceptionInfo( &exceptionInfo );
1153}
cristyeaedf062010-05-29 22:36:02 +00001154void Magick::Image::frame ( const size_t width_,
1155 const size_t height_,
cristyd99b0962010-05-29 23:14:26 +00001156 const ssize_t outerBevel_, const ssize_t innerBevel_ )
cristy3ed852e2009-09-05 21:47:34 +00001157{
1158 FrameInfo info;
cristybb503372010-05-27 20:51:26 +00001159 info.x = static_cast<ssize_t>(width_);
1160 info.y = static_cast<ssize_t>(height_);
1161 info.width = columns() + ( static_cast<size_t>(info.x) << 1 );
1162 info.height = rows() + ( static_cast<size_t>(info.y) << 1 );
1163 info.outer_bevel = static_cast<ssize_t>(outerBevel_);
1164 info.inner_bevel = static_cast<ssize_t>(innerBevel_);
cristy3ed852e2009-09-05 21:47:34 +00001165
1166 ExceptionInfo exceptionInfo;
1167 GetExceptionInfo( &exceptionInfo );
1168 MagickCore::Image* newImage =
cristy633f0c62011-09-15 13:27:36 +00001169 FrameImage( image(), &info, image()->compose, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001170 replaceImage( newImage );
1171 throwException( exceptionInfo );
1172 (void) DestroyExceptionInfo( &exceptionInfo );
1173}
1174
cristyc9550792009-11-13 20:05:42 +00001175// Fx image. Applies a mathematical expression to the image.
1176void Magick::Image::fx ( const std::string expression )
1177{
1178 ExceptionInfo exceptionInfo;
1179 GetExceptionInfo( &exceptionInfo );
1180 MagickCore::Image* newImage =
cristy490408a2011-07-07 14:42:05 +00001181 FxImage ( image(), expression.c_str(), &exceptionInfo );
cristyc9550792009-11-13 20:05:42 +00001182 replaceImage( newImage );
1183 throwException( exceptionInfo );
1184 (void) DestroyExceptionInfo( &exceptionInfo );
1185}
cristy3ed852e2009-09-05 21:47:34 +00001186void Magick::Image::fx ( const std::string expression,
1187 const Magick::ChannelType channel )
1188{
1189 ExceptionInfo exceptionInfo;
1190 GetExceptionInfo( &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +00001191 ChannelType channel_mask = SetPixelChannelMask( image(), channel );
cristy3ed852e2009-09-05 21:47:34 +00001192 MagickCore::Image* newImage =
cristy490408a2011-07-07 14:42:05 +00001193 FxImage ( image(), expression.c_str(), &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +00001194 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +00001195 replaceImage( newImage );
1196 throwException( exceptionInfo );
1197 (void) DestroyExceptionInfo( &exceptionInfo );
1198}
1199
1200// Gamma correct image
1201void Magick::Image::gamma ( const double gamma_ )
1202{
cristyb3e7c6c2011-07-24 01:43:55 +00001203 ExceptionInfo exceptionInfo;
1204 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001205 modifyImage();
cristyb3e7c6c2011-07-24 01:43:55 +00001206 GammaImage ( image(), gamma_, &exceptionInfo );
1207 throwException( exceptionInfo );
1208 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001209}
1210
1211void Magick::Image::gamma ( const double gammaRed_,
1212 const double gammaGreen_,
1213 const double gammaBlue_ )
1214{
1215 char gamma[MaxTextExtent + 1];
cristyb51dff52011-05-19 16:55:47 +00001216 FormatLocaleString( gamma, MaxTextExtent, "%3.6f/%3.6f/%3.6f/",
cristy3ed852e2009-09-05 21:47:34 +00001217 gammaRed_, gammaGreen_, gammaBlue_);
1218
cristyb3e7c6c2011-07-24 01:43:55 +00001219 ExceptionInfo exceptionInfo;
1220 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001221 modifyImage();
cristyb3e7c6c2011-07-24 01:43:55 +00001222 GammaImage ( image(), atof(gamma), &exceptionInfo );
1223 throwException( exceptionInfo );
1224 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001225}
1226
1227// Gaussian blur image
1228// The number of neighbor pixels to be included in the convolution
1229// mask is specified by 'width_'. The standard deviation of the
1230// gaussian bell curve is specified by 'sigma_'.
1231void Magick::Image::gaussianBlur ( const double width_, const double sigma_ )
1232{
1233 ExceptionInfo exceptionInfo;
1234 GetExceptionInfo( &exceptionInfo );
1235 MagickCore::Image* newImage =
cristyd89705a2012-01-20 02:52:24 +00001236 GaussianBlurImage( image(), width_, sigma_, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001237 replaceImage( newImage );
1238 throwException( exceptionInfo );
1239 (void) DestroyExceptionInfo( &exceptionInfo );
1240}
1241
1242void Magick::Image::gaussianBlurChannel ( const ChannelType channel_,
1243 const double width_,
1244 const double sigma_ )
1245{
1246 ExceptionInfo exceptionInfo;
1247 GetExceptionInfo( &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +00001248 ChannelType channel_mask = SetPixelChannelMask( image(), channel_ );
cristy3ed852e2009-09-05 21:47:34 +00001249 MagickCore::Image* newImage =
cristyd89705a2012-01-20 02:52:24 +00001250 GaussianBlurImage( image(), width_, sigma_, &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +00001251 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +00001252 replaceImage( newImage );
1253 throwException( exceptionInfo );
1254 (void) DestroyExceptionInfo( &exceptionInfo );
1255}
1256
cristyb32b90a2009-09-07 21:45:48 +00001257// Apply a color lookup table (Hald CLUT) to the image.
1258void Magick::Image::haldClut ( const Image &clutImage_ )
1259{
cristy7c0a0a42011-08-23 17:57:25 +00001260 ExceptionInfo exceptionInfo;
1261 GetExceptionInfo( &exceptionInfo );
cristyb32b90a2009-09-07 21:45:48 +00001262 modifyImage();
cristy7c0a0a42011-08-23 17:57:25 +00001263 (void) HaldClutImage( image(), clutImage_.constImage(), &exceptionInfo );
1264 throwException( exceptionInfo );
1265 (void) DestroyExceptionInfo( &exceptionInfo );
cristyb32b90a2009-09-07 21:45:48 +00001266}
1267
cristy3ed852e2009-09-05 21:47:34 +00001268// Implode image
1269void Magick::Image::implode ( const double factor_ )
1270{
1271 ExceptionInfo exceptionInfo;
1272 GetExceptionInfo( &exceptionInfo );
1273 MagickCore::Image* newImage =
cristy76f512e2011-09-12 01:26:56 +00001274 ImplodeImage( image(), factor_, image()->interpolate, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001275 replaceImage( newImage );
1276 throwException( exceptionInfo );
1277 (void) DestroyExceptionInfo( &exceptionInfo );
1278}
1279
cristy529fcc22009-11-14 18:15:08 +00001280// implements the inverse discrete Fourier transform (IFT) of the image either
1281// as a magnitude / phase or real / imaginary image pair.
1282void Magick::Image::inverseFourierTransform ( const Image &phase_ )
1283{
1284 ExceptionInfo exceptionInfo;
1285 GetExceptionInfo( &exceptionInfo );
1286 MagickCore::Image* newImage = InverseFourierTransformImage( image(),
1287 phase_.constImage(), MagickTrue, &exceptionInfo);
1288 replaceImage( newImage );
1289 throwException( exceptionInfo );
1290 (void) DestroyExceptionInfo( &exceptionInfo );
1291}
1292void Magick::Image::inverseFourierTransform ( const Image &phase_,
1293 const bool magnitude_ )
1294{
1295 ExceptionInfo exceptionInfo;
1296 GetExceptionInfo( &exceptionInfo );
1297 MagickCore::Image* newImage = InverseFourierTransformImage( image(),
1298 phase_.constImage(), magnitude_ == true ? MagickTrue : MagickFalse,
1299 &exceptionInfo);
1300 replaceImage( newImage );
1301 throwException( exceptionInfo );
1302 (void) DestroyExceptionInfo( &exceptionInfo );
1303}
1304
cristy3ed852e2009-09-05 21:47:34 +00001305// Level image. Adjust the levels of the image by scaling the colors
1306// falling between specified white and black points to the full
1307// available quantum range. The parameters provided represent the
1308// black, mid (gamma), and white points. The black point specifies
1309// the darkest color in the image. Colors darker than the black point
1310// are set to zero. Mid point (gamma) specifies a gamma correction to
1311// apply to the image. White point specifies the lightest color in the
1312// image. Colors brighter than the white point are set to the maximum
1313// quantum value. The black and white point have the valid range 0 to
1314// QuantumRange while gamma has a useful range of 0 to ten.
1315void Magick::Image::level ( const double black_point,
1316 const double white_point,
1317 const double gamma )
1318{
cristy01e9afd2011-08-10 17:38:41 +00001319 ExceptionInfo exceptionInfo;
1320 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001321 modifyImage();
cristy01e9afd2011-08-10 17:38:41 +00001322 (void) LevelImage( image(), black_point, white_point, gamma, &exceptionInfo );
1323 throwException( exceptionInfo );
1324 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001325}
1326
1327// Magnify image by integral size
1328void Magick::Image::magnify ( void )
1329{
1330 ExceptionInfo exceptionInfo;
1331 GetExceptionInfo( &exceptionInfo );
1332 MagickCore::Image* newImage =
1333 MagnifyImage( image(), &exceptionInfo );
1334 replaceImage( newImage );
1335 throwException( exceptionInfo );
1336 (void) DestroyExceptionInfo( &exceptionInfo );
1337}
1338
1339// Remap image colors with closest color from reference image
1340void Magick::Image::map ( const Image &mapImage_ , const bool dither_ )
1341{
cristy018f07f2011-09-04 21:15:19 +00001342 ExceptionInfo exceptionInfo;
1343 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001344 modifyImage();
1345 options()->quantizeDither( dither_ );
1346 RemapImage ( options()->quantizeInfo(), image(),
cristy018f07f2011-09-04 21:15:19 +00001347 mapImage_.constImage(), &exceptionInfo);
1348 throwException( exceptionInfo );
1349 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001350}
cristy4c08aed2011-07-01 19:47:50 +00001351// Floodfill designated area with replacement alpha value
cristy3ed852e2009-09-05 21:47:34 +00001352void Magick::Image::matteFloodfill ( const Color &target_ ,
cristy4c08aed2011-07-01 19:47:50 +00001353 const unsigned int alpha_,
cristyd99b0962010-05-29 23:14:26 +00001354 const ssize_t x_, const ssize_t y_,
cristy3ed852e2009-09-05 21:47:34 +00001355 const Magick::PaintMethod method_ )
1356{
1357 modifyImage();
cristy4c08aed2011-07-01 19:47:50 +00001358 PixelInfo target;
1359 GetPixelInfo(constImage(),&target);
cristy101ab702011-10-13 13:06:32 +00001360 target.red=static_cast<PixelInfo>(target_).red;
1361 target.green=static_cast<PixelInfo>(target_).green;
1362 target.blue=static_cast<PixelInfo>(target_).blue;
cristy4c08aed2011-07-01 19:47:50 +00001363 target.alpha=alpha_;
cristybd5a96c2011-08-21 00:04:26 +00001364 ChannelType channel_mask = SetPixelChannelMask( image(), AlphaChannel );
cristy189e84c2011-08-27 18:08:53 +00001365 ExceptionInfo exceptionInfo;
1366 GetExceptionInfo( &exceptionInfo );
cristyd42d9952011-07-08 14:21:50 +00001367 FloodfillPaintImage ( image(), options()->drawInfo(), &target, x_, y_,
cristy189e84c2011-08-27 18:08:53 +00001368 method_ == FloodfillMethod ? MagickFalse : MagickTrue, &exceptionInfo);
cristye2a912b2011-12-05 20:02:07 +00001369 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy189e84c2011-08-27 18:08:53 +00001370 throwException( exceptionInfo );
1371 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001372}
1373
1374// Filter image by replacing each pixel component with the median
1375// color in a circular neighborhood
1376void Magick::Image::medianFilter ( const double radius_ )
1377{
1378 ExceptionInfo exceptionInfo;
1379 GetExceptionInfo( &exceptionInfo );
1380 MagickCore::Image* newImage =
cristy95c38342011-03-18 22:39:51 +00001381 StatisticImage ( image(), MedianStatistic, (size_t) radius_, (size_t)
1382 radius_,&exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001383 replaceImage( newImage );
1384 throwException( exceptionInfo );
1385 (void) DestroyExceptionInfo( &exceptionInfo );
1386}
1387
1388// Reduce image by integral size
1389void Magick::Image::minify ( void )
1390{
1391 ExceptionInfo exceptionInfo;
1392 GetExceptionInfo( &exceptionInfo );
1393 MagickCore::Image* newImage =
1394 MinifyImage( image(), &exceptionInfo );
1395 replaceImage( newImage );
1396 throwException( exceptionInfo );
1397 (void) DestroyExceptionInfo( &exceptionInfo );
1398}
1399
1400// Modulate percent hue, saturation, and brightness of an image
1401void Magick::Image::modulate ( const double brightness_,
1402 const double saturation_,
1403 const double hue_ )
1404{
1405 char modulate[MaxTextExtent + 1];
cristyb51dff52011-05-19 16:55:47 +00001406 FormatLocaleString( modulate, MaxTextExtent, "%3.6f,%3.6f,%3.6f",
cristy3ed852e2009-09-05 21:47:34 +00001407 brightness_, saturation_, hue_);
1408
cristy33bd5152011-08-24 01:42:24 +00001409 ExceptionInfo exceptionInfo;
1410 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001411 modifyImage();
cristy33bd5152011-08-24 01:42:24 +00001412 ModulateImage( image(), modulate, &exceptionInfo );
1413 throwException( exceptionInfo );
1414 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001415}
1416
1417// Motion blur image with specified blur factor
1418// The radius_ parameter specifies the radius of the Gaussian, in
1419// pixels, not counting the center pixel. The sigma_ parameter
1420// specifies the standard deviation of the Laplacian, in pixels.
1421// The angle_ parameter specifies the angle the object appears
1422// to be comming from (zero degrees is from the right).
1423void Magick::Image::motionBlur ( const double radius_,
1424 const double sigma_,
1425 const double angle_ )
1426{
1427 ExceptionInfo exceptionInfo;
1428 GetExceptionInfo( &exceptionInfo );
1429 MagickCore::Image* newImage =
cristyaa2c16c2012-03-25 22:21:35 +00001430 MotionBlurImage( image(), radius_, sigma_, angle_, &exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00001431 replaceImage( newImage );
1432 throwException( exceptionInfo );
1433 (void) DestroyExceptionInfo( &exceptionInfo );
1434}
1435
1436// Negate image. Set grayscale_ to true to effect grayscale values
1437// only
1438void Magick::Image::negate ( const bool grayscale_ )
1439{
cristyb3e7c6c2011-07-24 01:43:55 +00001440 ExceptionInfo exceptionInfo;
1441 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001442 modifyImage();
cristyb3e7c6c2011-07-24 01:43:55 +00001443 NegateImage ( image(), grayscale_ == true ? MagickTrue : MagickFalse,
1444 &exceptionInfo );
1445 throwException( exceptionInfo );
1446 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001447}
1448
1449// Normalize image
1450void Magick::Image::normalize ( void )
1451{
1452 modifyImage();
cristye23ec9d2011-08-16 18:15:40 +00001453 ExceptionInfo exceptionInfo;
1454 GetExceptionInfo( &exceptionInfo );
1455 NormalizeImage ( image(), &exceptionInfo );
1456 throwException( exceptionInfo );
1457 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001458}
1459
1460// Oilpaint image
cristy14973ba2011-08-27 23:48:07 +00001461void Magick::Image::oilPaint ( const double radius_, const double sigma_ )
cristy3ed852e2009-09-05 21:47:34 +00001462{
1463 ExceptionInfo exceptionInfo;
1464 GetExceptionInfo( &exceptionInfo );
1465 MagickCore::Image* newImage =
cristy14973ba2011-08-27 23:48:07 +00001466 OilPaintImage( image(), radius_, sigma_, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001467 replaceImage( newImage );
1468 throwException( exceptionInfo );
1469 (void) DestroyExceptionInfo( &exceptionInfo );
1470}
1471
cristy4c08aed2011-07-01 19:47:50 +00001472// Set or attenuate the alpha channel. If the image pixels are
1473// opaque then they are set to the specified alpha value, otherwise
1474// they are blended with the supplied alpha value. The value of
1475// alpha_ ranges from 0 (completely opaque) to QuantumRange. The defines
1476// OpaqueAlpha and TransparentAlpha are available to specify
cristy3ed852e2009-09-05 21:47:34 +00001477// completely opaque or completely transparent, respectively.
cristy4c08aed2011-07-01 19:47:50 +00001478void Magick::Image::alpha ( const unsigned int alpha_ )
cristy3ed852e2009-09-05 21:47:34 +00001479{
1480 modifyImage();
cristye941a752011-10-15 01:52:48 +00001481 ExceptionInfo exceptionInfo;
1482 GetExceptionInfo( &exceptionInfo );
1483 SetImageAlpha( image(), alpha_, &exceptionInfo );
1484 throwException( exceptionInfo );
1485 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001486}
1487
1488// Change the color of an opaque pixel to the pen color.
1489void Magick::Image::opaque ( const Color &opaqueColor_,
1490 const Color &penColor_ )
1491{
1492 if ( !opaqueColor_.isValid() )
1493 {
1494 throwExceptionExplicit( OptionError,
1495 "Opaque color argument is invalid" );
1496 }
1497 if ( !penColor_.isValid() )
1498 {
1499 throwExceptionExplicit( OptionError,
1500 "Pen color argument is invalid" );
1501 }
1502
1503 modifyImage();
1504 std::string opaqueColor = opaqueColor_;
1505 std::string penColor = penColor_;
1506
cristy4c08aed2011-07-01 19:47:50 +00001507 PixelInfo opaque;
1508 PixelInfo pen;
cristy189e84c2011-08-27 18:08:53 +00001509 ExceptionInfo exceptionInfo;
1510 GetExceptionInfo( &exceptionInfo );
cristy269c9412011-10-13 23:41:15 +00001511 (void) QueryColorCompliance(std::string(opaqueColor_).c_str(),
cristy9950d572011-10-01 18:22:35 +00001512 AllCompliance, &opaque, &exceptionInfo);
cristy269c9412011-10-13 23:41:15 +00001513 (void) QueryColorCompliance(std::string(penColor_).c_str(),
cristy9950d572011-10-01 18:22:35 +00001514 AllCompliance, &pen, &exceptionInfo);
cristy189e84c2011-08-27 18:08:53 +00001515 OpaquePaintImage ( image(), &opaque, &pen, MagickFalse, &exceptionInfo );
1516 throwException( exceptionInfo );
1517 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001518}
1519
1520// Ping is similar to read except only enough of the image is read to
1521// determine the image columns, rows, and filesize. Access the
1522// columns(), rows(), and fileSize() attributes after invoking ping.
1523// The image data is not valid after calling ping.
1524void Magick::Image::ping ( const std::string &imageSpec_ )
1525{
1526 options()->fileName( imageSpec_ );
1527 ExceptionInfo exceptionInfo;
1528 GetExceptionInfo( &exceptionInfo );
1529 MagickCore::Image* image =
1530 PingImage( imageInfo(), &exceptionInfo );
1531 replaceImage( image );
1532 throwException( exceptionInfo );
1533 (void) DestroyExceptionInfo( &exceptionInfo );
1534}
1535
1536// Ping is similar to read except only enough of the image is read
1537// to determine the image columns, rows, and filesize. Access the
1538// columns(), rows(), and fileSize() attributes after invoking
1539// ping. The image data is not valid after calling ping.
1540void Magick::Image::ping ( const Blob& blob_ )
1541{
1542 ExceptionInfo exceptionInfo;
1543 GetExceptionInfo( &exceptionInfo );
1544 MagickCore::Image* image =
1545 PingBlob( imageInfo(), blob_.data(), blob_.length(), &exceptionInfo );
1546 replaceImage( image );
1547 throwException( exceptionInfo );
1548 (void) DestroyExceptionInfo( &exceptionInfo );
1549}
1550
1551// Execute a named process module using an argc/argv syntax similar to
1552// that accepted by a C 'main' routine. An exception is thrown if the
1553// requested process module doesn't exist, fails to load, or fails during
1554// execution.
cristyd99b0962010-05-29 23:14:26 +00001555void Magick::Image::process( std::string name_, const ssize_t argc, const char **argv )
cristy3ed852e2009-09-05 21:47:34 +00001556{
1557 modifyImage();
1558
cristyc82a27b2011-10-21 01:07:16 +00001559 ExceptionInfo exceptionInfo;
1560 GetExceptionInfo( &exceptionInfo );
cristyeaedf062010-05-29 22:36:02 +00001561 size_t status =
cristy3ed852e2009-09-05 21:47:34 +00001562 InvokeDynamicImageFilter( name_.c_str(), &image(), argc, argv,
cristyc82a27b2011-10-21 01:07:16 +00001563 &exceptionInfo );
cristye07716f2012-04-08 02:09:18 +00001564 (void) status;
cristyc82a27b2011-10-21 01:07:16 +00001565 throwException( exceptionInfo );
1566 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001567}
1568
1569// Quantize colors in image using current quantization settings
1570// Set measureError_ to true in order to measure quantization error
1571void Magick::Image::quantize ( const bool measureError_ )
1572{
1573 modifyImage();
1574
1575 if (measureError_)
1576 options()->quantizeInfo()->measure_error=MagickTrue;
1577 else
1578 options()->quantizeInfo()->measure_error=MagickFalse;
1579
cristy018f07f2011-09-04 21:15:19 +00001580 ExceptionInfo exceptionInfo;
1581 GetExceptionInfo( &exceptionInfo );
1582 QuantizeImage( options()->quantizeInfo(), image(), &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001583
cristy018f07f2011-09-04 21:15:19 +00001584 throwException( exceptionInfo );
1585 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001586}
1587
1588// Apply an arithmetic or bitwise operator to the image pixel quantums.
1589void Magick::Image::quantumOperator ( const ChannelType channel_,
1590 const MagickEvaluateOperator operator_,
1591 double rvalue_)
1592{
1593 ExceptionInfo exceptionInfo;
1594 GetExceptionInfo( &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +00001595 ChannelType channel_mask = SetPixelChannelMask( image(), channel_ );
cristyd42d9952011-07-08 14:21:50 +00001596 EvaluateImage( image(), operator_, rvalue_, &exceptionInfo);
cristye2a912b2011-12-05 20:02:07 +00001597 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +00001598 throwException( exceptionInfo );
1599 (void) DestroyExceptionInfo( &exceptionInfo );
1600}
1601
cristyd99b0962010-05-29 23:14:26 +00001602void Magick::Image::quantumOperator ( const ssize_t x_,const ssize_t y_,
cristyeaedf062010-05-29 22:36:02 +00001603 const size_t columns_,
1604 const size_t rows_,
cristy3ed852e2009-09-05 21:47:34 +00001605 const ChannelType channel_,
1606 const MagickEvaluateOperator operator_,
1607 const double rvalue_)
1608{
1609 ExceptionInfo exceptionInfo;
1610 GetExceptionInfo( &exceptionInfo );
1611 RectangleInfo geometry;
1612 geometry.width = columns_;
1613 geometry.height = rows_;
1614 geometry.x = x_;
1615 geometry.y = y_;
1616 MagickCore::Image *crop_image = CropImage( image(), &geometry,
1617 &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +00001618 ChannelType channel_mask = SetPixelChannelMask( image(), channel_);
cristyd42d9952011-07-08 14:21:50 +00001619 EvaluateImage( crop_image, operator_, rvalue_, &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +00001620 (void) SetPixelChannelMapMask( image(), channel_mask );
cristyfeb3e962012-03-29 17:25:55 +00001621 (void) CompositeImage( image(), crop_image, image()->matte != MagickFalse ?
1622 OverCompositeOp : CopyCompositeOp, MagickFalse, geometry.x, geometry.y,
cristye941a752011-10-15 01:52:48 +00001623 &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001624 crop_image = DestroyImageList(crop_image);
1625 throwException( exceptionInfo );
1626 (void) DestroyExceptionInfo( &exceptionInfo );
1627}
1628
1629// Raise image (lighten or darken the edges of an image to give a 3-D
1630// raised or lowered effect)
1631void Magick::Image::raise ( const Geometry &geometry_ ,
1632 const bool raisedFlag_ )
1633{
cristy6170ac32011-08-28 14:15:37 +00001634 ExceptionInfo exceptionInfo;
1635 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001636 RectangleInfo raiseInfo = geometry_;
1637 modifyImage();
cristy6170ac32011-08-28 14:15:37 +00001638 RaiseImage ( image(), &raiseInfo, raisedFlag_ == true ? MagickTrue : MagickFalse, &exceptionInfo );
1639 throwException( exceptionInfo );
1640 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001641}
1642
1643
1644// Random threshold image.
1645//
1646// Changes the value of individual pixels based on the intensity
1647// of each pixel compared to a random threshold. The result is a
1648// low-contrast, two color image. The thresholds_ argument is a
1649// geometry containing LOWxHIGH thresholds. If the string
1650// contains 2x2, 3x3, or 4x4, then an ordered dither of order 2,
1651// 3, or 4 will be performed instead. If a channel_ argument is
1652// specified then only the specified channel is altered. This is
1653// a very fast alternative to 'quantize' based dithering.
1654void Magick::Image::randomThreshold( const Geometry &thresholds_ )
1655{
1656 randomThresholdChannel(thresholds_,DefaultChannels);
1657}
1658void Magick::Image::randomThresholdChannel( const Geometry &thresholds_,
1659 const ChannelType channel_ )
1660{
1661 ExceptionInfo exceptionInfo;
1662 GetExceptionInfo( &exceptionInfo );
1663 modifyImage();
cristybd5a96c2011-08-21 00:04:26 +00001664 ChannelType channel_mask = SetPixelChannelMask( image(), channel_);
cristyf4ad9df2011-07-08 16:49:03 +00001665 (void) RandomThresholdImage( image(),
cristy3ed852e2009-09-05 21:47:34 +00001666 static_cast<std::string>(thresholds_).c_str(),
1667 &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +00001668 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +00001669 throwImageException();
1670 (void) DestroyExceptionInfo( &exceptionInfo );
1671}
1672
1673// Read image into current object
1674void Magick::Image::read ( const std::string &imageSpec_ )
1675{
1676 options()->fileName( imageSpec_ );
1677
1678 ExceptionInfo exceptionInfo;
1679 GetExceptionInfo( &exceptionInfo );
1680 MagickCore::Image* image =
1681 ReadImage( imageInfo(), &exceptionInfo );
1682
1683 // Ensure that multiple image frames were not read.
1684 if ( image && image->next )
1685 {
1686 // Destroy any extra image frames
1687 MagickCore::Image* next = image->next;
1688 image->next = 0;
1689 next->previous = 0;
1690 DestroyImageList( next );
1691
1692 }
1693 replaceImage( image );
1694 throwException( exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001695 (void) DestroyExceptionInfo( &exceptionInfo );
1696}
1697
1698// Read image of specified size into current object
1699void Magick::Image::read ( const Geometry &size_,
1700 const std::string &imageSpec_ )
1701{
1702 size( size_ );
1703 read( imageSpec_ );
1704}
1705
1706// Read image from in-memory BLOB
1707void Magick::Image::read ( const Blob &blob_ )
1708{
1709 ExceptionInfo exceptionInfo;
1710 GetExceptionInfo( &exceptionInfo );
1711 MagickCore::Image* image =
1712 BlobToImage( imageInfo(),
1713 static_cast<const void *>(blob_.data()),
1714 blob_.length(), &exceptionInfo );
1715 replaceImage( image );
1716 throwException( exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001717 (void) DestroyExceptionInfo( &exceptionInfo );
1718}
1719
1720// Read image of specified size from in-memory BLOB
1721void Magick::Image::read ( const Blob &blob_,
1722 const Geometry &size_ )
1723{
1724 // Set image size
1725 size( size_ );
1726 // Read from Blob
1727 read( blob_ );
1728}
1729
1730// Read image of specified size and depth from in-memory BLOB
1731void Magick::Image::read ( const Blob &blob_,
1732 const Geometry &size_,
cristyeaedf062010-05-29 22:36:02 +00001733 const size_t depth_ )
cristy3ed852e2009-09-05 21:47:34 +00001734{
1735 // Set image size
1736 size( size_ );
1737 // Set image depth
1738 depth( depth_ );
1739 // Read from Blob
1740 read( blob_ );
1741}
1742
1743// Read image of specified size, depth, and format from in-memory BLOB
1744void Magick::Image::read ( const Blob &blob_,
1745 const Geometry &size_,
cristyeaedf062010-05-29 22:36:02 +00001746 const size_t depth_,
cristy3ed852e2009-09-05 21:47:34 +00001747 const std::string &magick_ )
1748{
1749 // Set image size
1750 size( size_ );
1751 // Set image depth
1752 depth( depth_ );
1753 // Set image magick
1754 magick( magick_ );
1755 // Set explicit image format
1756 fileName( magick_ + ':');
1757 // Read from Blob
1758 read( blob_ );
1759}
1760
1761// Read image of specified size, and format from in-memory BLOB
1762void Magick::Image::read ( const Blob &blob_,
1763 const Geometry &size_,
1764 const std::string &magick_ )
1765{
1766 // Set image size
1767 size( size_ );
1768 // Set image magick
1769 magick( magick_ );
1770 // Set explicit image format
1771 fileName( magick_ + ':');
1772 // Read from Blob
1773 read( blob_ );
1774}
1775
1776// Read image based on raw pixels in memory (ConstituteImage)
cristyeaedf062010-05-29 22:36:02 +00001777void Magick::Image::read ( const size_t width_,
1778 const size_t height_,
cristy3ed852e2009-09-05 21:47:34 +00001779 const std::string &map_,
1780 const StorageType type_,
1781 const void *pixels_ )
1782{
1783 ExceptionInfo exceptionInfo;
1784 GetExceptionInfo( &exceptionInfo );
1785 MagickCore::Image* image =
1786 ConstituteImage( width_, height_, map_.c_str(), type_, pixels_,
1787 &exceptionInfo );
1788 replaceImage( image );
1789 throwException( exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001790 (void) DestroyExceptionInfo( &exceptionInfo );
1791}
1792
cristy3ed852e2009-09-05 21:47:34 +00001793// Reduce noise in image
1794void Magick::Image::reduceNoise ( const double order_ )
1795{
1796 ExceptionInfo exceptionInfo;
1797 GetExceptionInfo( &exceptionInfo );
1798 MagickCore::Image* newImage =
cristy95c38342011-03-18 22:39:51 +00001799 StatisticImage( image(), NonpeakStatistic, (size_t) order_, (size_t) order_,
1800 &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001801 replaceImage( newImage );
1802 throwException( exceptionInfo );
1803 (void) DestroyExceptionInfo( &exceptionInfo );
1804}
1805
1806// Resize image
1807void Magick::Image::resize( const Geometry &geometry_ )
1808{
1809 // Calculate new size. This code should be supported using binary arguments
1810 // in the ImageMagick library.
cristybb503372010-05-27 20:51:26 +00001811 ssize_t x = 0;
1812 ssize_t y = 0;
1813 size_t width = columns();
1814 size_t height = rows();
cristy3ed852e2009-09-05 21:47:34 +00001815
1816 ParseMetaGeometry (static_cast<std::string>(geometry_).c_str(),
1817 &x, &y,
1818 &width, &height );
1819
1820 ExceptionInfo exceptionInfo;
1821 GetExceptionInfo( &exceptionInfo );
1822 MagickCore::Image* newImage =
1823 ResizeImage( image(),
1824 width,
1825 height,
1826 image()->filter,
cristy3ed852e2009-09-05 21:47:34 +00001827 &exceptionInfo);
1828 replaceImage( newImage );
1829 throwException( exceptionInfo );
1830 (void) DestroyExceptionInfo( &exceptionInfo );
1831}
1832
1833// Roll image
1834void Magick::Image::roll ( const Geometry &roll_ )
1835{
cristybb503372010-05-27 20:51:26 +00001836 ssize_t xOff = roll_.xOff();
cristy3ed852e2009-09-05 21:47:34 +00001837 if ( roll_.xNegative() )
1838 xOff = 0 - xOff;
cristybb503372010-05-27 20:51:26 +00001839 ssize_t yOff = roll_.yOff();
cristy3ed852e2009-09-05 21:47:34 +00001840 if ( roll_.yNegative() )
1841 yOff = 0 - yOff;
1842
1843 ExceptionInfo exceptionInfo;
1844 GetExceptionInfo( &exceptionInfo );
1845 MagickCore::Image* newImage =
1846 RollImage( image(), xOff, yOff, &exceptionInfo );
1847 replaceImage( newImage );
1848 throwException( exceptionInfo );
1849 (void) DestroyExceptionInfo( &exceptionInfo );
1850}
cristyeaedf062010-05-29 22:36:02 +00001851void Magick::Image::roll ( const size_t columns_,
1852 const size_t rows_ )
cristy3ed852e2009-09-05 21:47:34 +00001853{
1854 ExceptionInfo exceptionInfo;
1855 GetExceptionInfo( &exceptionInfo );
1856 MagickCore::Image* newImage =
1857 RollImage( image(),
cristybb503372010-05-27 20:51:26 +00001858 static_cast<ssize_t>(columns_),
1859 static_cast<ssize_t>(rows_), &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001860 replaceImage( newImage );
1861 throwException( exceptionInfo );
1862 (void) DestroyExceptionInfo( &exceptionInfo );
1863}
1864
1865// Rotate image
1866void Magick::Image::rotate ( const double degrees_ )
1867{
1868 ExceptionInfo exceptionInfo;
1869 GetExceptionInfo( &exceptionInfo );
1870 MagickCore::Image* newImage =
1871 RotateImage( image(), degrees_, &exceptionInfo);
1872 replaceImage( newImage );
1873 throwException( exceptionInfo );
1874 (void) DestroyExceptionInfo( &exceptionInfo );
1875}
1876
1877// Sample image
1878void Magick::Image::sample ( const Geometry &geometry_ )
1879{
cristybb503372010-05-27 20:51:26 +00001880 ssize_t x = 0;
1881 ssize_t y = 0;
1882 size_t width = columns();
1883 size_t height = rows();
cristy3ed852e2009-09-05 21:47:34 +00001884
1885 ParseMetaGeometry (static_cast<std::string>(geometry_).c_str(),
1886 &x, &y,
1887 &width, &height );
1888
1889 ExceptionInfo exceptionInfo;
1890 GetExceptionInfo( &exceptionInfo );
1891 MagickCore::Image* newImage =
1892 SampleImage( image(), width, height, &exceptionInfo );
1893 replaceImage( newImage );
1894 throwException( exceptionInfo );
1895 (void) DestroyExceptionInfo( &exceptionInfo );
1896}
1897
1898// Scale image
1899void Magick::Image::scale ( const Geometry &geometry_ )
1900{
cristybb503372010-05-27 20:51:26 +00001901 ssize_t x = 0;
1902 ssize_t y = 0;
1903 size_t width = columns();
1904 size_t height = rows();
cristy3ed852e2009-09-05 21:47:34 +00001905
1906 ParseMetaGeometry (static_cast<std::string>(geometry_).c_str(),
1907 &x, &y,
1908 &width, &height );
1909
1910 ExceptionInfo exceptionInfo;
1911 GetExceptionInfo( &exceptionInfo );
1912 MagickCore::Image* newImage =
1913 ScaleImage( image(), width, height, &exceptionInfo );
1914 replaceImage( newImage );
1915 throwException( exceptionInfo );
1916 (void) DestroyExceptionInfo( &exceptionInfo );
1917}
1918
1919// Segment (coalesce similar image components) by analyzing the
1920// histograms of the color components and identifying units that are
1921// homogeneous with the fuzzy c-means technique.
1922void Magick::Image::segment ( const double clusterThreshold_,
1923 const double smoothingThreshold_ )
1924{
cristy018f07f2011-09-04 21:15:19 +00001925 ExceptionInfo exceptionInfo;
1926 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001927 modifyImage();
1928 SegmentImage ( image(),
1929 options()->quantizeColorSpace(),
1930 (MagickBooleanType) options()->verbose(),
1931 clusterThreshold_,
cristy018f07f2011-09-04 21:15:19 +00001932 smoothingThreshold_, &exceptionInfo );
cristyea1a8aa2011-10-20 13:24:06 +00001933 SyncImage( image(), &exceptionInfo );
cristy018f07f2011-09-04 21:15:19 +00001934 throwException( exceptionInfo );
1935 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00001936}
1937
1938// Shade image using distant light source
1939void Magick::Image::shade ( const double azimuth_,
1940 const double elevation_,
1941 const bool colorShading_ )
1942{
1943 ExceptionInfo exceptionInfo;
1944 GetExceptionInfo( &exceptionInfo );
1945 MagickCore::Image* newImage =
1946 ShadeImage( image(),
1947 colorShading_ == true ? MagickTrue : MagickFalse,
1948 azimuth_,
1949 elevation_,
1950 &exceptionInfo);
1951 replaceImage( newImage );
1952 throwException( exceptionInfo );
1953 (void) DestroyExceptionInfo( &exceptionInfo );
1954}
1955
cristy171a5b32012-02-01 00:38:49 +00001956// Simulate an image shadow
cristy6fee7fb2012-01-31 20:14:01 +00001957void Magick::Image::shadow( const double percent_opacity_, const double sigma_,
1958 const ssize_t x_, const ssize_t y_ )
1959{
1960 ExceptionInfo exceptionInfo;
1961 GetExceptionInfo( &exceptionInfo );
1962 MagickCore::Image* newImage = ShadowImage( image(), percent_opacity_, sigma_,
cristyaa2c16c2012-03-25 22:21:35 +00001963 x_, y_, &exceptionInfo );
cristy6fee7fb2012-01-31 20:14:01 +00001964 replaceImage( newImage );
1965 throwException( exceptionInfo );
1966 (void) DestroyExceptionInfo( &exceptionInfo );
1967}
1968
cristy3ed852e2009-09-05 21:47:34 +00001969// Sharpen pixels in image
1970void Magick::Image::sharpen ( const double radius_, const double sigma_ )
1971{
1972 ExceptionInfo exceptionInfo;
1973 GetExceptionInfo( &exceptionInfo );
1974 MagickCore::Image* newImage =
1975 SharpenImage( image(),
1976 radius_,
1977 sigma_,
1978 &exceptionInfo );
1979 replaceImage( newImage );
1980 throwException( exceptionInfo );
1981 (void) DestroyExceptionInfo( &exceptionInfo );
1982}
1983
1984void Magick::Image::sharpenChannel ( const ChannelType channel_,
1985 const double radius_, const double sigma_ )
1986{
1987 ExceptionInfo exceptionInfo;
1988 GetExceptionInfo( &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +00001989 ChannelType channel_mask = SetPixelChannelMask( image(), channel_ );
cristy3ed852e2009-09-05 21:47:34 +00001990 MagickCore::Image* newImage =
cristyf4ad9df2011-07-08 16:49:03 +00001991 SharpenImage( image(),
cristy3ed852e2009-09-05 21:47:34 +00001992 radius_,
1993 sigma_,
1994 &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +00001995 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +00001996 replaceImage( newImage );
1997 throwException( exceptionInfo );
1998 (void) DestroyExceptionInfo( &exceptionInfo );
1999}
2000
2001// Shave pixels from image edges.
2002void Magick::Image::shave ( const Geometry &geometry_ )
2003{
2004 RectangleInfo shaveInfo = geometry_;
2005 ExceptionInfo exceptionInfo;
2006 GetExceptionInfo( &exceptionInfo );
2007 MagickCore::Image* newImage =
2008 ShaveImage( image(),
2009 &shaveInfo,
2010 &exceptionInfo);
2011 replaceImage( newImage );
2012 throwException( exceptionInfo );
2013 (void) DestroyExceptionInfo( &exceptionInfo );
2014}
2015
2016// Shear image
2017void Magick::Image::shear ( const double xShearAngle_,
2018 const double yShearAngle_ )
2019{
2020 ExceptionInfo exceptionInfo;
2021 GetExceptionInfo( &exceptionInfo );
2022 MagickCore::Image* newImage =
2023 ShearImage( image(),
2024 xShearAngle_,
2025 yShearAngle_,
2026 &exceptionInfo );
2027 replaceImage( newImage );
2028 throwException( exceptionInfo );
2029 (void) DestroyExceptionInfo( &exceptionInfo );
2030}
2031
2032// Contrast image
cristyeaedf062010-05-29 22:36:02 +00002033void Magick::Image::sigmoidalContrast ( const size_t sharpen_, const double contrast, const double midpoint )
cristy3ed852e2009-09-05 21:47:34 +00002034{
cristy33bd5152011-08-24 01:42:24 +00002035 ExceptionInfo exceptionInfo;
2036 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002037 modifyImage();
cristy33bd5152011-08-24 01:42:24 +00002038 (void) SigmoidalContrastImage( image(), (MagickBooleanType) sharpen_, contrast, midpoint, &exceptionInfo );
2039 throwException( exceptionInfo );
2040 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002041}
2042
2043// Solarize image (similar to effect seen when exposing a photographic
2044// film to light during the development process)
2045void Magick::Image::solarize ( const double factor_ )
2046{
cristy5cbc0162011-08-29 00:36:28 +00002047 ExceptionInfo exceptionInfo;
2048 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002049 modifyImage();
cristy5cbc0162011-08-29 00:36:28 +00002050 SolarizeImage ( image(), factor_, &exceptionInfo );
2051 throwException( exceptionInfo );
2052 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002053}
2054
2055// Sparse color image, given a set of coordinates, interpolates the colors
2056// found at those coordinates, across the whole image, using various methods.
2057//
2058void Magick::Image::sparseColor ( const ChannelType channel,
2059 const SparseColorMethod method,
cristybb503372010-05-27 20:51:26 +00002060 const size_t number_arguments,
cristy3ed852e2009-09-05 21:47:34 +00002061 const double *arguments )
2062{
2063 ExceptionInfo exceptionInfo;
2064 GetExceptionInfo( &exceptionInfo );
cristy3884f692011-07-08 18:00:18 +00002065
cristybd5a96c2011-08-21 00:04:26 +00002066 ChannelType channel_mask = SetPixelChannelMask( image(), channel );
cristy3884f692011-07-08 18:00:18 +00002067 MagickCore::Image* newImage = SparseColorImage ( image(), method,
cristy3ed852e2009-09-05 21:47:34 +00002068 number_arguments, arguments, &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +00002069 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +00002070 replaceImage( newImage );
2071 throwException( exceptionInfo );
2072 (void) DestroyExceptionInfo( &exceptionInfo );
2073}
2074
2075// Spread pixels randomly within image by specified ammount
cristyeaedf062010-05-29 22:36:02 +00002076void Magick::Image::spread ( const size_t amount_ )
cristy3ed852e2009-09-05 21:47:34 +00002077{
2078 ExceptionInfo exceptionInfo;
2079 GetExceptionInfo( &exceptionInfo );
2080 MagickCore::Image* newImage =
2081 SpreadImage( image(),
2082 amount_,
cristy5c4e2582011-09-11 19:21:03 +00002083 image()->interpolate,
cristy3ed852e2009-09-05 21:47:34 +00002084 &exceptionInfo );
2085 replaceImage( newImage );
2086 throwException( exceptionInfo );
2087 (void) DestroyExceptionInfo( &exceptionInfo );
2088}
2089
2090// Add a digital watermark to the image (based on second image)
2091void Magick::Image::stegano ( const Image &watermark_ )
2092{
2093 ExceptionInfo exceptionInfo;
2094 GetExceptionInfo( &exceptionInfo );
2095 MagickCore::Image* newImage =
2096 SteganoImage( image(),
2097 watermark_.constImage(),
2098 &exceptionInfo);
2099 replaceImage( newImage );
2100 throwException( exceptionInfo );
2101 (void) DestroyExceptionInfo( &exceptionInfo );
2102}
2103
2104// Stereo image (left image is current image)
2105void Magick::Image::stereo ( const Image &rightImage_ )
2106{
2107 ExceptionInfo exceptionInfo;
2108 GetExceptionInfo( &exceptionInfo );
2109 MagickCore::Image* newImage =
2110 StereoImage( image(),
2111 rightImage_.constImage(),
2112 &exceptionInfo);
2113 replaceImage( newImage );
2114 throwException( exceptionInfo );
2115 (void) DestroyExceptionInfo( &exceptionInfo );
2116}
2117
2118// Swirl image
2119void Magick::Image::swirl ( const double degrees_ )
2120{
2121 ExceptionInfo exceptionInfo;
2122 GetExceptionInfo( &exceptionInfo );
2123 MagickCore::Image* newImage =
cristy76f512e2011-09-12 01:26:56 +00002124 SwirlImage( image(), degrees_, image()->interpolate,
cristy3ed852e2009-09-05 21:47:34 +00002125 &exceptionInfo);
2126 replaceImage( newImage );
2127 throwException( exceptionInfo );
2128 (void) DestroyExceptionInfo( &exceptionInfo );
2129}
2130
2131// Texture image
2132void Magick::Image::texture ( const Image &texture_ )
2133{
2134 modifyImage();
cristye941a752011-10-15 01:52:48 +00002135 ExceptionInfo exceptionInfo;
2136 GetExceptionInfo( &exceptionInfo );
2137 TextureImage( image(), texture_.constImage(), &exceptionInfo );
2138 throwException( exceptionInfo );
2139 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002140}
2141
2142// Threshold image
2143void Magick::Image::threshold ( const double threshold_ )
2144{
2145 modifyImage();
cristye941a752011-10-15 01:52:48 +00002146 ExceptionInfo exceptionInfo;
2147 GetExceptionInfo( &exceptionInfo );
2148 BilevelImage( image(), threshold_, &exceptionInfo );
2149 throwException( exceptionInfo );
2150 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002151}
2152
2153// Transform image based on image geometry only
2154void Magick::Image::transform ( const Geometry &imageGeometry_ )
2155{
2156 modifyImage();
cristye941a752011-10-15 01:52:48 +00002157 ExceptionInfo exceptionInfo;
2158 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002159 TransformImage ( &(image()), 0,
cristye941a752011-10-15 01:52:48 +00002160 std::string(imageGeometry_).c_str(), &exceptionInfo );
2161 throwException( exceptionInfo );
2162 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002163}
2164// Transform image based on image and crop geometries
2165void Magick::Image::transform ( const Geometry &imageGeometry_,
2166 const Geometry &cropGeometry_ )
2167{
2168 modifyImage();
cristye941a752011-10-15 01:52:48 +00002169 ExceptionInfo exceptionInfo;
2170 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002171 TransformImage ( &(image()), std::string(cropGeometry_).c_str(),
cristye941a752011-10-15 01:52:48 +00002172 std::string(imageGeometry_).c_str(), &exceptionInfo );
2173 throwException( exceptionInfo );
2174 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002175}
2176
2177// Add matte image to image, setting pixels matching color to transparent
2178void Magick::Image::transparent ( const Color &color_ )
2179{
2180 if ( !color_.isValid() )
2181 {
2182 throwExceptionExplicit( OptionError,
2183 "Color argument is invalid" );
2184 }
2185
2186 std::string color = color_;
2187
cristy4c08aed2011-07-01 19:47:50 +00002188 PixelInfo target;
cristy189e84c2011-08-27 18:08:53 +00002189 ExceptionInfo exceptionInfo;
2190 GetExceptionInfo( &exceptionInfo );
cristyc82a27b2011-10-21 01:07:16 +00002191 (void) QueryColorCompliance(std::string(color_).c_str(),AllCompliance,
2192 &target,&exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00002193 modifyImage();
cristy189e84c2011-08-27 18:08:53 +00002194 TransparentPaintImage ( image(), &target, TransparentAlpha, MagickFalse,
2195 &exceptionInfo );
2196 throwException( exceptionInfo );
2197 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002198}
2199
2200// Add matte image to image, setting pixels matching color to transparent
2201void Magick::Image::transparentChroma(const Color &colorLow_,
2202 const Color &colorHigh_)
2203{
2204 if ( !colorLow_.isValid() || !colorHigh_.isValid() )
2205 {
2206 throwExceptionExplicit( OptionError,
2207 "Color argument is invalid" );
2208 }
2209
2210 std::string colorLow = colorLow_;
2211 std::string colorHigh = colorHigh_;
2212
cristy4c08aed2011-07-01 19:47:50 +00002213 PixelInfo targetLow;
2214 PixelInfo targetHigh;
cristy189e84c2011-08-27 18:08:53 +00002215 ExceptionInfo exceptionInfo;
2216 GetExceptionInfo( &exceptionInfo );
cristyc82a27b2011-10-21 01:07:16 +00002217 (void) QueryColorCompliance(std::string(colorLow_).c_str(),
2218 AllCompliance,&targetLow,&exceptionInfo);
2219 (void) QueryColorCompliance(std::string(colorHigh_).c_str(),
2220 AllCompliance,&targetHigh,&exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00002221 modifyImage();
2222 TransparentPaintImageChroma ( image(), &targetLow, &targetHigh,
cristy189e84c2011-08-27 18:08:53 +00002223 TransparentAlpha, MagickFalse, &exceptionInfo );
2224 throwException( exceptionInfo );
2225 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002226}
2227
2228
2229// Trim edges that are the background color from the image
2230void Magick::Image::trim ( void )
2231{
2232 ExceptionInfo exceptionInfo;
2233 GetExceptionInfo( &exceptionInfo );
2234 MagickCore::Image* newImage =
2235 TrimImage( image(), &exceptionInfo);
2236 replaceImage( newImage );
2237 throwException( exceptionInfo );
2238 (void) DestroyExceptionInfo( &exceptionInfo );
2239}
2240
2241// Replace image with a sharpened version of the original image
2242// using the unsharp mask algorithm.
2243// radius_
2244// the radius of the Gaussian, in pixels, not counting the
2245// center pixel.
2246// sigma_
2247// the standard deviation of the Gaussian, in pixels.
2248// amount_
2249// the percentage of the difference between the original and
2250// the blur image that is added back into the original.
2251// threshold_
2252// the threshold in pixels needed to apply the diffence amount.
2253void Magick::Image::unsharpmask ( const double radius_,
2254 const double sigma_,
2255 const double amount_,
2256 const double threshold_ )
2257{
2258 ExceptionInfo exceptionInfo;
2259 GetExceptionInfo( &exceptionInfo );
2260 MagickCore::Image* newImage =
2261 UnsharpMaskImage( image(),
2262 radius_,
2263 sigma_,
2264 amount_,
2265 threshold_,
2266 &exceptionInfo );
2267 replaceImage( newImage );
2268 throwException( exceptionInfo );
2269 (void) DestroyExceptionInfo( &exceptionInfo );
2270}
2271
2272void Magick::Image::unsharpmaskChannel ( const ChannelType channel_,
2273 const double radius_,
2274 const double sigma_,
2275 const double amount_,
2276 const double threshold_ )
2277{
2278 ExceptionInfo exceptionInfo;
2279 GetExceptionInfo( &exceptionInfo );
cristybd5a96c2011-08-21 00:04:26 +00002280 ChannelType channel_mask = SetPixelChannelMask( image(), channel_ );
cristy3ed852e2009-09-05 21:47:34 +00002281 MagickCore::Image* newImage =
cristyf4ad9df2011-07-08 16:49:03 +00002282 UnsharpMaskImage( image(),
cristy3ed852e2009-09-05 21:47:34 +00002283 radius_,
2284 sigma_,
2285 amount_,
2286 threshold_,
2287 &exceptionInfo );
cristye2a912b2011-12-05 20:02:07 +00002288 (void) SetPixelChannelMapMask( image(), channel_mask );
cristy3ed852e2009-09-05 21:47:34 +00002289 replaceImage( newImage );
2290 throwException( exceptionInfo );
2291 (void) DestroyExceptionInfo( &exceptionInfo );
2292}
2293
2294// Map image pixels to a sine wave
2295void Magick::Image::wave ( const double amplitude_, const double wavelength_ )
2296{
2297 ExceptionInfo exceptionInfo;
2298 GetExceptionInfo( &exceptionInfo );
2299 MagickCore::Image* newImage =
2300 WaveImage( image(),
2301 amplitude_,
2302 wavelength_,
cristy5c4e2582011-09-11 19:21:03 +00002303 image()->interpolate,
cristy3ed852e2009-09-05 21:47:34 +00002304 &exceptionInfo);
2305 replaceImage( newImage );
2306 throwException( exceptionInfo );
2307 (void) DestroyExceptionInfo( &exceptionInfo );
2308}
2309
2310// Write image to file
2311void Magick::Image::write( const std::string &imageSpec_ )
2312{
cristy6f9e0d32011-08-28 16:32:09 +00002313 ExceptionInfo exceptionInfo;
2314 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002315 modifyImage();
2316 fileName( imageSpec_ );
cristy6f9e0d32011-08-28 16:32:09 +00002317 WriteImage( imageInfo(), image(), &exceptionInfo );
2318 throwException( exceptionInfo );
2319 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002320}
2321
2322// Write image to in-memory BLOB
2323void Magick::Image::write ( Blob *blob_ )
2324{
2325 modifyImage();
2326 size_t length = 2048; // Efficient size for small images
2327 ExceptionInfo exceptionInfo;
2328 GetExceptionInfo( &exceptionInfo );
2329 void* data = ImageToBlob( imageInfo(),
2330 image(),
2331 &length,
2332 &exceptionInfo);
2333 throwException( exceptionInfo );
2334 blob_->updateNoCopy( data, length, Blob::MallocAllocator );
2335 throwImageException();
2336 (void) DestroyExceptionInfo( &exceptionInfo );
2337}
2338void Magick::Image::write ( Blob *blob_,
2339 const std::string &magick_ )
2340{
2341 modifyImage();
2342 magick(magick_);
2343 size_t length = 2048; // Efficient size for small images
2344 ExceptionInfo exceptionInfo;
2345 GetExceptionInfo( &exceptionInfo );
2346 void* data = ImageToBlob( imageInfo(),
2347 image(),
2348 &length,
2349 &exceptionInfo);
2350 throwException( exceptionInfo );
2351 blob_->updateNoCopy( data, length, Blob::MallocAllocator );
2352 throwImageException();
2353 (void) DestroyExceptionInfo( &exceptionInfo );
2354}
2355void Magick::Image::write ( Blob *blob_,
2356 const std::string &magick_,
cristyeaedf062010-05-29 22:36:02 +00002357 const size_t depth_ )
cristy3ed852e2009-09-05 21:47:34 +00002358{
2359 modifyImage();
2360 magick(magick_);
2361 depth(depth_);
2362 size_t length = 2048; // Efficient size for small images
2363 ExceptionInfo exceptionInfo;
2364 GetExceptionInfo( &exceptionInfo );
2365 void* data = ImageToBlob( imageInfo(),
2366 image(),
2367 &length,
2368 &exceptionInfo);
2369 throwException( exceptionInfo );
2370 blob_->updateNoCopy( data, length, Blob::MallocAllocator );
2371 throwImageException();
2372 (void) DestroyExceptionInfo( &exceptionInfo );
2373}
2374
2375// Write image to an array of pixels with storage type specified
2376// by user (ExportImagePixels), e.g.
2377// image.write( 0, 0, 640, 1, "RGB", 0, pixels );
cristyd99b0962010-05-29 23:14:26 +00002378void Magick::Image::write ( const ssize_t x_,
2379 const ssize_t y_,
cristyeaedf062010-05-29 22:36:02 +00002380 const size_t columns_,
2381 const size_t rows_,
cristy3ed852e2009-09-05 21:47:34 +00002382 const std::string &map_,
2383 const StorageType type_,
2384 void *pixels_ )
2385{
2386 ExceptionInfo exceptionInfo;
2387 GetExceptionInfo( &exceptionInfo );
2388 ExportImagePixels( image(), x_, y_, columns_, rows_, map_.c_str(), type_,
2389 pixels_,
2390 &exceptionInfo);
2391 throwException( exceptionInfo );
2392 (void) DestroyExceptionInfo( &exceptionInfo );
2393}
2394
2395// Zoom image
2396void Magick::Image::zoom( const Geometry &geometry_ )
2397{
2398 // Calculate new size. This code should be supported using binary arguments
2399 // in the ImageMagick library.
cristybb503372010-05-27 20:51:26 +00002400 ssize_t x = 0;
2401 ssize_t y = 0;
2402 size_t width = columns();
2403 size_t height = rows();
cristy3ed852e2009-09-05 21:47:34 +00002404
2405 ParseMetaGeometry (static_cast<std::string>(geometry_).c_str(),
2406 &x, &y,
2407 &width, &height );
2408
2409 ExceptionInfo exceptionInfo;
2410 GetExceptionInfo( &exceptionInfo );
2411 MagickCore::Image* newImage =
cristy391f1ce2010-09-09 17:23:28 +00002412 ResizeImage( image(),
cristy3ed852e2009-09-05 21:47:34 +00002413 width,
2414 height,
cristy391f1ce2010-09-09 17:23:28 +00002415 image()->filter,
cristy3ed852e2009-09-05 21:47:34 +00002416 &exceptionInfo);
2417 replaceImage( newImage );
2418 throwException( exceptionInfo );
2419 (void) DestroyExceptionInfo( &exceptionInfo );
2420}
2421
2422/*
2423 * Methods for setting image attributes
2424 *
2425 */
2426
2427// Join images into a single multi-image file
2428void Magick::Image::adjoin ( const bool flag_ )
2429{
2430 modifyImage();
2431 options()->adjoin( flag_ );
2432}
2433bool Magick::Image::adjoin ( void ) const
2434{
2435 return constOptions()->adjoin();
2436}
2437
2438// Remove pixel aliasing
2439void Magick::Image::antiAlias( const bool flag_ )
2440{
2441 modifyImage();
cristyeaedf062010-05-29 22:36:02 +00002442 options()->antiAlias( static_cast<size_t>(flag_) );
cristy3ed852e2009-09-05 21:47:34 +00002443}
2444bool Magick::Image::antiAlias( void )
2445{
2446 return static_cast<bool>( options()->antiAlias( ) );
2447}
2448
2449// Animation inter-frame delay
cristyeaedf062010-05-29 22:36:02 +00002450void Magick::Image::animationDelay ( const size_t delay_ )
cristy3ed852e2009-09-05 21:47:34 +00002451{
2452 modifyImage();
2453 image()->delay = delay_;
2454}
cristyeaedf062010-05-29 22:36:02 +00002455size_t Magick::Image::animationDelay ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00002456{
2457 return constImage()->delay;
2458}
2459
2460// Number of iterations to play animation
cristyeaedf062010-05-29 22:36:02 +00002461void Magick::Image::animationIterations ( const size_t iterations_ )
cristy3ed852e2009-09-05 21:47:34 +00002462{
2463 modifyImage();
2464 image()->iterations = iterations_;
2465}
cristyeaedf062010-05-29 22:36:02 +00002466size_t Magick::Image::animationIterations ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00002467{
2468 return constImage()->iterations;
2469}
2470
2471// Access/Update a named image attribute
2472void Magick::Image::attribute ( const std::string name_,
2473 const std::string value_ )
2474{
2475 modifyImage();
cristyd15e6592011-10-15 00:13:06 +00002476 ExceptionInfo exceptionInfo;
2477 GetExceptionInfo( &exceptionInfo );
2478 SetImageProperty( image(), name_.c_str(), value_.c_str(), &exceptionInfo );
2479 throwException( exceptionInfo );
2480 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002481}
2482std::string Magick::Image::attribute ( const std::string name_ )
2483{
cristyd15e6592011-10-15 00:13:06 +00002484 ExceptionInfo exceptionInfo;
2485 GetExceptionInfo( &exceptionInfo );
2486 const char *value = GetImageProperty( constImage(), name_.c_str(),
2487 &exceptionInfo );
2488 throwException( exceptionInfo );
2489 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002490
2491 if ( value )
2492 return std::string( value );
2493
2494 return std::string(); // Intentionally no exception
2495}
2496
2497// Background color
cristy391f1ce2010-09-09 17:23:28 +00002498void Magick::Image::backgroundColor ( const Color &backgroundColor_ )
cristy3ed852e2009-09-05 21:47:34 +00002499{
2500 modifyImage();
2501
cristy391f1ce2010-09-09 17:23:28 +00002502 if ( backgroundColor_.isValid() )
cristy3ed852e2009-09-05 21:47:34 +00002503 {
cristy391f1ce2010-09-09 17:23:28 +00002504 image()->background_color = backgroundColor_;
cristy3ed852e2009-09-05 21:47:34 +00002505 }
2506 else
2507 {
cristy391f1ce2010-09-09 17:23:28 +00002508 image()->background_color = Color();
cristy3ed852e2009-09-05 21:47:34 +00002509 }
2510
cristy391f1ce2010-09-09 17:23:28 +00002511 options()->backgroundColor( backgroundColor_ );
cristy3ed852e2009-09-05 21:47:34 +00002512}
2513Magick::Color Magick::Image::backgroundColor ( void ) const
2514{
2515 return constOptions()->backgroundColor( );
2516}
2517
2518// Background fill texture
2519void Magick::Image::backgroundTexture ( const std::string &backgroundTexture_ )
2520{
2521 modifyImage();
2522 options()->backgroundTexture( backgroundTexture_ );
2523}
2524std::string Magick::Image::backgroundTexture ( void ) const
2525{
2526 return constOptions()->backgroundTexture( );
2527}
2528
2529// Original image columns
cristyeaedf062010-05-29 22:36:02 +00002530size_t Magick::Image::baseColumns ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00002531{
2532 return constImage()->magick_columns;
2533}
2534
2535// Original image name
2536std::string Magick::Image::baseFilename ( void ) const
2537{
2538 return std::string(constImage()->magick_filename);
2539}
2540
2541// Original image rows
cristyeaedf062010-05-29 22:36:02 +00002542size_t Magick::Image::baseRows ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00002543{
2544 return constImage()->magick_rows;
2545}
2546
2547// Border color
cristy391f1ce2010-09-09 17:23:28 +00002548void Magick::Image::borderColor ( const Color &borderColor_ )
cristy3ed852e2009-09-05 21:47:34 +00002549{
2550 modifyImage();
2551
cristy391f1ce2010-09-09 17:23:28 +00002552 if ( borderColor_.isValid() )
cristy3ed852e2009-09-05 21:47:34 +00002553 {
cristy391f1ce2010-09-09 17:23:28 +00002554 image()->border_color = borderColor_;
cristy3ed852e2009-09-05 21:47:34 +00002555 }
2556 else
2557 {
cristy391f1ce2010-09-09 17:23:28 +00002558 image()->border_color = Color();
cristy3ed852e2009-09-05 21:47:34 +00002559 }
2560
cristy391f1ce2010-09-09 17:23:28 +00002561 options()->borderColor( borderColor_ );
cristy3ed852e2009-09-05 21:47:34 +00002562}
2563Magick::Color Magick::Image::borderColor ( void ) const
2564{
2565 return constOptions()->borderColor( );
2566}
2567
2568// Return smallest bounding box enclosing non-border pixels. The
2569// current fuzz value is used when discriminating between pixels.
2570// This is the crop bounding box used by crop(Geometry(0,0));
2571Magick::Geometry Magick::Image::boundingBox ( void ) const
2572{
2573 ExceptionInfo exceptionInfo;
2574 GetExceptionInfo( &exceptionInfo );
2575 RectangleInfo bbox = GetImageBoundingBox( constImage(), &exceptionInfo);
2576 throwException( exceptionInfo );
2577 (void) DestroyExceptionInfo( &exceptionInfo );
2578 return Geometry( bbox );
2579}
2580
2581// Text bounding-box base color
2582void Magick::Image::boxColor ( const Color &boxColor_ )
2583{
2584 modifyImage();
2585 options()->boxColor( boxColor_ );
2586}
2587Magick::Color Magick::Image::boxColor ( void ) const
2588{
2589 return constOptions()->boxColor( );
2590}
2591
2592// Pixel cache threshold. Once this threshold is exceeded, all
2593// subsequent pixels cache operations are to/from disk.
2594// This setting is shared by all Image objects.
2595/* static */
cristyeaedf062010-05-29 22:36:02 +00002596void Magick::Image::cacheThreshold ( const size_t threshold_ )
cristy3ed852e2009-09-05 21:47:34 +00002597{
2598 SetMagickResourceLimit( MemoryResource, threshold_ );
2599}
2600
2601void Magick::Image::chromaBluePrimary ( const double x_, const double y_ )
2602{
2603 modifyImage();
2604 image()->chromaticity.blue_primary.x = x_;
2605 image()->chromaticity.blue_primary.y = y_;
2606}
2607void Magick::Image::chromaBluePrimary ( double *x_, double *y_ ) const
2608{
2609 *x_ = constImage()->chromaticity.blue_primary.x;
2610 *y_ = constImage()->chromaticity.blue_primary.y;
2611}
2612
2613void Magick::Image::chromaGreenPrimary ( const double x_, const double y_ )
2614{
2615 modifyImage();
2616 image()->chromaticity.green_primary.x = x_;
2617 image()->chromaticity.green_primary.y = y_;
2618}
2619void Magick::Image::chromaGreenPrimary ( double *x_, double *y_ ) const
2620{
2621 *x_ = constImage()->chromaticity.green_primary.x;
2622 *y_ = constImage()->chromaticity.green_primary.y;
2623}
2624
2625void Magick::Image::chromaRedPrimary ( const double x_, const double y_ )
2626{
2627 modifyImage();
2628 image()->chromaticity.red_primary.x = x_;
2629 image()->chromaticity.red_primary.y = y_;
2630}
2631void Magick::Image::chromaRedPrimary ( double *x_, double *y_ ) const
2632{
2633 *x_ = constImage()->chromaticity.red_primary.x;
2634 *y_ = constImage()->chromaticity.red_primary.y;
2635}
2636
2637void Magick::Image::chromaWhitePoint ( const double x_, const double y_ )
2638{
2639 modifyImage();
2640 image()->chromaticity.white_point.x = x_;
2641 image()->chromaticity.white_point.y = y_;
2642}
2643void Magick::Image::chromaWhitePoint ( double *x_, double *y_ ) const
2644{
2645 *x_ = constImage()->chromaticity.white_point.x;
2646 *y_ = constImage()->chromaticity.white_point.y;
2647}
2648
2649// Set image storage class
2650void Magick::Image::classType ( const ClassType class_ )
2651{
2652 if ( classType() == PseudoClass && class_ == DirectClass )
2653 {
2654 // Use SyncImage to synchronize the DirectClass pixels with the
2655 // color map and then set to DirectClass type.
2656 modifyImage();
cristyea1a8aa2011-10-20 13:24:06 +00002657 ExceptionInfo exceptionInfo;
2658 GetExceptionInfo( &exceptionInfo );
2659 SyncImage( image(), &exceptionInfo );
2660 throwException( exceptionInfo );
2661 (void) DestroyExceptionInfo( &exceptionInfo );
cristy101ab702011-10-13 13:06:32 +00002662 image()->colormap = (PixelInfo *)
cristy3ed852e2009-09-05 21:47:34 +00002663 RelinquishMagickMemory( image()->colormap );
2664 image()->storage_class = static_cast<MagickCore::ClassType>(DirectClass);
2665 return;
2666 }
2667
2668 if ( classType() == DirectClass && class_ == PseudoClass )
2669 {
2670 // Quantize to create PseudoClass color map
2671 modifyImage();
cristye6bbc092010-05-12 17:00:47 +00002672 quantizeColors(MaxColormapSize);
cristy3ed852e2009-09-05 21:47:34 +00002673 quantize();
2674 image()->storage_class = static_cast<MagickCore::ClassType>(PseudoClass);
2675 }
2676}
2677
2678// Associate a clip mask with the image. The clip mask must be the
2679// same dimensions as the image. Pass an invalid image to unset an
2680// existing clip mask.
2681void Magick::Image::clipMask ( const Magick::Image & clipMask_ )
2682{
2683 modifyImage();
2684
cristy018f07f2011-09-04 21:15:19 +00002685 ExceptionInfo exceptionInfo;
2686 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002687 if( clipMask_.isValid() )
2688 {
2689 // Set clip mask
cristy10a6c612012-01-29 21:41:05 +00002690 SetImageMask( image(), clipMask_.constImage(), &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002691 }
2692 else
2693 {
2694 // Unset existing clip mask
cristy10a6c612012-01-29 21:41:05 +00002695 SetImageMask( image(), 0, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002696 }
cristy018f07f2011-09-04 21:15:19 +00002697 throwException( exceptionInfo );
2698 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002699}
2700Magick::Image Magick::Image::clipMask ( void ) const
2701{
cristy018f07f2011-09-04 21:15:19 +00002702 ExceptionInfo exceptionInfo;
2703 GetExceptionInfo( &exceptionInfo );
cristyc94ba6f2012-01-29 23:19:58 +00002704 MagickCore::Image* image = GetImageMask( constImage(), &exceptionInfo );
cristy018f07f2011-09-04 21:15:19 +00002705 throwException( exceptionInfo );
2706 (void) DestroyExceptionInfo( &exceptionInfo );
2707 return Magick::Image( image );
cristy3ed852e2009-09-05 21:47:34 +00002708}
2709
2710void Magick::Image::colorFuzz ( const double fuzz_ )
2711{
2712 modifyImage();
2713 image()->fuzz = fuzz_;
2714 options()->colorFuzz( fuzz_ );
2715}
2716double Magick::Image::colorFuzz ( void ) const
2717{
2718 return constOptions()->colorFuzz( );
2719}
2720
2721// Set color in colormap at index
cristyeaedf062010-05-29 22:36:02 +00002722void Magick::Image::colorMap ( const size_t index_,
cristy3ed852e2009-09-05 21:47:34 +00002723 const Color &color_ )
2724{
2725 MagickCore::Image* imageptr = image();
2726
2727 if (index_ > (MaxColormapSize-1) )
2728 throwExceptionExplicit( OptionError,
2729 "Colormap index must be less than MaxColormapSize" );
2730
2731 if ( !color_.isValid() )
2732 throwExceptionExplicit( OptionError,
2733 "Color argument is invalid");
2734 modifyImage();
2735
2736 // Ensure that colormap size is large enough
2737 if ( colorMapSize() < (index_+1) )
2738 colorMapSize( index_ + 1 );
2739
2740 // Set color at index in colormap
2741 (imageptr->colormap)[index_] = color_;
2742}
2743// Return color in colormap at index
cristyeaedf062010-05-29 22:36:02 +00002744Magick::Color Magick::Image::colorMap ( const size_t index_ ) const
cristy3ed852e2009-09-05 21:47:34 +00002745{
2746 const MagickCore::Image* imageptr = constImage();
2747
2748 if ( !imageptr->colormap )
2749 throwExceptionExplicit( OptionError,
2750 "Image does not contain a colormap");
2751
2752 if ( index_ > imageptr->colors-1 )
2753 throwExceptionExplicit( OptionError,
2754 "Index out of range");
2755
2756 return Magick::Color( (imageptr->colormap)[index_] );
2757}
2758
2759// Colormap size (number of colormap entries)
cristyeaedf062010-05-29 22:36:02 +00002760void Magick::Image::colorMapSize ( const size_t entries_ )
cristy3ed852e2009-09-05 21:47:34 +00002761{
2762 if (entries_ >MaxColormapSize )
2763 throwExceptionExplicit( OptionError,
2764 "Colormap entries must not exceed MaxColormapSize" );
2765
2766 modifyImage();
2767
2768 MagickCore::Image* imageptr = image();
2769
2770 if( !imageptr->colormap )
2771 {
2772 // Allocate colormap
2773 imageptr->colormap =
cristy101ab702011-10-13 13:06:32 +00002774 static_cast<PixelInfo*>(AcquireMagickMemory(entries_*sizeof(PixelInfo)));
cristy3ed852e2009-09-05 21:47:34 +00002775 imageptr->colors = 0;
2776 }
2777 else if ( entries_ > imageptr->colors )
2778 {
2779 // Re-allocate colormap
cristy101ab702011-10-13 13:06:32 +00002780 imageptr->colormap=(PixelInfo *)
2781 ResizeMagickMemory(imageptr->colormap,(entries_)*sizeof(PixelInfo));
cristy3ed852e2009-09-05 21:47:34 +00002782 }
2783
2784 // Initialize any new colormap entries as all black
2785 Color black(0,0,0);
cristyeaedf062010-05-29 22:36:02 +00002786 for( size_t i=imageptr->colors; i<(entries_-1); i++ )
cristy3ed852e2009-09-05 21:47:34 +00002787 (imageptr->colormap)[i] = black;
2788
2789 imageptr->colors = entries_;
2790}
cristyeaedf062010-05-29 22:36:02 +00002791size_t Magick::Image::colorMapSize ( void )
cristy3ed852e2009-09-05 21:47:34 +00002792{
2793 const MagickCore::Image* imageptr = constImage();
2794
2795 if ( !imageptr->colormap )
2796 throwExceptionExplicit( OptionError,
2797 "Image does not contain a colormap");
2798
2799 return imageptr->colors;
2800}
2801
2802// Image colorspace
2803void Magick::Image::colorSpace( const ColorspaceType colorSpace_ )
2804{
cristy68f837b2012-04-18 14:46:38 +00002805 if ( image()->colorspace == colorSpace_ )
2806 return;
cristy3ed852e2009-09-05 21:47:34 +00002807
cristy68f837b2012-04-18 14:46:38 +00002808 modifyImage();
cristy18873882011-10-18 12:54:53 +00002809 ExceptionInfo exceptionInfo;
2810 GetExceptionInfo( &exceptionInfo );
cristya45d6552012-04-18 14:44:15 +00002811 TransformImageColorspace(image(), colorSpace_, &exceptionInfo);
cristy18873882011-10-18 12:54:53 +00002812 throwException( exceptionInfo );
2813 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002814}
2815Magick::ColorspaceType Magick::Image::colorSpace ( void ) const
2816{
2817 return constImage()->colorspace;
2818}
2819
2820// Set image colorspace type.
2821void Magick::Image::colorspaceType( const ColorspaceType colorSpace_ )
2822{
2823 modifyImage();
2824 options()->colorspaceType( colorSpace_ );
2825}
2826Magick::ColorspaceType Magick::Image::colorspaceType ( void ) const
2827{
2828 return constOptions()->colorspaceType();
2829}
2830
2831
2832// Comment string
2833void Magick::Image::comment ( const std::string &comment_ )
2834{
2835 modifyImage();
cristyd15e6592011-10-15 00:13:06 +00002836 ExceptionInfo exceptionInfo;
2837 GetExceptionInfo( &exceptionInfo );
2838 SetImageProperty( image(), "Comment", NULL, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002839 if ( comment_.length() > 0 )
cristyd15e6592011-10-15 00:13:06 +00002840 SetImageProperty( image(), "Comment", comment_.c_str(), &exceptionInfo );
2841 throwException( exceptionInfo );
2842 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002843}
2844std::string Magick::Image::comment ( void ) const
2845{
cristyd15e6592011-10-15 00:13:06 +00002846 ExceptionInfo exceptionInfo;
2847 GetExceptionInfo( &exceptionInfo );
2848 const char *value = GetImageProperty( constImage(), "Comment",
2849 &exceptionInfo );
2850 throwException( exceptionInfo );
2851 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00002852
2853 if ( value )
2854 return std::string( value );
2855
2856 return std::string(); // Intentionally no exception
2857}
2858
2859// Composition operator to be used when composition is implicitly used
2860// (such as for image flattening).
2861void Magick::Image::compose (const CompositeOperator compose_)
2862{
2863 image()->compose=compose_;
2864}
2865
2866Magick::CompositeOperator Magick::Image::compose ( void ) const
2867{
2868 return constImage()->compose;
2869}
2870
2871// Compression algorithm
2872void Magick::Image::compressType ( const CompressionType compressType_ )
2873{
2874 modifyImage();
2875 image()->compression = compressType_;
2876 options()->compressType( compressType_ );
2877}
2878Magick::CompressionType Magick::Image::compressType ( void ) const
2879{
2880 return constImage()->compression;
2881}
2882
2883// Enable printing of debug messages from ImageMagick
2884void Magick::Image::debug ( const bool flag_ )
2885{
2886 modifyImage();
2887 options()->debug( flag_ );
2888}
2889bool Magick::Image::debug ( void ) const
2890{
2891 return constOptions()->debug();
2892}
2893
2894// Tagged image format define (set/access coder-specific option) The
2895// magick_ option specifies the coder the define applies to. The key_
2896// option provides the key specific to that coder. The value_ option
2897// provides the value to set (if any). See the defineSet() method if the
2898// key must be removed entirely.
2899void Magick::Image::defineValue ( const std::string &magick_,
2900 const std::string &key_,
2901 const std::string &value_ )
2902{
2903 modifyImage();
2904 std::string format = magick_ + ":" + key_;
2905 std::string option = value_;
2906 (void) SetImageOption ( imageInfo(), format.c_str(), option.c_str() );
2907}
2908std::string Magick::Image::defineValue ( const std::string &magick_,
2909 const std::string &key_ ) const
2910{
2911 std::string definition = magick_ + ":" + key_;
2912 const char *option =
2913 GetImageOption ( constImageInfo(), definition.c_str() );
2914 if (option)
2915 return std::string( option );
2916 return std::string( );
2917}
2918
2919// Tagged image format define. Similar to the defineValue() method
2920// except that passing the flag_ value 'true' creates a value-less
2921// define with that format and key. Passing the flag_ value 'false'
2922// removes any existing matching definition. The method returns 'true'
2923// if a matching key exists, and 'false' if no matching key exists.
2924void Magick::Image::defineSet ( const std::string &magick_,
2925 const std::string &key_,
2926 bool flag_ )
2927{
2928 modifyImage();
2929 std::string definition = magick_ + ":" + key_;
2930 if (flag_)
2931 {
2932 (void) SetImageOption ( imageInfo(), definition.c_str(), "" );
2933 }
2934 else
2935 {
2936 DeleteImageOption( imageInfo(), definition.c_str() );
2937 }
2938}
2939bool Magick::Image::defineSet ( const std::string &magick_,
2940 const std::string &key_ ) const
2941{
2942 std::string key = magick_ + ":" + key_;
2943 const char *option =
2944 GetImageOption ( constImageInfo(), key.c_str() );
2945 if (option)
2946 return true;
2947 return false;
2948}
2949
2950// Pixel resolution
2951void Magick::Image::density ( const Geometry &density_ )
2952{
2953 modifyImage();
2954 options()->density( density_ );
2955 if ( density_.isValid() )
2956 {
cristy2a11bef2011-10-28 18:33:11 +00002957 image()->resolution.x = density_.width();
cristy3ed852e2009-09-05 21:47:34 +00002958 if ( density_.height() != 0 )
2959 {
cristy2a11bef2011-10-28 18:33:11 +00002960 image()->resolution.y = density_.height();
cristy3ed852e2009-09-05 21:47:34 +00002961 }
2962 else
2963 {
cristy2a11bef2011-10-28 18:33:11 +00002964 image()->resolution.y = density_.width();
cristy3ed852e2009-09-05 21:47:34 +00002965 }
2966 }
2967 else
2968 {
2969 // Reset to default
cristy2a11bef2011-10-28 18:33:11 +00002970 image()->resolution.x = 0;
2971 image()->resolution.y = 0;
cristy3ed852e2009-09-05 21:47:34 +00002972 }
2973}
2974Magick::Geometry Magick::Image::density ( void ) const
2975{
2976 if (isValid())
2977 {
cristy35ef8242010-06-03 16:24:13 +00002978 ssize_t x_resolution=72;
2979 ssize_t y_resolution=72;
cristy3ed852e2009-09-05 21:47:34 +00002980
cristy2a11bef2011-10-28 18:33:11 +00002981 if (constImage()->resolution.x > 0.0)
2982 x_resolution=static_cast<ssize_t>(constImage()->resolution.x + 0.5);
cristy3ed852e2009-09-05 21:47:34 +00002983
cristy2a11bef2011-10-28 18:33:11 +00002984 if (constImage()->resolution.y > 0.0)
2985 y_resolution=static_cast<ssize_t>(constImage()->resolution.y + 0.5);
cristy3ed852e2009-09-05 21:47:34 +00002986
2987 return Geometry(x_resolution,y_resolution);
2988 }
2989
2990 return constOptions()->density( );
2991}
2992
2993// Image depth (bits allocated to red/green/blue components)
cristyeaedf062010-05-29 22:36:02 +00002994void Magick::Image::depth ( const size_t depth_ )
cristy3ed852e2009-09-05 21:47:34 +00002995{
cristyeaedf062010-05-29 22:36:02 +00002996 size_t depth = depth_;
cristy3ed852e2009-09-05 21:47:34 +00002997
2998 if (depth > MAGICKCORE_QUANTUM_DEPTH)
2999 depth=MAGICKCORE_QUANTUM_DEPTH;
3000
3001 modifyImage();
3002 image()->depth=depth;
3003 options()->depth( depth );
3004}
cristyeaedf062010-05-29 22:36:02 +00003005size_t Magick::Image::depth ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003006{
3007 return constImage()->depth;
3008}
3009
3010std::string Magick::Image::directory ( void ) const
3011{
3012 if ( constImage()->directory )
3013 return std::string( constImage()->directory );
3014
3015 throwExceptionExplicit( CorruptImageWarning,
3016 "Image does not contain a directory");
3017
3018 return std::string();
3019}
3020
3021// Endianness (little like Intel or big like SPARC) for image
3022// formats which support endian-specific options.
3023void Magick::Image::endian ( const Magick::EndianType endian_ )
3024{
3025 modifyImage();
3026 options()->endian( endian_ );
3027 image()->endian = endian_;
3028}
3029Magick::EndianType Magick::Image::endian ( void ) const
3030{
3031 return constImage()->endian;
3032}
3033
3034// EXIF profile (BLOB)
3035void Magick::Image::exifProfile( const Magick::Blob &exifProfile_ )
3036{
3037 modifyImage();
3038 if ( exifProfile_.data() != 0 )
3039 {
3040 StringInfo * exif_profile = AcquireStringInfo( exifProfile_.length() );
3041 SetStringInfoDatum(exif_profile ,(unsigned char *) exifProfile_.data());
cristyd15e6592011-10-15 00:13:06 +00003042 ExceptionInfo exceptionInfo;
3043 GetExceptionInfo( &exceptionInfo );
3044 (void) SetImageProfile( image(), "exif", exif_profile, &exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00003045 exif_profile =DestroyStringInfo( exif_profile );
cristyd15e6592011-10-15 00:13:06 +00003046 throwException( exceptionInfo );
3047 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003048 }
3049}
3050Magick::Blob Magick::Image::exifProfile( void ) const
3051{
3052 const StringInfo * exif_profile = GetImageProfile( constImage(), "exif" );
3053 if ( exif_profile == (StringInfo *) NULL)
3054 return Blob( 0, 0 );
3055 return Blob(GetStringInfoDatum(exif_profile),GetStringInfoLength(exif_profile));
3056}
3057
3058// Image file name
3059void Magick::Image::fileName ( const std::string &fileName_ )
3060{
3061 modifyImage();
3062
3063 fileName_.copy( image()->filename,
3064 sizeof(image()->filename) - 1 );
3065 image()->filename[ fileName_.length() ] = 0; // Null terminate
3066
3067 options()->fileName( fileName_ );
3068
3069}
3070std::string Magick::Image::fileName ( void ) const
3071{
3072 return constOptions()->fileName( );
3073}
3074
3075// Image file size
3076off_t Magick::Image::fileSize ( void ) const
3077{
3078 return (off_t) GetBlobSize( constImage() );
3079}
3080
3081// Color to use when drawing inside an object
3082void Magick::Image::fillColor ( const Magick::Color &fillColor_ )
3083{
3084 modifyImage();
3085 options()->fillColor(fillColor_);
3086}
3087Magick::Color Magick::Image::fillColor ( void ) const
3088{
3089 return constOptions()->fillColor();
3090}
3091
3092// Rule to use when filling drawn objects
3093void Magick::Image::fillRule ( const Magick::FillRule &fillRule_ )
3094{
3095 modifyImage();
3096 options()->fillRule(fillRule_);
3097}
3098Magick::FillRule Magick::Image::fillRule ( void ) const
3099{
3100 return constOptions()->fillRule();
3101}
3102
3103// Pattern to use while filling drawn objects.
3104void Magick::Image::fillPattern ( const Image &fillPattern_ )
3105{
3106 modifyImage();
3107 if(fillPattern_.isValid())
3108 options()->fillPattern( fillPattern_.constImage() );
3109 else
3110 options()->fillPattern( static_cast<MagickCore::Image*>(NULL) );
3111}
3112Magick::Image Magick::Image::fillPattern ( void ) const
3113{
3114 // FIXME: This is inordinately innefficient
3115 Image texture;
3116
3117 const MagickCore::Image* tmpTexture = constOptions()->fillPattern( );
3118
3119 if ( tmpTexture )
3120 {
3121 ExceptionInfo exceptionInfo;
3122 GetExceptionInfo( &exceptionInfo );
3123 MagickCore::Image* image =
3124 CloneImage( tmpTexture,
3125 0, // columns
3126 0, // rows
3127 MagickTrue, // orphan
3128 &exceptionInfo);
3129 texture.replaceImage( image );
3130 throwException( exceptionInfo );
cristyd15e6592011-10-15 00:13:06 +00003131 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003132 }
3133 return texture;
3134}
3135
3136// Filter used by zoom
3137void Magick::Image::filterType ( const Magick::FilterTypes filterType_ )
3138{
3139 modifyImage();
3140 image()->filter = filterType_;
3141}
3142Magick::FilterTypes Magick::Image::filterType ( void ) const
3143{
3144 return constImage()->filter;
3145}
3146
3147// Font name
3148void Magick::Image::font ( const std::string &font_ )
3149{
3150 modifyImage();
3151 options()->font( font_ );
3152}
3153std::string Magick::Image::font ( void ) const
3154{
3155 return constOptions()->font( );
3156}
3157
3158// Font point size
3159void Magick::Image::fontPointsize ( const double pointSize_ )
3160{
3161 modifyImage();
3162 options()->fontPointsize( pointSize_ );
3163}
3164double Magick::Image::fontPointsize ( void ) const
3165{
3166 return constOptions()->fontPointsize( );
3167}
3168
3169// Font type metrics
3170void Magick::Image::fontTypeMetrics( const std::string &text_,
3171 TypeMetric *metrics )
3172{
3173 DrawInfo *drawInfo = options()->drawInfo();
3174 drawInfo->text = const_cast<char *>(text_.c_str());
cristy5cbc0162011-08-29 00:36:28 +00003175 ExceptionInfo exceptionInfo;
3176 GetExceptionInfo( &exceptionInfo );
3177 GetTypeMetrics( image(), drawInfo, &(metrics->_typeMetric), &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003178 drawInfo->text = 0;
cristy5cbc0162011-08-29 00:36:28 +00003179 throwException( exceptionInfo );
3180 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003181}
3182
3183// Image format string
3184std::string Magick::Image::format ( void ) const
3185{
3186 ExceptionInfo exceptionInfo;
3187 GetExceptionInfo( &exceptionInfo );
3188 const MagickInfo * magick_info
3189 = GetMagickInfo( constImage()->magick, &exceptionInfo);
3190 throwException( exceptionInfo );
3191 (void) DestroyExceptionInfo( &exceptionInfo );
3192
3193 if (( magick_info != 0 ) &&
3194 ( *magick_info->description != '\0' ))
3195 return std::string(magick_info->description);
3196
3197 throwExceptionExplicit( CorruptImageWarning,
3198 "Unrecognized image magick type" );
3199 return std::string();
3200}
3201
3202// Gamma adjustment
3203double Magick::Image::gamma ( void ) const
3204{
3205 return constImage()->gamma;
3206}
3207
3208Magick::Geometry Magick::Image::geometry ( void ) const
3209{
3210 if ( constImage()->geometry )
3211 {
3212 return Geometry(constImage()->geometry);
3213 }
3214
3215 throwExceptionExplicit( OptionWarning,
3216 "Image does not contain a geometry");
3217
3218 return Geometry();
3219}
3220
cristyeaedf062010-05-29 22:36:02 +00003221void Magick::Image::gifDisposeMethod ( const size_t disposeMethod_ )
cristy3ed852e2009-09-05 21:47:34 +00003222{
3223 modifyImage();
3224 image()->dispose = (DisposeType) disposeMethod_;
3225}
cristyeaedf062010-05-29 22:36:02 +00003226size_t Magick::Image::gifDisposeMethod ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003227{
3228 // FIXME: It would be better to return an enumeration
3229 return constImage()->dispose;
3230}
3231
3232// ICC ICM color profile (BLOB)
3233void Magick::Image::iccColorProfile( const Magick::Blob &colorProfile_ )
3234{
3235 profile("icm",colorProfile_);
3236}
3237Magick::Blob Magick::Image::iccColorProfile( void ) const
3238{
3239 const StringInfo * color_profile = GetImageProfile( constImage(), "icc" );
3240 if ( color_profile == (StringInfo *) NULL)
3241 return Blob( 0, 0 );
3242 return Blob( GetStringInfoDatum(color_profile), GetStringInfoLength(color_profile) );
3243}
3244
3245void Magick::Image::interlaceType ( const Magick::InterlaceType interlace_ )
3246{
3247 modifyImage();
3248 image()->interlace = interlace_;
3249 options()->interlaceType ( interlace_ );
3250}
3251Magick::InterlaceType Magick::Image::interlaceType ( void ) const
3252{
3253 return constImage()->interlace;
3254}
3255
3256// IPTC profile (BLOB)
3257void Magick::Image::iptcProfile( const Magick::Blob &iptcProfile_ )
3258{
3259 modifyImage();
3260 if ( iptcProfile_.data() != 0 )
3261 {
3262 StringInfo * iptc_profile = AcquireStringInfo( iptcProfile_.length() );
3263 SetStringInfoDatum(iptc_profile ,(unsigned char *) iptcProfile_.data());
cristyd15e6592011-10-15 00:13:06 +00003264 ExceptionInfo exceptionInfo;
3265 GetExceptionInfo( &exceptionInfo );
3266 (void) SetImageProfile( image(), "iptc", iptc_profile, &exceptionInfo);
3267 iptc_profile =DestroyStringInfo( iptc_profile );
3268 throwException( exceptionInfo );
3269 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003270 }
3271}
3272Magick::Blob Magick::Image::iptcProfile( void ) const
3273{
3274 const StringInfo * iptc_profile = GetImageProfile( constImage(), "iptc" );
3275 if ( iptc_profile == (StringInfo *) NULL)
3276 return Blob( 0, 0 );
3277 return Blob( GetStringInfoDatum(iptc_profile), GetStringInfoLength(iptc_profile));
3278}
3279
3280// Does object contain valid image?
3281void Magick::Image::isValid ( const bool isValid_ )
3282{
3283 if ( !isValid_ )
3284 {
3285 delete _imgRef;
3286 _imgRef = new ImageRef;
3287 }
3288 else if ( !isValid() )
3289 {
3290 // Construct with single-pixel black image to make
3291 // image valid. This is an obvious hack.
3292 size( Geometry(1,1) );
3293 read( "xc:#000000" );
3294 }
3295}
3296
3297bool Magick::Image::isValid ( void ) const
3298{
3299 if ( rows() && columns() )
3300 return true;
3301
3302 return false;
3303}
3304
3305// Label image
3306void Magick::Image::label ( const std::string &label_ )
3307{
3308 modifyImage();
cristyd15e6592011-10-15 00:13:06 +00003309 ExceptionInfo exceptionInfo;
3310 GetExceptionInfo( &exceptionInfo );
3311 SetImageProperty ( image(), "Label", NULL, &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003312 if ( label_.length() > 0 )
cristyd15e6592011-10-15 00:13:06 +00003313 SetImageProperty ( image(), "Label", label_.c_str(), &exceptionInfo );
3314 throwException( exceptionInfo );
3315 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003316}
3317std::string Magick::Image::label ( void ) const
3318{
cristyd15e6592011-10-15 00:13:06 +00003319 ExceptionInfo exceptionInfo;
3320 GetExceptionInfo( &exceptionInfo );
3321 const char *value = GetImageProperty( constImage(), "Label", &exceptionInfo );
3322 throwException( exceptionInfo );
3323 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003324
3325 if ( value )
3326 return std::string( value );
3327
3328 return std::string();
3329}
3330
3331void Magick::Image::magick ( const std::string &magick_ )
3332{
3333 modifyImage();
3334
3335 magick_.copy( image()->magick,
3336 sizeof(image()->magick) - 1 );
3337 image()->magick[ magick_.length() ] = 0;
3338
3339 options()->magick( magick_ );
3340}
3341std::string Magick::Image::magick ( void ) const
3342{
3343 if ( *(constImage()->magick) != '\0' )
3344 return std::string(constImage()->magick);
3345
3346 return constOptions()->magick( );
3347}
3348
3349void Magick::Image::matte ( const bool matteFlag_ )
3350{
3351 modifyImage();
3352
3353 // If matte channel is requested, but image doesn't already have a
3354 // matte channel, then create an opaque matte channel. Likewise, if
3355 // the image already has a matte channel but a matte channel is not
3356 // desired, then set the matte channel to opaque.
cristye941a752011-10-15 01:52:48 +00003357 ExceptionInfo exceptionInfo;
3358 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003359 if ((matteFlag_ && !constImage()->matte) ||
3360 (constImage()->matte && !matteFlag_))
cristye941a752011-10-15 01:52:48 +00003361 SetImageAlpha(image(),OpaqueAlpha,&exceptionInfo);
3362 throwException( exceptionInfo );
3363 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003364
3365 image()->matte = (MagickBooleanType) matteFlag_;
3366}
3367bool Magick::Image::matte ( void ) const
3368{
3369 if ( constImage()->matte )
3370 return true;
3371 else
3372 return false;
3373}
3374
3375void Magick::Image::matteColor ( const Color &matteColor_ )
3376{
3377 modifyImage();
3378
3379 if ( matteColor_.isValid() )
3380 {
cristy391f1ce2010-09-09 17:23:28 +00003381 image()->matte_color = matteColor_;
3382 options()->matteColor( matteColor_ );
cristy3ed852e2009-09-05 21:47:34 +00003383 }
3384 else
3385 {
3386 // Set to default matte color
3387 Color tmpColor( "#BDBDBD" );
cristy391f1ce2010-09-09 17:23:28 +00003388 image()->matte_color = tmpColor;
cristy3ed852e2009-09-05 21:47:34 +00003389 options()->matteColor( tmpColor );
3390 }
3391}
3392Magick::Color Magick::Image::matteColor ( void ) const
3393{
cristy9c0b6f72012-01-01 23:14:31 +00003394 return Color( ClampToQuantum( constImage()->matte_color.red ),
3395 ClampToQuantum( constImage()->matte_color.green ),
3396 ClampToQuantum( constImage()->matte_color.blue ) );
cristy3ed852e2009-09-05 21:47:34 +00003397}
3398
3399double Magick::Image::meanErrorPerPixel ( void ) const
3400{
3401 return(constImage()->error.mean_error_per_pixel);
3402}
3403
3404// Image modulus depth (minimum number of bits required to support
3405// red/green/blue components without loss of accuracy)
cristyeaedf062010-05-29 22:36:02 +00003406void Magick::Image::modulusDepth ( const size_t depth_ )
cristy3ed852e2009-09-05 21:47:34 +00003407{
3408 modifyImage();
cristy8a11cb12011-10-19 23:53:34 +00003409 ExceptionInfo exceptionInfo;
3410 GetExceptionInfo( &exceptionInfo );
3411 SetImageDepth( image(), depth_, &exceptionInfo );
3412 throwException( exceptionInfo );
3413 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003414 options()->depth( depth_ );
3415}
cristyeaedf062010-05-29 22:36:02 +00003416size_t Magick::Image::modulusDepth ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003417{
3418 ExceptionInfo exceptionInfo;
3419 GetExceptionInfo( &exceptionInfo );
cristyeaedf062010-05-29 22:36:02 +00003420 size_t depth=GetImageDepth( constImage(), &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003421 throwException( exceptionInfo );
3422 (void) DestroyExceptionInfo( &exceptionInfo );
3423 return depth;
3424}
3425
3426void Magick::Image::monochrome ( const bool monochromeFlag_ )
3427{
3428 modifyImage();
3429 options()->monochrome( monochromeFlag_ );
3430}
3431bool Magick::Image::monochrome ( void ) const
3432{
3433 return constOptions()->monochrome( );
3434}
3435
3436Magick::Geometry Magick::Image::montageGeometry ( void ) const
3437{
3438 if ( constImage()->montage )
3439 return Magick::Geometry(constImage()->montage);
3440
3441 throwExceptionExplicit( CorruptImageWarning,
3442 "Image does not contain a montage" );
3443
3444 return Magick::Geometry();
3445}
3446
3447double Magick::Image::normalizedMaxError ( void ) const
3448{
3449 return(constImage()->error.normalized_maximum_error);
3450}
3451
3452double Magick::Image::normalizedMeanError ( void ) const
3453{
3454 return constImage()->error.normalized_mean_error;
3455}
3456
3457// Image orientation
3458void Magick::Image::orientation ( const Magick::OrientationType orientation_ )
3459{
3460 modifyImage();
3461 image()->orientation = orientation_;
3462}
3463Magick::OrientationType Magick::Image::orientation ( void ) const
3464{
3465 return constImage()->orientation;
3466}
3467
3468void Magick::Image::penColor ( const Color &penColor_ )
3469{
3470 modifyImage();
3471 options()->fillColor(penColor_);
3472 options()->strokeColor(penColor_);
3473}
3474Magick::Color Magick::Image::penColor ( void ) const
3475{
3476 return constOptions()->fillColor();
3477}
3478
3479void Magick::Image::penTexture ( const Image &penTexture_ )
3480{
3481 modifyImage();
3482 if(penTexture_.isValid())
3483 options()->fillPattern( penTexture_.constImage() );
3484 else
3485 options()->fillPattern( static_cast<MagickCore::Image*>(NULL) );
3486}
3487
3488Magick::Image Magick::Image::penTexture ( void ) const
3489{
3490 // FIXME: This is inordinately innefficient
3491 Image texture;
3492
3493 const MagickCore::Image* tmpTexture = constOptions()->fillPattern( );
3494
3495 if ( tmpTexture )
3496 {
3497 ExceptionInfo exceptionInfo;
3498 GetExceptionInfo( &exceptionInfo );
3499 MagickCore::Image* image =
3500 CloneImage( tmpTexture,
3501 0, // columns
3502 0, // rows
3503 MagickTrue, // orphan
3504 &exceptionInfo);
3505 texture.replaceImage( image );
3506 throwException( exceptionInfo );
3507 (void) DestroyExceptionInfo( &exceptionInfo );
3508 }
3509 return texture;
3510}
3511
3512// Set the color of a pixel.
cristy35ef8242010-06-03 16:24:13 +00003513void Magick::Image::pixelColor ( const ssize_t x_, const ssize_t y_,
cristy3ed852e2009-09-05 21:47:34 +00003514 const Color &color_ )
3515{
3516 // Test arguments to ensure they are within the image.
cristy07fb9182010-06-06 23:37:14 +00003517 if ( y_ > (ssize_t) rows() || x_ > (ssize_t) columns() )
cristy3ed852e2009-09-05 21:47:34 +00003518 throwExceptionExplicit( OptionError,
3519 "Access outside of image boundary" );
3520
3521 modifyImage();
3522
3523 // Set image to DirectClass
3524 classType( DirectClass );
3525
3526 // Get pixel view
3527 Pixels pixels(*this);
3528 // Set pixel value
cristy4c08aed2011-07-01 19:47:50 +00003529 Quantum *pixel = pixels.get(x_, y_, 1, 1 );
cristy101ab702011-10-13 13:06:32 +00003530 PixelInfo packet = color_;
cristy803640d2011-11-17 02:11:32 +00003531 MagickCore::SetPixelInfoPixel(constImage(),&packet,pixel);
cristy3ed852e2009-09-05 21:47:34 +00003532 // Tell ImageMagick that pixels have been updated
3533 pixels.sync();
3534
3535 return;
3536}
3537
3538// Get the color of a pixel
cristy35ef8242010-06-03 16:24:13 +00003539Magick::Color Magick::Image::pixelColor ( const ssize_t x_,
3540 const ssize_t y_ ) const
cristy3ed852e2009-09-05 21:47:34 +00003541{
3542 ClassType storage_class;
3543 storage_class = classType();
cristye07716f2012-04-08 02:09:18 +00003544 (void) storage_class;
cristy3ed852e2009-09-05 21:47:34 +00003545 // DirectClass
cristy4c08aed2011-07-01 19:47:50 +00003546 const Quantum* pixel = getConstPixels( x_, y_, 1, 1 );
3547 if ( pixel )
cristy3ed852e2009-09-05 21:47:34 +00003548 {
cristy101ab702011-10-13 13:06:32 +00003549 PixelInfo packet;
3550 MagickCore::GetPixelInfoPixel(constImage(),pixel,&packet);
cristy4c08aed2011-07-01 19:47:50 +00003551 return Color( packet );
cristy3ed852e2009-09-05 21:47:34 +00003552 }
3553
3554 return Color(); // invalid
3555}
3556
3557// Preferred size and location of an image canvas.
3558void Magick::Image::page ( const Magick::Geometry &pageSize_ )
3559{
3560 modifyImage();
3561 options()->page( pageSize_ );
3562 image()->page = pageSize_;
3563}
3564Magick::Geometry Magick::Image::page ( void ) const
3565{
3566 return Geometry( constImage()->page.width,
3567 constImage()->page.height,
3568 AbsoluteValue(constImage()->page.x),
3569 AbsoluteValue(constImage()->page.y),
3570 constImage()->page.x < 0 ? true : false,
3571 constImage()->page.y < 0 ? true : false);
3572}
3573
3574// Add a named profile to an image or remove a named profile by
3575// passing an empty Blob (use default Blob constructor).
3576// Valid names are:
3577// "*", "8BIM", "ICM", "IPTC", or a generic profile name.
3578void Magick::Image::profile( const std::string name_,
3579 const Magick::Blob &profile_ )
3580{
3581 modifyImage();
cristy092d71c2011-10-14 18:01:29 +00003582 ExceptionInfo exceptionInfo;
3583 GetExceptionInfo( &exceptionInfo );
cristyd99b0962010-05-29 23:14:26 +00003584 ssize_t result = ProfileImage( image(), name_.c_str(),
cristy3ed852e2009-09-05 21:47:34 +00003585 (unsigned char *)profile_.data(),
cristy092d71c2011-10-14 18:01:29 +00003586 profile_.length(), &exceptionInfo);
cristye07716f2012-04-08 02:09:18 +00003587 (void) result;
cristy092d71c2011-10-14 18:01:29 +00003588 throwException( exceptionInfo );
3589 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003590
cristy3ed852e2009-09-05 21:47:34 +00003591}
3592
3593// Retrieve a named profile from the image.
3594// Valid names are:
3595// "8BIM", "8BIMTEXT", "APP1", "APP1JPEG", "ICC", "ICM", & "IPTC" or
3596// an existing generic profile name.
3597Magick::Blob Magick::Image::profile( const std::string name_ ) const
3598{
3599 const MagickCore::Image* image = constImage();
3600
3601 const StringInfo * profile = GetImageProfile( image, name_.c_str() );
3602
3603 if ( profile != (StringInfo *) NULL)
3604 return Blob( (void*) GetStringInfoDatum(profile), GetStringInfoLength(profile));
3605
3606 Blob blob;
3607 Image temp_image = *this;
3608 temp_image.write( &blob, name_ );
3609 return blob;
3610}
3611
cristyeaedf062010-05-29 22:36:02 +00003612void Magick::Image::quality ( const size_t quality_ )
cristy3ed852e2009-09-05 21:47:34 +00003613{
3614 modifyImage();
3615 image()->quality = quality_;
3616 options()->quality( quality_ );
3617}
cristyeaedf062010-05-29 22:36:02 +00003618size_t Magick::Image::quality ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003619{
3620 return constImage()->quality;
3621}
3622
cristyeaedf062010-05-29 22:36:02 +00003623void Magick::Image::quantizeColors ( const size_t colors_ )
cristy3ed852e2009-09-05 21:47:34 +00003624{
3625 modifyImage();
3626 options()->quantizeColors( colors_ );
3627}
cristyeaedf062010-05-29 22:36:02 +00003628size_t Magick::Image::quantizeColors ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003629{
3630 return constOptions()->quantizeColors( );
3631}
3632
3633void Magick::Image::quantizeColorSpace
3634 ( const Magick::ColorspaceType colorSpace_ )
3635{
3636 modifyImage();
3637 options()->quantizeColorSpace( colorSpace_ );
3638}
3639Magick::ColorspaceType Magick::Image::quantizeColorSpace ( void ) const
3640{
3641 return constOptions()->quantizeColorSpace( );
3642}
3643
3644void Magick::Image::quantizeDither ( const bool ditherFlag_ )
3645{
3646 modifyImage();
3647 options()->quantizeDither( ditherFlag_ );
3648}
3649bool Magick::Image::quantizeDither ( void ) const
3650{
3651 return constOptions()->quantizeDither( );
3652}
3653
cristyeaedf062010-05-29 22:36:02 +00003654void Magick::Image::quantizeTreeDepth ( const size_t treeDepth_ )
cristy3ed852e2009-09-05 21:47:34 +00003655{
3656 modifyImage();
3657 options()->quantizeTreeDepth( treeDepth_ );
3658}
cristyeaedf062010-05-29 22:36:02 +00003659size_t Magick::Image::quantizeTreeDepth ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003660{
3661 return constOptions()->quantizeTreeDepth( );
3662}
3663
3664void Magick::Image::renderingIntent
3665 ( const Magick::RenderingIntent renderingIntent_ )
3666{
3667 modifyImage();
3668 image()->rendering_intent = renderingIntent_;
3669}
3670Magick::RenderingIntent Magick::Image::renderingIntent ( void ) const
3671{
3672 return static_cast<Magick::RenderingIntent>(constImage()->rendering_intent);
3673}
3674
3675void Magick::Image::resolutionUnits
3676 ( const Magick::ResolutionType resolutionUnits_ )
3677{
3678 modifyImage();
3679 image()->units = resolutionUnits_;
3680 options()->resolutionUnits( resolutionUnits_ );
3681}
3682Magick::ResolutionType Magick::Image::resolutionUnits ( void ) const
3683{
3684 return constOptions()->resolutionUnits( );
3685}
3686
cristyeaedf062010-05-29 22:36:02 +00003687void Magick::Image::scene ( const size_t scene_ )
cristy3ed852e2009-09-05 21:47:34 +00003688{
3689 modifyImage();
3690 image()->scene = scene_;
3691}
cristyeaedf062010-05-29 22:36:02 +00003692size_t Magick::Image::scene ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003693{
3694 return constImage()->scene;
3695}
3696
3697std::string Magick::Image::signature ( const bool force_ ) const
3698{
3699 Lock( &_imgRef->_mutexLock );
3700
3701 // Re-calculate image signature if necessary
cristy018f07f2011-09-04 21:15:19 +00003702 ExceptionInfo exceptionInfo;
3703 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003704 if ( force_ ||
cristyd15e6592011-10-15 00:13:06 +00003705 !GetImageProperty(constImage(), "Signature", &exceptionInfo) ||
cristy3ed852e2009-09-05 21:47:34 +00003706 constImage()->taint )
3707 {
cristy018f07f2011-09-04 21:15:19 +00003708 SignatureImage( const_cast<MagickCore::Image *>(constImage()), &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003709 }
3710
cristyd15e6592011-10-15 00:13:06 +00003711 const char *property = GetImageProperty(constImage(), "Signature",
3712 &exceptionInfo);
cristy018f07f2011-09-04 21:15:19 +00003713 throwException( exceptionInfo );
3714 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003715
3716 return std::string( property );
3717}
3718
3719void Magick::Image::size ( const Geometry &geometry_ )
3720{
3721 modifyImage();
3722 options()->size( geometry_ );
3723 image()->rows = geometry_.height();
3724 image()->columns = geometry_.width();
3725}
3726Magick::Geometry Magick::Image::size ( void ) const
3727{
3728 return Magick::Geometry( constImage()->columns, constImage()->rows );
3729}
3730
cristy8198a752009-09-28 23:59:24 +00003731// Splice image
3732void Magick::Image::splice( const Geometry &geometry_ )
3733{
3734 RectangleInfo spliceInfo = geometry_;
3735 ExceptionInfo exceptionInfo;
3736 GetExceptionInfo( &exceptionInfo );
3737 MagickCore::Image* newImage =
3738 SpliceImage( image(), &spliceInfo, &exceptionInfo);
3739 replaceImage( newImage );
3740 throwException( exceptionInfo );
3741 (void) DestroyExceptionInfo( &exceptionInfo );
3742}
3743
cristy3ed852e2009-09-05 21:47:34 +00003744// Obtain image statistics. Statistics are normalized to the range of
3745// 0.0 to 1.0 and are output to the specified ImageStatistics
3746// structure.
cristyd42d9952011-07-08 14:21:50 +00003747void Magick::Image::statistics ( ImageStatistics *statistics )
cristy3ed852e2009-09-05 21:47:34 +00003748{
3749 double
3750 maximum,
3751 minimum;
3752
3753 ExceptionInfo exceptionInfo;
3754 GetExceptionInfo( &exceptionInfo );
cristyd42d9952011-07-08 14:21:50 +00003755
cristybd5a96c2011-08-21 00:04:26 +00003756 ChannelType channel_mask = SetPixelChannelMask( image(), RedChannel);
cristyd42d9952011-07-08 14:21:50 +00003757 (void) GetImageRange( image(),&minimum,&maximum,&exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00003758 statistics->red.minimum=minimum;
cristyd42d9952011-07-08 14:21:50 +00003759 statistics->red.maximum=maximum;
3760 (void) GetImageMean( image(),&statistics->red.mean,
3761 &statistics->red.standard_deviation,&exceptionInfo);
3762 (void) GetImageKurtosis( image(),&statistics->red.kurtosis,
3763 &statistics->red.skewness,&exceptionInfo);
cristye2a912b2011-12-05 20:02:07 +00003764 (void) SetPixelChannelMapMask( image(), channel_mask );
cristyd42d9952011-07-08 14:21:50 +00003765
cristybd5a96c2011-08-21 00:04:26 +00003766 channel_mask = SetPixelChannelMask( image(), GreenChannel);
cristyd42d9952011-07-08 14:21:50 +00003767 (void) GetImageRange( image(),&minimum,&maximum,&exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00003768 statistics->green.minimum=minimum;
cristyd42d9952011-07-08 14:21:50 +00003769 statistics->green.maximum=maximum;
3770 (void) GetImageMean( image(),&statistics->green.mean,
3771 &statistics->green.standard_deviation,&exceptionInfo);
3772 (void) GetImageKurtosis( image(),&statistics->green.kurtosis,
3773 &statistics->green.skewness,&exceptionInfo);
cristye2a912b2011-12-05 20:02:07 +00003774 (void) SetPixelChannelMapMask( image(), channel_mask );
cristyd42d9952011-07-08 14:21:50 +00003775
cristybd5a96c2011-08-21 00:04:26 +00003776 channel_mask = SetPixelChannelMask( image(), GreenChannel);
cristyd42d9952011-07-08 14:21:50 +00003777 (void) GetImageRange( image(),&minimum,&maximum,&exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00003778 statistics->blue.minimum=minimum;
cristyd42d9952011-07-08 14:21:50 +00003779 statistics->blue.maximum=maximum;
3780 (void) GetImageMean( image(),&statistics->blue.mean,
3781 &statistics->blue.standard_deviation,&exceptionInfo);
3782 (void) GetImageKurtosis( image(),&statistics->blue.kurtosis,
3783 &statistics->blue.skewness,&exceptionInfo);
cristye2a912b2011-12-05 20:02:07 +00003784 (void) SetPixelChannelMapMask( image(), channel_mask );
cristyd42d9952011-07-08 14:21:50 +00003785
cristybd5a96c2011-08-21 00:04:26 +00003786 channel_mask = SetPixelChannelMask( image(), AlphaChannel);
cristyd42d9952011-07-08 14:21:50 +00003787 (void) GetImageRange( image(),&minimum,&maximum,&exceptionInfo);
cristy4c08aed2011-07-01 19:47:50 +00003788 statistics->alpha.minimum=minimum;
3789 statistics->alpha.maximum=maximum;
cristyd42d9952011-07-08 14:21:50 +00003790 (void) GetImageMean( image(),&statistics->alpha.mean,
3791 &statistics->alpha.standard_deviation,&exceptionInfo);
3792 (void) GetImageKurtosis( image(),&statistics->alpha.kurtosis,
3793 &statistics->alpha.skewness,&exceptionInfo);
cristye2a912b2011-12-05 20:02:07 +00003794 (void) SetPixelChannelMapMask( image(), channel_mask );
cristyd42d9952011-07-08 14:21:50 +00003795
cristy3ed852e2009-09-05 21:47:34 +00003796 throwException( exceptionInfo );
3797 (void) DestroyExceptionInfo( &exceptionInfo );
3798}
3799
cristy9f89a3f2011-02-12 17:02:35 +00003800// Strip strips an image of all profiles and comments.
3801void Magick::Image::strip ( void )
3802{
3803 modifyImage();
cristye941a752011-10-15 01:52:48 +00003804 ExceptionInfo exceptionInfo;
3805 GetExceptionInfo( &exceptionInfo );
3806 StripImage( image(), &exceptionInfo );
3807 throwException( exceptionInfo );
3808 (void) DestroyExceptionInfo( &exceptionInfo );
cristy9f89a3f2011-02-12 17:02:35 +00003809}
3810
cristy3ed852e2009-09-05 21:47:34 +00003811// enabled/disable stroke anti-aliasing
3812void Magick::Image::strokeAntiAlias ( const bool flag_ )
3813{
3814 modifyImage();
3815 options()->strokeAntiAlias(flag_);
3816}
3817bool Magick::Image::strokeAntiAlias ( void ) const
3818{
3819 return constOptions()->strokeAntiAlias();
3820}
3821
3822// Color to use when drawing object outlines
3823void Magick::Image::strokeColor ( const Magick::Color &strokeColor_ )
3824{
3825 modifyImage();
3826 options()->strokeColor(strokeColor_);
3827}
3828Magick::Color Magick::Image::strokeColor ( void ) const
3829{
3830 return constOptions()->strokeColor();
3831}
3832
3833// dash pattern for drawing vector objects (default one)
3834void Magick::Image::strokeDashArray ( const double* strokeDashArray_ )
3835{
3836 modifyImage();
3837 options()->strokeDashArray( strokeDashArray_ );
3838}
3839
3840const double* Magick::Image::strokeDashArray ( void ) const
3841{
3842 return constOptions()->strokeDashArray( );
3843}
3844
3845// dash offset for drawing vector objects (default one)
3846void Magick::Image::strokeDashOffset ( const double strokeDashOffset_ )
3847{
3848 modifyImage();
3849 options()->strokeDashOffset( strokeDashOffset_ );
3850}
3851
3852double Magick::Image::strokeDashOffset ( void ) const
3853{
3854 return constOptions()->strokeDashOffset( );
3855}
3856
3857// Specify the shape to be used at the end of open subpaths when they
3858// are stroked. Values of LineCap are UndefinedCap, ButtCap, RoundCap,
3859// and SquareCap.
3860void Magick::Image::strokeLineCap ( const Magick::LineCap lineCap_ )
3861{
3862 modifyImage();
3863 options()->strokeLineCap( lineCap_ );
3864}
3865Magick::LineCap Magick::Image::strokeLineCap ( void ) const
3866{
3867 return constOptions()->strokeLineCap( );
3868}
3869
3870// Specify the shape to be used at the corners of paths (or other
3871// vector shapes) when they are stroked. Values of LineJoin are
3872// UndefinedJoin, MiterJoin, RoundJoin, and BevelJoin.
3873void Magick::Image::strokeLineJoin ( const Magick::LineJoin lineJoin_ )
3874{
3875 modifyImage();
3876 options()->strokeLineJoin( lineJoin_ );
3877}
3878Magick::LineJoin Magick::Image::strokeLineJoin ( void ) const
3879{
3880 return constOptions()->strokeLineJoin( );
3881}
3882
3883// Specify miter limit. When two line segments meet at a sharp angle
3884// and miter joins have been specified for 'lineJoin', it is possible
3885// for the miter to extend far beyond the thickness of the line
3886// stroking the path. The miterLimit' imposes a limit on the ratio of
3887// the miter length to the 'lineWidth'. The default value of this
3888// parameter is 4.
cristyeaedf062010-05-29 22:36:02 +00003889void Magick::Image::strokeMiterLimit ( const size_t strokeMiterLimit_ )
cristy3ed852e2009-09-05 21:47:34 +00003890{
3891 modifyImage();
3892 options()->strokeMiterLimit( strokeMiterLimit_ );
3893}
cristyeaedf062010-05-29 22:36:02 +00003894size_t Magick::Image::strokeMiterLimit ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003895{
3896 return constOptions()->strokeMiterLimit( );
3897}
3898
3899// Pattern to use while stroking drawn objects.
3900void Magick::Image::strokePattern ( const Image &strokePattern_ )
3901{
3902 modifyImage();
3903 if(strokePattern_.isValid())
3904 options()->strokePattern( strokePattern_.constImage() );
3905 else
3906 options()->strokePattern( static_cast<MagickCore::Image*>(NULL) );
3907}
3908Magick::Image Magick::Image::strokePattern ( void ) const
3909{
3910 // FIXME: This is inordinately innefficient
3911 Image texture;
3912
3913 const MagickCore::Image* tmpTexture = constOptions()->strokePattern( );
3914
3915 if ( tmpTexture )
3916 {
3917 ExceptionInfo exceptionInfo;
3918 GetExceptionInfo( &exceptionInfo );
3919 MagickCore::Image* image =
3920 CloneImage( tmpTexture,
3921 0, // columns
3922 0, // rows
3923 MagickTrue, // orphan
3924 &exceptionInfo);
3925 throwException( exceptionInfo );
cristye941a752011-10-15 01:52:48 +00003926 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00003927 texture.replaceImage( image );
3928 }
3929 return texture;
3930}
3931
3932// Stroke width for drawing lines, circles, ellipses, etc.
3933void Magick::Image::strokeWidth ( const double strokeWidth_ )
3934{
3935 modifyImage();
3936 options()->strokeWidth( strokeWidth_ );
3937}
3938double Magick::Image::strokeWidth ( void ) const
3939{
3940 return constOptions()->strokeWidth( );
3941}
3942
cristyeaedf062010-05-29 22:36:02 +00003943void Magick::Image::subImage ( const size_t subImage_ )
cristy3ed852e2009-09-05 21:47:34 +00003944{
3945 modifyImage();
3946 options()->subImage( subImage_ );
3947}
cristyeaedf062010-05-29 22:36:02 +00003948size_t Magick::Image::subImage ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003949{
3950 return constOptions()->subImage( );
3951}
3952
cristyeaedf062010-05-29 22:36:02 +00003953void Magick::Image::subRange ( const size_t subRange_ )
cristy3ed852e2009-09-05 21:47:34 +00003954{
3955 modifyImage();
3956 options()->subRange( subRange_ );
3957}
cristyeaedf062010-05-29 22:36:02 +00003958size_t Magick::Image::subRange ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00003959{
3960 return constOptions()->subRange( );
3961}
3962
3963// Annotation text encoding (e.g. "UTF-16")
3964void Magick::Image::textEncoding ( const std::string &encoding_ )
3965{
3966 modifyImage();
3967 options()->textEncoding( encoding_ );
3968}
3969std::string Magick::Image::textEncoding ( void ) const
3970{
3971 return constOptions()->textEncoding( );
3972}
3973
cristybb503372010-05-27 20:51:26 +00003974size_t Magick::Image::totalColors ( void )
cristy3ed852e2009-09-05 21:47:34 +00003975{
3976 ExceptionInfo exceptionInfo;
3977 GetExceptionInfo( &exceptionInfo );
cristybb503372010-05-27 20:51:26 +00003978 size_t colors = GetNumberColors( image(), 0, &exceptionInfo);
cristy3ed852e2009-09-05 21:47:34 +00003979 throwException( exceptionInfo );
3980 (void) DestroyExceptionInfo( &exceptionInfo );
3981 return colors;
3982}
3983
3984// Origin of coordinate system to use when annotating with text or drawing
3985void Magick::Image::transformOrigin ( const double x_, const double y_ )
3986{
3987 modifyImage();
3988 options()->transformOrigin( x_, y_ );
3989}
3990
3991// Rotation to use when annotating with text or drawing
3992void Magick::Image::transformRotation ( const double angle_ )
3993{
3994 modifyImage();
3995 options()->transformRotation( angle_ );
3996}
3997
3998// Reset transformation parameters to default
3999void Magick::Image::transformReset ( void )
4000{
4001 modifyImage();
4002 options()->transformReset();
4003}
4004
4005// Scale to use when annotating with text or drawing
4006void Magick::Image::transformScale ( const double sx_, const double sy_ )
4007{
4008 modifyImage();
4009 options()->transformScale( sx_, sy_ );
4010}
4011
4012// Skew to use in X axis when annotating with text or drawing
4013void Magick::Image::transformSkewX ( const double skewx_ )
4014{
4015 modifyImage();
4016 options()->transformSkewX( skewx_ );
4017}
4018
4019// Skew to use in Y axis when annotating with text or drawing
4020void Magick::Image::transformSkewY ( const double skewy_ )
4021{
4022 modifyImage();
4023 options()->transformSkewY( skewy_ );
4024}
4025
4026// Image representation type
4027Magick::ImageType Magick::Image::type ( void ) const
4028{
4029
4030 ExceptionInfo exceptionInfo;
4031 GetExceptionInfo( &exceptionInfo );
4032 ImageType image_type = constOptions()->type();
4033 if ( image_type == UndefinedType )
4034 image_type= GetImageType( constImage(), &exceptionInfo);
4035 throwException( exceptionInfo );
4036 (void) DestroyExceptionInfo( &exceptionInfo );
4037 return image_type;
4038}
4039void Magick::Image::type ( const Magick::ImageType type_)
4040{
cristy018f07f2011-09-04 21:15:19 +00004041 ExceptionInfo exceptionInfo;
4042 GetExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00004043 modifyImage();
4044 options()->type( type_ );
cristy018f07f2011-09-04 21:15:19 +00004045 SetImageType( image(), type_, &exceptionInfo );
4046 throwException( exceptionInfo );
4047 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00004048}
4049
4050void Magick::Image::verbose ( const bool verboseFlag_ )
4051{
4052 modifyImage();
4053 options()->verbose( verboseFlag_ );
4054}
4055bool Magick::Image::verbose ( void ) const
4056{
4057 return constOptions()->verbose( );
4058}
4059
4060void Magick::Image::view ( const std::string &view_ )
4061{
4062 modifyImage();
4063 options()->view( view_ );
4064}
4065std::string Magick::Image::view ( void ) const
4066{
4067 return constOptions()->view( );
4068}
4069
4070// Virtual pixel method
4071void Magick::Image::virtualPixelMethod ( const VirtualPixelMethod virtual_pixel_method_ )
4072{
4073 modifyImage();
cristy387430f2012-02-07 13:09:46 +00004074 ExceptionInfo exceptionInfo;
4075 GetExceptionInfo( &exceptionInfo );
4076 SetImageVirtualPixelMethod( image(), virtual_pixel_method_, &exceptionInfo );
4077 throwException( exceptionInfo );
4078 (void) DestroyExceptionInfo( &exceptionInfo );
cristy3ed852e2009-09-05 21:47:34 +00004079}
4080Magick::VirtualPixelMethod Magick::Image::virtualPixelMethod ( void ) const
4081{
4082 return GetImageVirtualPixelMethod( constImage() );
4083}
4084
4085void Magick::Image::x11Display ( const std::string &display_ )
4086{
4087 modifyImage();
4088 options()->x11Display( display_ );
4089}
4090std::string Magick::Image::x11Display ( void ) const
4091{
4092 return constOptions()->x11Display( );
4093}
4094
4095double Magick::Image::xResolution ( void ) const
4096{
cristy2a11bef2011-10-28 18:33:11 +00004097 return constImage()->resolution.x;
cristy3ed852e2009-09-05 21:47:34 +00004098}
4099double Magick::Image::yResolution ( void ) const
4100{
cristy2a11bef2011-10-28 18:33:11 +00004101 return constImage()->resolution.y;
cristy3ed852e2009-09-05 21:47:34 +00004102}
4103
4104// Copy Constructor
4105Magick::Image::Image( const Image & image_ )
4106 : _imgRef(image_._imgRef)
4107{
4108 Lock( &_imgRef->_mutexLock );
4109
4110 // Increase reference count
4111 ++_imgRef->_refCount;
4112}
4113
4114// Assignment operator
4115Magick::Image& Magick::Image::operator=( const Magick::Image &image_ )
4116{
4117 if( this != &image_ )
4118 {
4119 {
4120 Lock( &image_._imgRef->_mutexLock );
4121 ++image_._imgRef->_refCount;
4122 }
4123
4124 bool doDelete = false;
4125 {
4126 Lock( &_imgRef->_mutexLock );
4127 if ( --_imgRef->_refCount == 0 )
4128 doDelete = true;
4129 }
4130
4131 if ( doDelete )
4132 {
4133 // Delete old image reference with associated image and options.
4134 delete _imgRef;
4135 _imgRef = 0;
4136 }
4137 // Use new image reference
4138 _imgRef = image_._imgRef;
4139 }
4140
4141 return *this;
4142}
4143
4144//////////////////////////////////////////////////////////////////////
4145//
4146// Low-level Pixel Access Routines
4147//
4148// Also see the Pixels class, which provides support for multiple
4149// cache views. The low-level pixel access routines in the Image
4150// class are provided in order to support backward compatability.
4151//
4152//////////////////////////////////////////////////////////////////////
4153
4154// Transfers read-only pixels from the image to the pixel cache as
4155// defined by the specified region
cristy4c08aed2011-07-01 19:47:50 +00004156const Magick::Quantum* Magick::Image::getConstPixels
cristyd99b0962010-05-29 23:14:26 +00004157 ( const ssize_t x_, const ssize_t y_,
cristyeaedf062010-05-29 22:36:02 +00004158 const size_t columns_,
4159 const size_t rows_ ) const
cristy3ed852e2009-09-05 21:47:34 +00004160{
4161 ExceptionInfo exceptionInfo;
4162 GetExceptionInfo( &exceptionInfo );
cristy4c08aed2011-07-01 19:47:50 +00004163 const Quantum* p = (*GetVirtualPixels)( constImage(),
cristy3ed852e2009-09-05 21:47:34 +00004164 x_, y_,
4165 columns_, rows_,
4166 &exceptionInfo );
4167 throwException( exceptionInfo );
4168 (void) DestroyExceptionInfo( &exceptionInfo );
4169 return p;
4170}
4171
cristy4c08aed2011-07-01 19:47:50 +00004172// Obtain read-only pixel associated pixels channels
4173const void* Magick::Image::getConstMetacontent ( void ) const
cristy3ed852e2009-09-05 21:47:34 +00004174{
cristy4c08aed2011-07-01 19:47:50 +00004175 const void* result = GetVirtualMetacontent( constImage() );
cristy3ed852e2009-09-05 21:47:34 +00004176
4177 if( !result )
4178 throwImageException();
4179
4180 return result;
4181}
4182
cristy4c08aed2011-07-01 19:47:50 +00004183// Obtain image pixel associated pixels channels
4184void* Magick::Image::getMetacontent ( void )
cristy3ed852e2009-09-05 21:47:34 +00004185{
cristy4c08aed2011-07-01 19:47:50 +00004186 void* result = GetAuthenticMetacontent( image() );
cristy3ed852e2009-09-05 21:47:34 +00004187
4188 if( !result )
4189 throwImageException();
4190
4191 return ( result );
4192}
4193
4194// Transfers pixels from the image to the pixel cache as defined
4195// by the specified region. Modified pixels may be subsequently
4196// transferred back to the image via syncPixels.
cristy4c08aed2011-07-01 19:47:50 +00004197Magick::Quantum* Magick::Image::getPixels ( const ssize_t x_, const ssize_t y_,
cristyeaedf062010-05-29 22:36:02 +00004198 const size_t columns_,
4199 const size_t rows_ )
cristy3ed852e2009-09-05 21:47:34 +00004200{
4201 modifyImage();
4202 ExceptionInfo exceptionInfo;
4203 GetExceptionInfo( &exceptionInfo );
cristy4c08aed2011-07-01 19:47:50 +00004204 Quantum* result = (*GetAuthenticPixels)( image(),
cristy3ed852e2009-09-05 21:47:34 +00004205 x_, y_,
4206 columns_, rows_, &exceptionInfo );
4207 throwException( exceptionInfo );
4208 (void) DestroyExceptionInfo( &exceptionInfo );
4209
4210 return result;
4211}
4212
4213// Allocates a pixel cache region to store image pixels as defined
4214// by the region rectangle. This area is subsequently transferred
4215// from the pixel cache to the image via syncPixels.
cristy4c08aed2011-07-01 19:47:50 +00004216Magick::Quantum* Magick::Image::setPixels ( const ssize_t x_, const ssize_t y_,
cristyeaedf062010-05-29 22:36:02 +00004217 const size_t columns_,
4218 const size_t rows_ )
cristy3ed852e2009-09-05 21:47:34 +00004219{
4220 modifyImage();
4221 ExceptionInfo exceptionInfo;
4222 GetExceptionInfo( &exceptionInfo );
cristy4c08aed2011-07-01 19:47:50 +00004223 Quantum* result = (*QueueAuthenticPixels)( image(),
cristy3ed852e2009-09-05 21:47:34 +00004224 x_, y_,
4225 columns_, rows_, &exceptionInfo );
4226 throwException( exceptionInfo );
4227 (void) DestroyExceptionInfo( &exceptionInfo );
4228
4229 return result;
4230}
4231
4232// Transfers the image cache pixels to the image.
4233void Magick::Image::syncPixels ( void )
4234{
4235 ExceptionInfo exceptionInfo;
4236 GetExceptionInfo( &exceptionInfo );
4237 (*SyncAuthenticPixels)( image(), &exceptionInfo );
4238 throwException( exceptionInfo );
4239 (void) DestroyExceptionInfo( &exceptionInfo );
4240}
4241
4242// Transfers one or more pixel components from a buffer or file
4243// into the image pixel cache of an image.
4244// Used to support image decoders.
4245void Magick::Image::readPixels ( const Magick::QuantumType quantum_,
4246 const unsigned char *source_ )
4247{
4248 QuantumInfo
4249 *quantum_info;
4250
4251 quantum_info=AcquireQuantumInfo(imageInfo(),image());
4252 ExceptionInfo exceptionInfo;
4253 GetExceptionInfo( &exceptionInfo );
4254 ImportQuantumPixels(image(),(MagickCore::CacheView *) NULL,quantum_info,
4255 quantum_,source_, &exceptionInfo);
4256 throwException( exceptionInfo );
4257 (void) DestroyExceptionInfo( &exceptionInfo );
4258 quantum_info=DestroyQuantumInfo(quantum_info);
4259}
4260
4261// Transfers one or more pixel components from the image pixel
4262// cache to a buffer or file.
4263// Used to support image encoders.
4264void Magick::Image::writePixels ( const Magick::QuantumType quantum_,
4265 unsigned char *destination_ )
4266{
4267 QuantumInfo
4268 *quantum_info;
4269
4270 quantum_info=AcquireQuantumInfo(imageInfo(),image());
4271 ExceptionInfo exceptionInfo;
4272 GetExceptionInfo( &exceptionInfo );
4273 ExportQuantumPixels(image(),(MagickCore::CacheView *) NULL,quantum_info,
4274 quantum_,destination_, &exceptionInfo);
4275 quantum_info=DestroyQuantumInfo(quantum_info);
4276 throwException( exceptionInfo );
4277 (void) DestroyExceptionInfo( &exceptionInfo );
4278}
4279
4280/////////////////////////////////////////////////////////////////////
4281//
4282// No end-user methods beyond this point
4283//
4284/////////////////////////////////////////////////////////////////////
4285
4286
4287//
4288// Construct using existing image and default options
4289//
4290Magick::Image::Image ( MagickCore::Image* image_ )
4291 : _imgRef(new ImageRef( image_))
4292{
4293}
4294
4295// Get Magick::Options*
4296Magick::Options* Magick::Image::options( void )
4297{
4298 return _imgRef->options();
4299}
4300const Magick::Options* Magick::Image::constOptions( void ) const
4301{
4302 return _imgRef->options();
4303}
4304
4305// Get MagickCore::Image*
4306MagickCore::Image*& Magick::Image::image( void )
4307{
4308 return _imgRef->image();
4309}
4310const MagickCore::Image* Magick::Image::constImage( void ) const
4311{
4312 return _imgRef->image();
4313}
4314
4315// Get ImageInfo *
4316MagickCore::ImageInfo* Magick::Image::imageInfo( void )
4317{
4318 return _imgRef->options()->imageInfo();
4319}
4320const MagickCore::ImageInfo * Magick::Image::constImageInfo( void ) const
4321{
4322 return _imgRef->options()->imageInfo();
4323}
4324
4325// Get QuantizeInfo *
4326MagickCore::QuantizeInfo* Magick::Image::quantizeInfo( void )
4327{
4328 return _imgRef->options()->quantizeInfo();
4329}
4330const MagickCore::QuantizeInfo * Magick::Image::constQuantizeInfo( void ) const
4331{
4332 return _imgRef->options()->quantizeInfo();
4333}
4334
4335//
4336// Replace current image
4337//
4338MagickCore::Image * Magick::Image::replaceImage
4339 ( MagickCore::Image* replacement_ )
4340{
4341 MagickCore::Image* image;
4342
4343 if( replacement_ )
4344 image = replacement_;
4345 else
cristy9950d572011-10-01 18:22:35 +00004346 {
4347 ExceptionInfo exceptionInfo;
4348 GetExceptionInfo( &exceptionInfo );
4349 image = AcquireImage(constImageInfo(), &exceptionInfo);
4350 throwException( exceptionInfo );
4351 (void) DestroyExceptionInfo( &exceptionInfo );
4352 }
cristy3ed852e2009-09-05 21:47:34 +00004353
4354 {
4355 Lock( &_imgRef->_mutexLock );
4356
4357 if ( _imgRef->_refCount == 1 )
4358 {
4359 // We own the image, just replace it, and de-register
4360 _imgRef->id( -1 );
4361 _imgRef->image(image);
4362 }
4363 else
4364 {
4365 // We don't own the image, dereference and replace with copy
4366 --_imgRef->_refCount;
4367 _imgRef = new ImageRef( image, constOptions() );
4368 }
4369 }
4370
4371 return _imgRef->_image;
4372}
4373
4374//
4375// Prepare to modify image or image options
4376// Replace current image and options with copy if reference count > 1
4377//
4378void Magick::Image::modifyImage( void )
4379{
4380 {
4381 Lock( &_imgRef->_mutexLock );
4382 if ( _imgRef->_refCount == 1 )
4383 {
4384 // De-register image and return
4385 _imgRef->id( -1 );
4386 return;
4387 }
4388 }
4389
4390 ExceptionInfo exceptionInfo;
4391 GetExceptionInfo( &exceptionInfo );
4392 replaceImage( CloneImage( image(),
4393 0, // columns
4394 0, // rows
4395 MagickTrue, // orphan
4396 &exceptionInfo) );
4397 throwException( exceptionInfo );
4398 (void) DestroyExceptionInfo( &exceptionInfo );
4399 return;
4400}
4401
4402//
4403// Test for an ImageMagick reported error and throw exception if one
4404// has been reported. Secretly resets image->exception back to default
4405// state even though this method is const.
4406//
4407void Magick::Image::throwImageException( void ) const
4408{
4409 // Throw C++ exception while resetting Image exception to default state
cristy3ed852e2009-09-05 21:47:34 +00004410}
4411
4412// Register image with image registry or obtain registration id
cristybb503372010-05-27 20:51:26 +00004413ssize_t Magick::Image::registerId( void )
cristy3ed852e2009-09-05 21:47:34 +00004414{
4415 Lock( &_imgRef->_mutexLock );
4416 if( _imgRef->id() < 0 )
4417 {
4418 char id[MaxTextExtent];
4419 ExceptionInfo exceptionInfo;
4420 GetExceptionInfo( &exceptionInfo );
4421 _imgRef->id(_imgRef->id()+1);
cristye8c25f92010-06-03 00:53:06 +00004422 sprintf(id,"%.20g\n",(double) _imgRef->id());
cristy3ed852e2009-09-05 21:47:34 +00004423 SetImageRegistry(ImageRegistryType, id, image(), &exceptionInfo);
4424 throwException( exceptionInfo );
4425 (void) DestroyExceptionInfo( &exceptionInfo );
4426 }
4427 return _imgRef->id();
4428}
4429
4430// Unregister image from image registry
4431void Magick::Image::unregisterId( void )
4432{
4433 modifyImage();
4434 _imgRef->id( -1 );
4435}
4436
4437//
4438// Create a local wrapper around MagickCoreTerminus
4439//
4440namespace Magick
4441{
4442 extern "C" {
4443 void MagickPlusPlusDestroyMagick(void);
4444 }
4445}
4446
4447void Magick::MagickPlusPlusDestroyMagick(void)
4448{
4449 if (magick_initialized)
4450 {
4451 magick_initialized=false;
4452 MagickCore::MagickCoreTerminus();
4453 }
4454}
4455
4456// C library initialization routine
cristyaf1dd252011-09-07 19:04:02 +00004457void MagickPPExport Magick::InitializeMagick(const char *path_)
cristy3ed852e2009-09-05 21:47:34 +00004458{
4459 MagickCore::MagickCoreGenesis(path_,MagickFalse);
4460 if (!magick_initialized)
4461 magick_initialized=true;
4462}
4463
4464//
4465// Cleanup class to ensure that ImageMagick singletons are destroyed
4466// so as to avoid any resemblence to a memory leak (which seems to
4467// confuse users)
4468//
4469namespace Magick
4470{
4471
4472 class MagickCleanUp
4473 {
4474 public:
4475 MagickCleanUp( void );
4476 ~MagickCleanUp( void );
4477 };
4478
4479 // The destructor for this object is invoked when the destructors for
4480 // static objects in this translation unit are invoked.
4481 static MagickCleanUp magickCleanUpGuard;
4482}
4483
4484Magick::MagickCleanUp::MagickCleanUp ( void )
4485{
4486 // Don't even think about invoking InitializeMagick here!
4487}
4488
4489Magick::MagickCleanUp::~MagickCleanUp ( void )
4490{
4491 MagickPlusPlusDestroyMagick();
4492}