1 // This may look like C code, but it is really -*- C++ -*-
2 //
3 // Copyright Bob Friesenhahn, 1999, 2000, 2001, 2002, 2003
4 // Copyright Dirk Lemstra 2013-2017
5 //
6 // Definition and implementation of template functions for using
7 // Magick::Image with STL containers.
8 //
9 
10 #ifndef Magick_STL_header
11 #define Magick_STL_header
12 
13 #include "Magick++/Include.h"
14 #include <algorithm>
15 #include <functional>
16 #include <iterator>
17 #include <map>
18 #include <utility>
19 
20 #include "Magick++/CoderInfo.h"
21 #include "Magick++/Drawable.h"
22 #include "Magick++/Exception.h"
23 #include "Magick++/Montage.h"
24 
25 namespace Magick
26 {
27   //
28   // STL function object declarations/definitions
29   //
30 
31   // Function objects provide the means to invoke an operation on one
32   // or more image objects in an STL-compatable container.  The
33   // arguments to the function object constructor(s) are compatable
34   // with the arguments to the equivalent Image class method and
35   // provide the means to supply these options when the function
36   // object is invoked.
37 
38   // For example, to read a GIF animation, set the color red to
39   // transparent for all frames, and write back out:
40   //
41   // list<image> images;
42   // readImages( &images, "animation.gif" );
43   // for_each( images.begin(), images.end(), transparentImage( "red" ) );
44   // writeImages( images.begin(), images.end(), "animation.gif" );
45 
46   // Adaptive-blur image with specified blur factor
47   class MagickPPExport adaptiveBlurImage
48   {
49   public:
50     adaptiveBlurImage( const double radius_ = 1, const double sigma_ = 0.5 );
51 
52     void operator()( Image &image_ ) const;
53 
54   private:
55     double _radius;
56     double _sigma;
57   };
58 
59   // Local adaptive threshold image
60   // http://www.dai.ed.ac.uk/HIPR2/adpthrsh.htm
61   // Width x height define the size of the pixel neighborhood
62   // offset = constant to subtract from pixel neighborhood mean
63   class MagickPPExport adaptiveThresholdImage
64   {
65   public:
66     adaptiveThresholdImage( const size_t width_,
67                             const size_t height_,
68                             const ::ssize_t offset_ = 0  );
69 
70     void operator()( Image &image_ ) const;
71 
72   private:
73     size_t _width;
74     size_t _height;
75     ::ssize_t _offset;
76   };
77 
78   // Add noise to image with specified noise type
79   class MagickPPExport addNoiseImage
80   {
81   public:
82     addNoiseImage(const NoiseType noiseType_,const double attenuate_ = 1.0);
83 
84     void operator()(Image &image_) const;
85 
86   private:
87     NoiseType _noiseType;
88     double _attenuate;
89   };
90 
91   // Transform image by specified affine (or free transform) matrix.
92   class MagickPPExport affineTransformImage
93   {
94   public:
95     affineTransformImage( const DrawableAffine &affine_ );
96 
97     void operator()( Image &image_ ) const;
98 
99   private:
100     DrawableAffine _affine;
101   };
102 
103   // Annotate image (draw text on image)
104   class MagickPPExport annotateImage
105   {
106   public:
107     // Annotate using specified text, and placement location
108     annotateImage ( const std::string &text_,
109                     const Geometry &geometry_ );
110 
111     // Annotate using specified text, bounding area, and placement
112     // gravity
113     annotateImage ( const std::string &text_,
114         const Geometry &geometry_,
115         const GravityType gravity_ );
116 
117     // Annotate with text using specified text, bounding area,
118     // placement gravity, and rotation.
119     annotateImage ( const std::string &text_,
120                     const Geometry &geometry_,
121                     const GravityType gravity_,
122                     const double degrees_ );
123 
124     // Annotate with text (bounding area is entire image) and
125     // placement gravity.
126     annotateImage ( const std::string &text_,
127         const GravityType gravity_ );
128 
129     void operator()( Image &image_ ) const;
130 
131   private:
132     const std::string   _text;
133     const Geometry      _geometry;
134     const GravityType   _gravity;
135     const double        _degrees;
136   };
137 
138   // Blur image with specified blur factor
139   class MagickPPExport blurImage
140   {
141   public:
142     blurImage( const double radius_ = 1, const double sigma_ = 0.5 );
143 
144     void operator()( Image &image_ ) const;
145 
146   private:
147     double _radius;
148     double _sigma;
149   };
150 
151   // Border image (add border to image)
152   class MagickPPExport borderImage
153   {
154   public:
155     borderImage( const Geometry &geometry_ = borderGeometryDefault  );
156 
157     void operator()( Image &image_ ) const;
158 
159   private:
160     Geometry _geometry;
161   };
162 
163   // Extract channel from image
164   class MagickPPExport channelImage
165   {
166   public:
167     channelImage( const ChannelType channel_ );
168 
169     void operator()( Image &image_ ) const;
170 
171   private:
172     ChannelType _channel;
173   };
174 
175   // Charcoal effect image (looks like charcoal sketch)
176   class MagickPPExport charcoalImage
177   {
178   public:
179     charcoalImage( const double radius_ = 1, const double sigma_ = 0.5  );
180 
181     void operator()( Image &image_ ) const;
182 
183   private:
184     double _radius;
185     double _sigma;
186   };
187 
188   // Chop image (remove vertical or horizontal subregion of image)
189   class MagickPPExport chopImage
190   {
191   public:
192     chopImage( const Geometry &geometry_ );
193 
194     void operator()( Image &image_ ) const;
195 
196   private:
197     Geometry _geometry;
198   };
199 
200   // Accepts a lightweight Color Correction Collection (CCC) file which solely
201   // contains one or more color corrections and applies the correction to the
202   // image.
203   class MagickPPExport cdlImage
204   {
205   public:
206     cdlImage( const std::string &cdl_ );
207 
208     void operator()( Image &image_ ) const;
209 
210   private:
211     std::string   _cdl;
212   };
213 
214   // Colorize image using pen color at specified percent alpha
215   class MagickPPExport colorizeImage
216   {
217   public:
218     colorizeImage( const unsigned int alphaRed_,
219                    const unsigned int alphaGreen_,
220                    const unsigned int alphaBlue_,
221        const Color &penColor_ );
222 
223     colorizeImage( const unsigned int alpha_,
224                    const Color &penColor_ );
225 
226     void operator()( Image &image_ ) const;
227 
228   private:
229     unsigned int _alphaRed;
230     unsigned int _alphaGreen;
231     unsigned int _alphaBlue;
232     Color _penColor;
233   };
234 
235   // Apply a color matrix to the image channels.  The user supplied
236   // matrix may be of order 1 to 5 (1x1 through 5x5).
237   class MagickPPExport colorMatrixImage
238   {
239   public:
240     colorMatrixImage( const size_t order_,
241           const double *color_matrix_ );
242 
243     void operator()( Image &image_ ) const;
244 
245   private:
246     size_t  _order;
247     const double *_color_matrix;
248   };
249 
250   // Convert the image colorspace representation
251   class MagickPPExport colorSpaceImage
252   {
253   public:
254     colorSpaceImage( ColorspaceType colorSpace_ );
255 
256     void operator()( Image &image_ ) const;
257 
258   private:
259     ColorspaceType _colorSpace;
260   };
261 
262   // Comment image (add comment string to image)
263   class MagickPPExport commentImage
264   {
265   public:
266     commentImage( const std::string &comment_ );
267 
268     void operator()( Image &image_ ) const;
269 
270   private:
271     std::string _comment;
272   };
273 
274   // Compose an image onto another at specified offset and using
275   // specified algorithm
276   class MagickPPExport compositeImage
277   {
278   public:
279     compositeImage( const Image &compositeImage_,
280         ::ssize_t xOffset_,
281         ::ssize_t yOffset_,
282         CompositeOperator compose_ = InCompositeOp );
283 
284     compositeImage( const Image &compositeImage_,
285         const Geometry &offset_,
286         CompositeOperator compose_ = InCompositeOp );
287 
288     void operator()( Image &image_ ) const;
289 
290   private:
291     Image             _compositeImage;
292     ::ssize_t         _xOffset;
293     ::ssize_t         _yOffset;
294     CompositeOperator _compose;
295   };
296 
297   // Contrast image (enhance intensity differences in image)
298   class MagickPPExport contrastImage
299   {
300   public:
301     contrastImage( const size_t sharpen_ );
302 
303     void operator()( Image &image_ ) const;
304 
305   private:
306     size_t _sharpen;
307   };
308 
309   // Crop image (subregion of original image)
310   class MagickPPExport cropImage
311   {
312   public:
313     cropImage( const Geometry &geometry_ );
314 
315     void operator()( Image &image_ ) const;
316 
317   private:
318     Geometry _geometry;
319   };
320 
321   // Cycle image colormap
322   class MagickPPExport cycleColormapImage
323   {
324   public:
325     cycleColormapImage( const ::ssize_t amount_ );
326 
327     void operator()( Image &image_ ) const;
328 
329   private:
330     ::ssize_t _amount;
331   };
332 
333   // Despeckle image (reduce speckle noise)
334   class MagickPPExport despeckleImage
335   {
336   public:
337     despeckleImage( void );
338 
339     void operator()( Image &image_ ) const;
340 
341   private:
342   };
343 
344   // Distort image.  distorts an image using various distortion methods, by
345   // mapping color lookups of the source image to a new destination image
346   // usally of the same size as the source image, unless 'bestfit' is set to
347   // true.
348   class MagickPPExport distortImage
349   {
350   public:
351     distortImage( const Magick::DistortMethod method_,
352       const size_t number_arguments_,
353       const double *arguments_,
354       const bool bestfit_ );
355 
356     distortImage( const Magick::DistortMethod method_,
357       const size_t number_arguments_,
358       const double *arguments_ );
359 
360     void operator()( Image &image_ ) const;
361 
362   private:
363     DistortMethod _method;
364     size_t _number_arguments;
365     const double *_arguments;
366     bool _bestfit;
367   };
368 
369   // Draw on image
370   class MagickPPExport drawImage
371   {
372   public:
373     // Draw on image using a single drawable
374     // Store in list to make implementation easier
375     drawImage( const Drawable &drawable_ );
376 
377     // Draw on image using a drawable list
378     drawImage( const DrawableList &drawable_ );
379 
380     void operator()( Image &image_ ) const;
381 
382   private:
383     DrawableList _drawableList;
384   };
385 
386   // Edge image (hilight edges in image)
387   class MagickPPExport edgeImage
388   {
389   public:
390     edgeImage( const double radius_ = 0.0 );
391 
392     void operator()( Image &image_ ) const;
393 
394   private:
395     double _radius;
396   };
397 
398   // Emboss image (hilight edges with 3D effect)
399   class MagickPPExport embossImage
400   {
401   public:
402     embossImage( void );
403     embossImage( const double radius_, const double sigma_ );
404 
405     void operator()( Image &image_ ) const;
406 
407   private:
408     double _radius;
409     double _sigma;
410   };
411 
412   // Enhance image (minimize noise)
413   class MagickPPExport enhanceImage
414   {
415   public:
416     enhanceImage( void );
417 
418     void operator()( Image &image_ ) const;
419 
420   private:
421   };
422 
423   // Equalize image (histogram equalization)
424   class MagickPPExport equalizeImage
425   {
426   public:
427     equalizeImage( void );
428 
429     void operator()( Image &image_ ) const;
430 
431   private:
432   };
433 
434   // Color to use when filling drawn objects
435   class MagickPPExport fillColorImage
436   {
437   public:
438     fillColorImage( const Color &fillColor_ );
439 
440     void operator()( Image &image_ ) const;
441 
442   private:
443     Color _fillColor;
444   };
445 
446   // Flip image (reflect each scanline in the vertical direction)
447   class MagickPPExport flipImage
448   {
449   public:
450     flipImage( void );
451 
452     void operator()( Image &image_ ) const;
453 
454   private:
455   };
456 
457   // Floodfill designated area with a matte value
458   class MagickPPExport floodFillAlphaImage
459 
460   {
461   public:
462     floodFillAlphaImage(const ::ssize_t x_,const ::ssize_t y_,
463      const unsigned int alpha_,const Color &target_,const bool invert_=false);
464 
465     void operator()(Image &image_) const;
466 
467   private:
468     Color        _target;
469     unsigned int _alpha;
470     ::ssize_t    _x;
471     ::ssize_t    _y;
472     bool         _invert;
473   };
474 
475   // Flood-fill image with color
476   class MagickPPExport floodFillColorImage
477 
478   {
479   public:
480     // Flood-fill color across pixels starting at target-pixel and
481     // stopping at pixels matching specified border color.
482     // Uses current fuzz setting when determining color match.
483     floodFillColorImage(const Geometry &point_,const Color &fillColor_,
484       const bool invert_=false);
485     floodFillColorImage(const ::ssize_t x_,const ::ssize_t y_,
486       const Color &fillColor_,const bool invert_=false);
487 
488     // Flood-fill color across pixels starting at target-pixel and
489     // stopping at pixels matching specified border color.
490     // Uses current fuzz setting when determining color match.
491     floodFillColorImage(const Geometry &point_,const Color &fillColor_,
492       const Color &borderColor_,const bool invert_=false);
493     floodFillColorImage(const ::ssize_t x_,const ::ssize_t y_,
494       const Color &fillColor_,const Color &borderColor_,
495       const bool invert_=false);
496 
497     void operator()(Image &image_) const;
498 
499   private:
500     ::ssize_t _x;
501     ::ssize_t _y;
502     Color     _fillColor;
503     Color     _borderColor;
504     bool      _invert;
505   };
506 
507   // Flood-fill image with texture
508   class MagickPPExport floodFillTextureImage
509 
510   {
511   public:
512     // Flood-fill texture across pixels that match the color of the
513     // target pixel and are neighbors of the target pixel.
514     // Uses current fuzz setting when determining color match.
515     floodFillTextureImage(const ::ssize_t x_,const ::ssize_t y_,
516       const Image &texture_,const bool invert_=false);
517     floodFillTextureImage(const Geometry &point_,const Image &texture_,
518       const bool invert_=false);
519 
520     // Flood-fill texture across pixels starting at target-pixel and
521     // stopping at pixels matching specified border color.
522     // Uses current fuzz setting when determining color match.
523     floodFillTextureImage(const ::ssize_t x_,const ::ssize_t y_,
524       const Image &texture_,const Color &borderColor_,
525       const bool invert_=false);
526 
527     floodFillTextureImage(const Geometry &point_,const Image &texture_,
528       const Color &borderColor_,const bool invert_=false);
529 
530     void operator()(Image &image_) const;
531 
532   private:
533     ::ssize_t _x;
534     ::ssize_t _y;
535     Image     _texture;
536     Color     _borderColor;
537     bool      _invert;
538   };
539 
540   // Flop image (reflect each scanline in the horizontal direction)
541   class MagickPPExport flopImage
542   {
543   public:
544     flopImage( void );
545 
546     void operator()( Image &image_ ) const;
547 
548   private:
549   };
550 
551   // Frame image
552   class MagickPPExport frameImage
553   {
554   public:
555     frameImage( const Geometry &geometry_ = frameGeometryDefault );
556 
557     frameImage( const size_t width_, const size_t height_,
558     const ::ssize_t innerBevel_ = 6, const ::ssize_t outerBevel_ = 6 );
559 
560     void operator()( Image &image_ ) const;
561 
562   private:
563     size_t _width;
564     size_t _height;
565     ::ssize_t        _outerBevel;
566     ::ssize_t        _innerBevel;
567   };
568 
569   // Gamma correct image
570   class MagickPPExport gammaImage
571   {
572   public:
573     gammaImage( const double gamma_ );
574 
575     gammaImage ( const double gammaRed_,
576      const double gammaGreen_,
577      const double gammaBlue_ );
578 
579     void operator()( Image &image_ ) const;
580 
581   private:
582     double _gammaRed;
583     double _gammaGreen;
584     double _gammaBlue;
585   };
586 
587   // Gaussian blur image
588   // The number of neighbor pixels to be included in the convolution
589   // mask is specified by 'width_'. The standard deviation of the
590   // gaussian bell curve is specified by 'sigma_'.
591   class MagickPPExport gaussianBlurImage
592   {
593   public:
594     gaussianBlurImage( const double width_, const double sigma_ );
595 
596     void operator()( Image &image_ ) const;
597 
598   private:
599     double _width;
600     double _sigma;
601   };
602 
603   // Apply a color lookup table (Hald CLUT) to the image.
604   class MagickPPExport haldClutImage
605   {
606   public:
607     haldClutImage( const Image &haldClutImage_ );
608 
609     void operator()( Image &image_ ) const;
610 
611   private:
612     Image             _haldClutImage;
613   };
614 
615   // Implode image (special effect)
616   class MagickPPExport implodeImage
617   {
618   public:
619     implodeImage( const double factor_ = 50 );
620 
621     void operator()( Image &image_ ) const;
622 
623   private:
624     double _factor;
625   };
626 
627   // implements the inverse discrete Fourier transform (IFT) of the image
628   // either as a magnitude / phase or real / imaginary image pair.
629   class MagickPPExport inverseFourierTransformImage
630   {
631   public:
632     inverseFourierTransformImage( const Image &phaseImage_ );
633 
634     void operator()( Image &image_ ) const;
635 
636   private:
637     Image _phaseImage;
638   };
639 
640   // Set image validity. Valid images become empty (inValid) if
641   // argument is false.
642   class MagickPPExport isValidImage
643   {
644   public:
645     isValidImage( const bool isValid_ );
646 
647     void operator()( Image &image_ ) const;
648 
649   private:
650     bool _isValid;
651   };
652 
653   // Label image
654   class MagickPPExport labelImage
655   {
656   public:
657     labelImage( const std::string &label_ );
658 
659     void operator()( Image &image_ ) const;
660 
661   private:
662     std::string _label;
663   };
664 
665 
666   // Level image
667   class MagickPPExport levelImage
668   {
669   public:
670     levelImage( const double black_point,
671                 const double white_point,
672                 const double mid_point=1.0 );
673 
674     void operator()( Image &image_ ) const;
675 
676   private:
677     double _black_point;
678     double _white_point;
679     double _mid_point;
680   };
681 
682   // Magnify image by integral size
683   class MagickPPExport magnifyImage
684   {
685   public:
686     magnifyImage( void );
687 
688     void operator()( Image &image_ ) const;
689 
690   private:
691   };
692 
693   // Remap image colors with closest color from reference image
694   class MagickPPExport mapImage
695   {
696   public:
697     mapImage( const Image &mapImage_ ,
698               const bool dither_ = false );
699 
700     void operator()( Image &image_ ) const;
701 
702   private:
703     Image   _mapImage;
704     bool    _dither;
705   };
706 
707   // Filter image by replacing each pixel component with the median
708   // color in a circular neighborhood
709   class MagickPPExport medianConvolveImage
710   {
711   public:
712     medianConvolveImage( const double radius_ = 0.0 );
713 
714     void operator()( Image &image_ ) const;
715 
716   private:
717     double _radius;
718   };
719 
720   // Merge image layers
721   class MagickPPExport mergeLayersImage
722   {
723   public:
724     mergeLayersImage ( LayerMethod layerMethod_ );
725 
726     void operator()( Image &image_ ) const;
727 
728   private:
729     LayerMethod _layerMethod;
730   };
731 
732   // Reduce image by integral size
733   class MagickPPExport minifyImage
734   {
735   public:
736     minifyImage( void );
737 
738     void operator()( Image &image_ ) const;
739 
740   private:
741   };
742 
743   // Modulate percent hue, saturation, and brightness of an image
744   class MagickPPExport modulateImage
745   {
746   public:
747     modulateImage( const double brightness_,
748        const double saturation_,
749        const double hue_ );
750 
751     void operator()( Image &image_ ) const;
752 
753   private:
754     double _brightness;
755     double _saturation;
756     double _hue;
757   };
758 
759   // Negate colors in image.  Set grayscale to only negate grayscale
760   // values in image.
761   class MagickPPExport negateImage
762   {
763   public:
764     negateImage( const bool grayscale_ = false );
765 
766     void operator()( Image &image_ ) const;
767 
768   private:
769     bool _grayscale;
770   };
771 
772   // Normalize image (increase contrast by normalizing the pixel
773   // values to span the full range of color values)
774   class MagickPPExport normalizeImage
775   {
776   public:
777     normalizeImage( void );
778 
779     void operator()( Image &image_ ) const;
780 
781   private:
782   };
783 
784   // Oilpaint image (image looks like oil painting)
785   class MagickPPExport oilPaintImage
786   {
787   public:
788     oilPaintImage( const double radius_ = 3 );
789 
790     void operator()( Image &image_ ) const;
791 
792   private:
793     double _radius;
794   };
795 
796   // Set or attenuate the image alpha channel. If the image pixels
797   // are opaque then they are set to the specified alpha value,
798   // otherwise they are blended with the supplied alpha value.  The
799   // value of alpha_ ranges from 0 (completely opaque) to
800   // QuantumRange. The defines OpaqueAlpha and TransparentAlpha are
801   // available to specify completely opaque or completely transparent,
802   // respectively.
803   class MagickPPExport alphaImage
804   {
805   public:
806     alphaImage( const unsigned int alpha_ );
807 
808     void operator()( Image &image_ ) const;
809 
810   private:
811     unsigned int _alpha;
812   };
813 
814   // Change color of opaque pixel to specified pen color.
815   class MagickPPExport opaqueImage
816   {
817   public:
818     opaqueImage( const Color &opaqueColor_,
819      const Color &penColor_ );
820 
821     void operator()( Image &image_ ) const;
822 
823   private:
824     Color  _opaqueColor;
825     Color  _penColor;
826   };
827 
828   // Quantize image (reduce number of colors)
829   class MagickPPExport quantizeImage
830   {
831   public:
832     quantizeImage( const bool measureError_ = false );
833 
834     void operator()( Image &image_ ) const;
835 
836   private:
837     bool _measureError;
838   };
839 
840   // Raise image (lighten or darken the edges of an image to give a
841   // 3-D raised or lowered effect)
842   class MagickPPExport raiseImage
843   {
844   public:
845     raiseImage( const Geometry &geometry_ = raiseGeometryDefault,
846     const bool raisedFlag_ = false );
847 
848     void operator()( Image &image_ ) const;
849 
850   private:
851     Geometry   _geometry;
852     bool       _raisedFlag;
853   };
854 
855   class MagickPPExport ReadOptions
856   {
857   public:
858 
859     // Default constructor
860     ReadOptions(void);
861 
862     // Copy constructor
863     ReadOptions(const ReadOptions& options_);
864 
865     // Destructor
866     ~ReadOptions();
867 
868     // Vertical and horizontal resolution in pixels of the image
869     void density(const Geometry &geomery_);
870     Geometry density(void) const;
871 
872     // Image depth (8 or 16)
873     void depth(size_t depth_);
874     size_t depth(void) const;
875 
876     // Ping the image instead of reading it
877     void ping(const bool flag_);
878     bool ping(void) const;
879 
880     // Suppress all warning messages. Error messages are still reported.
881     void quiet(const bool quiet_);
882     bool quiet(void) const;
883 
884     // Image size (required for raw formats)
885     void size(const Geometry &geometry_);
886     Geometry size(void) const;
887 
888     //
889     // Internal implementation methods.  Please do not use.
890     //
891 
892     MagickCore::ImageInfo *imageInfo(void);
893 
894   private:
895 
896     // Assignment not supported
897     ReadOptions& operator=(const ReadOptions&);
898 
899     MagickCore::ImageInfo *_imageInfo;
900     bool                  _quiet;
901   };
902 
903   // Reduce noise in image using a noise peak elimination filter
904   class MagickPPExport reduceNoiseImage
905   {
906   public:
907     reduceNoiseImage( void );
908 
909     reduceNoiseImage (const  size_t order_ );
910 
911     void operator()( Image &image_ ) const;
912 
913   private:
914     size_t _order;
915   };
916 
917   // Resize image to specified size.
918   class MagickPPExport resizeImage
919   {
920   public:
921     resizeImage( const Geometry &geometry_ );
922 
923     void operator()( Image &image_ ) const;
924 
925   private:
926     Geometry _geometry;
927   };
928 
929   // Roll image (rolls image vertically and horizontally) by specified
930   // number of columnms and rows)
931   class MagickPPExport rollImage
932   {
933   public:
934     rollImage( const Geometry &roll_ );
935 
936     rollImage( const ::ssize_t columns_, const ::ssize_t rows_ );
937 
938     void operator()( Image &image_ ) const;
939 
940   private:
941     size_t _columns;
942     size_t _rows;
943   };
944 
945   // Rotate image counter-clockwise by specified number of degrees.
946   class MagickPPExport rotateImage
947   {
948   public:
949     rotateImage( const double degrees_ );
950 
951     void operator()( Image &image_ ) const;
952 
953   private:
954     double       _degrees;
955   };
956 
957   // Resize image by using pixel sampling algorithm
958   class MagickPPExport sampleImage
959   {
960   public:
961     sampleImage( const Geometry &geometry_ );
962 
963     void operator()( Image &image_ ) const;
964 
965   private:
966     Geometry  _geometry;
967   };
968 
969   // Resize image by using simple ratio algorithm
970   class MagickPPExport scaleImage
971   {
972   public:
973     scaleImage( const Geometry &geometry_ );
974 
975     void operator()( Image &image_ ) const;
976 
977   private:
978     Geometry  _geometry;
979   };
980 
981   // Segment (coalesce similar image components) by analyzing the
982   // histograms of the color components and identifying units that are
983   // homogeneous with the fuzzy c-means technique.
984   // Also uses QuantizeColorSpace and Verbose image attributes
985   class MagickPPExport segmentImage
986   {
987   public:
988     segmentImage( const double clusterThreshold_ = 1.0,
989       const double smoothingThreshold_ = 1.5 );
990 
991     void operator()( Image &image_ ) const;
992 
993   private:
994     double  _clusterThreshold;
995     double  _smoothingThreshold;
996   };
997 
998   // Shade image using distant light source
999   class MagickPPExport shadeImage
1000   {
1001   public:
1002     shadeImage( const double azimuth_ = 30,
1003     const double elevation_ = 30,
1004     const bool   colorShading_ = false );
1005 
1006     void operator()( Image &image_ ) const;
1007 
1008   private:
1009     double  _azimuth;
1010     double  _elevation;
1011     bool    _colorShading;
1012   };
1013 
1014   // Shadow effect image (simulate an image shadow)
1015   class MagickPPExport shadowImage
1016   {
1017   public:
1018     shadowImage( const double percent_opacity_ = 80, const double sigma_ = 0.5,
1019       const ssize_t x_ = 5, const ssize_t y_ = 5 );
1020 
1021     void operator()( Image &image_ ) const;
1022 
1023   private:
1024     double _percent_opacity;
1025     double _sigma;
1026     ssize_t _x;
1027     ssize_t _y;
1028   };
1029 
1030   // Sharpen pixels in image
1031   class MagickPPExport sharpenImage
1032   {
1033   public:
1034     sharpenImage( const double radius_ = 1, const double sigma_ = 0.5 );
1035 
1036     void operator()( Image &image_ ) const;
1037 
1038   private:
1039     double _radius;
1040     double _sigma;
1041   };
1042 
1043   // Shave pixels from image edges.
1044   class MagickPPExport shaveImage
1045   {
1046   public:
1047     shaveImage( const Geometry &geometry_ );
1048 
1049     void operator()( Image &image_ ) const;
1050 
1051   private:
1052     Geometry _geometry;
1053   };
1054 
1055 
1056   // Shear image (create parallelogram by sliding image by X or Y axis)
1057   class MagickPPExport shearImage
1058   {
1059   public:
1060     shearImage( const double xShearAngle_,
1061     const double yShearAngle_ );
1062 
1063     void operator()( Image &image_ ) const;
1064 
1065   private:
1066     double _xShearAngle;
1067     double _yShearAngle;
1068   };
1069 
1070   // Solarize image (similar to effect seen when exposing a
1071   // photographic film to light during the development process)
1072   class MagickPPExport solarizeImage
1073   {
1074   public:
1075     solarizeImage( const double factor_ );
1076 
1077     void operator()( Image &image_ ) const;
1078 
1079   private:
1080     double _factor;
1081   };
1082 
1083   // Splice the background color into the image.
1084   class MagickPPExport spliceImage
1085   {
1086   public:
1087     spliceImage( const Geometry &geometry_ );
1088 
1089     void operator()( Image &image_ ) const;
1090 
1091   private:
1092     Geometry _geometry;
1093   };
1094 
1095   // Spread pixels randomly within image by specified ammount
1096   class MagickPPExport spreadImage
1097   {
1098   public:
1099     spreadImage( const size_t amount_ = 3 );
1100 
1101     void operator()( Image &image_ ) const;
1102 
1103   private:
1104     size_t _amount;
1105   };
1106 
1107   // Add a digital watermark to the image (based on second image)
1108   class MagickPPExport steganoImage
1109   {
1110   public:
1111     steganoImage( const Image &waterMark_ );
1112 
1113     void operator()( Image &image_ ) const;
1114 
1115   private:
1116     Image _waterMark;
1117   };
1118 
1119   // Create an image which appears in stereo when viewed with red-blue glasses
1120   // (Red image on left, blue on right)
1121   class MagickPPExport stereoImage
1122   {
1123   public:
1124     stereoImage( const Image &rightImage_ );
1125 
1126     void operator()( Image &image_ ) const;
1127 
1128   private:
1129     Image _rightImage;
1130   };
1131 
1132   // Color to use when drawing object outlines
1133   class MagickPPExport strokeColorImage
1134   {
1135   public:
1136     strokeColorImage( const Color &strokeColor_ );
1137 
1138     void operator()( Image &image_ ) const;
1139 
1140   private:
1141     Color _strokeColor;
1142   };
1143 
1144   // Swirl image (image pixels are rotated by degrees)
1145   class MagickPPExport swirlImage
1146   {
1147   public:
1148     swirlImage( const double degrees_ );
1149 
1150     void operator()( Image &image_ ) const;
1151 
1152   private:
1153     double _degrees;
1154   };
1155 
1156   // Channel a texture on image background
1157   class MagickPPExport textureImage
1158   {
1159   public:
1160     textureImage( const Image &texture_ );
1161 
1162     void operator()( Image &image_ ) const;
1163 
1164   private:
1165     Image _texture;
1166   };
1167 
1168   // Threshold image
1169   class MagickPPExport thresholdImage
1170   {
1171   public:
1172     thresholdImage( const double threshold_ );
1173 
1174     void operator()( Image &image_ ) const;
1175 
1176   private:
1177     double _threshold;
1178   };
1179 
1180   // Set image color to transparent
1181   class MagickPPExport transparentImage
1182   {
1183   public:
1184     transparentImage( const Color& color_ );
1185 
1186     void operator()( Image &image_ ) const;
1187 
1188   private:
1189     Color _color;
1190   };
1191 
1192   // Trim edges that are the background color from the image
1193   class MagickPPExport trimImage
1194   {
1195   public:
1196     trimImage( void );
1197 
1198     void operator()( Image &image_ ) const;
1199 
1200   private:
1201   };
1202 
1203   // Map image pixels to a sine wave
1204   class MagickPPExport waveImage
1205   {
1206   public:
1207     waveImage( const double amplitude_ = 25.0,
1208          const double wavelength_ = 150.0 );
1209 
1210     void operator()( Image &image_ ) const;
1211 
1212   private:
1213     double _amplitude;
1214     double _wavelength;
1215   };
1216 
1217   // Zoom image to specified size.
1218   class MagickPPExport zoomImage
1219   {
1220   public:
1221     zoomImage( const Geometry &geometry_ );
1222 
1223     void operator()( Image &image_ ) const;
1224 
1225   private:
1226     Geometry _geometry;
1227   };
1228 
1229   //
1230   // Function object image attribute accessors
1231   //
1232 
1233   // Join images into a single multi-image file
1234   class MagickPPExport adjoinImage
1235   {
1236   public:
1237     adjoinImage( const bool flag_ );
1238 
1239     void operator()( Image &image_ ) const;
1240 
1241   private:
1242     bool _flag;
1243   };
1244 
1245   // Time in 1/100ths of a second which must expire before displaying
1246   // the next image in an animated sequence.
1247   class MagickPPExport animationDelayImage
1248   {
1249   public:
1250     animationDelayImage( const size_t delay_ );
1251 
1252     void operator()( Image &image_ ) const;
1253 
1254   private:
1255     size_t _delay;
1256   };
1257 
1258   // Number of iterations to loop an animation (e.g. Netscape loop
1259   // extension) for.
1260   class MagickPPExport animationIterationsImage
1261   {
1262   public:
1263     animationIterationsImage( const size_t iterations_ );
1264 
1265     void operator()( Image &image_ ) const;
1266 
1267   private:
1268     size_t _iterations;
1269   };
1270 
1271   // Image background color
1272   class MagickPPExport backgroundColorImage
1273   {
1274   public:
1275     backgroundColorImage( const Color &color_ );
1276 
1277     void operator()( Image &image_ ) const;
1278 
1279   private:
1280     Color _color;
1281   };
1282 
1283   // Name of texture image to tile onto the image background
1284   class MagickPPExport backgroundTextureImage
1285   {
1286   public:
1287     backgroundTextureImage( const std::string &backgroundTexture_ );
1288 
1289     void operator()( Image &image_ ) const;
1290 
1291   private:
1292     std::string _backgroundTexture;
1293   };
1294 
1295   // Image border color
1296   class MagickPPExport borderColorImage
1297   {
1298   public:
1299     borderColorImage( const Color &color_ );
1300 
1301     void operator()( Image &image_ ) const;
1302 
1303   private:
1304     Color _color;
1305   };
1306 
1307   // Text bounding-box base color (default none)
1308   class MagickPPExport boxColorImage
1309   {
1310   public:
1311     boxColorImage( const Color &boxColor_ );
1312 
1313     void operator()( Image &image_ ) const;
1314 
1315   private:
1316     Color _boxColor;
1317   };
1318 
1319   // Chromaticity blue primary point.
1320   class MagickPPExport chromaBluePrimaryImage
1321   {
1322   public:
1323     chromaBluePrimaryImage(const double x_,const double y_,const double z_);
1324 
1325     void operator()(Image &image_) const;
1326 
1327   private:
1328     double _x;
1329     double _y;
1330     double _z;
1331   };
1332 
1333   // Chromaticity green primary point.
1334   class MagickPPExport chromaGreenPrimaryImage
1335   {
1336   public:
1337     chromaGreenPrimaryImage(const double x_,const double y_,const double z_);
1338 
1339     void operator()(Image &image_) const;
1340 
1341   private:
1342     double _x;
1343     double _y;
1344     double _z;
1345   };
1346 
1347   // Chromaticity red primary point.
1348   class MagickPPExport chromaRedPrimaryImage
1349   {
1350   public:
1351     chromaRedPrimaryImage(const double x_,const double y_,const double z_);
1352 
1353     void operator()(Image &image_) const;
1354 
1355   private:
1356     double _x;
1357     double _y;
1358     double _z;
1359   };
1360 
1361   // Chromaticity white point.
1362   class MagickPPExport chromaWhitePointImage
1363   {
1364   public:
1365     chromaWhitePointImage(const double x_,const double y_,const double z_);
1366 
1367     void operator()(Image &image_) const;
1368 
1369   private:
1370     double _x;
1371     double _y;
1372     double _z;
1373   };
1374 
1375   // Colors within this distance are considered equal
1376   class MagickPPExport colorFuzzImage
1377   {
1378   public:
1379     colorFuzzImage( const double fuzz_ );
1380 
1381     void operator()( Image &image_ ) const;
1382 
1383   private:
1384     double _fuzz;
1385   };
1386 
1387   // Color at colormap position index_
1388   class MagickPPExport colorMapImage
1389   {
1390   public:
1391     colorMapImage( const size_t index_, const Color &color_ );
1392 
1393     void operator()( Image &image_ ) const;
1394 
1395   private:
1396     size_t _index;
1397     Color        _color;
1398   };
1399 
1400   // Composition operator to be used when composition is implicitly used
1401   // (such as for image flattening).
1402   class MagickPPExport composeImage
1403   {
1404   public:
1405     composeImage( const CompositeOperator compose_ );
1406 
1407     void operator()( Image &image_ ) const;
1408 
1409   private:
1410     CompositeOperator _compose;
1411   };
1412 
1413   // Compression type
1414   class MagickPPExport compressTypeImage
1415   {
1416   public:
1417     compressTypeImage( const CompressionType compressType_ );
1418 
1419     void operator()( Image &image_ ) const;
1420 
1421   private:
1422     CompressionType _compressType;
1423   };
1424 
1425   // Vertical and horizontal resolution in pixels of the image
1426   class MagickPPExport densityImage
1427   {
1428   public:
1429     densityImage( const Point &point_ );
1430 
1431     void operator()( Image &image_ ) const;
1432 
1433   private:
1434     Point _point;
1435   };
1436 
1437   // Image depth (bits allocated to red/green/blue components)
1438   class MagickPPExport depthImage
1439   {
1440   public:
1441     depthImage( const size_t depth_ );
1442 
1443     void operator()( Image &image_ ) const;
1444 
1445   private:
1446     size_t _depth;
1447   };
1448 
1449   // Endianness (LSBEndian like Intel or MSBEndian like SPARC) for image
1450   // formats which support endian-specific options.
1451   class MagickPPExport endianImage
1452   {
1453   public:
1454     endianImage( const EndianType endian_ );
1455 
1456     void operator()( Image &image_ ) const;
1457 
1458   private:
1459     EndianType  _endian;
1460   };
1461 
1462   // Image file name
1463   class MagickPPExport fileNameImage
1464   {
1465   public:
1466     fileNameImage( const std::string &fileName_ );
1467 
1468     void operator()( Image &image_ ) const;
1469 
1470   private:
1471     std::string _fileName;
1472   };
1473 
1474   // Filter to use when resizing image
1475   class MagickPPExport filterTypeImage
1476   {
1477   public:
1478     filterTypeImage( const FilterType filterType_ );
1479 
1480     void operator()( Image &image_ ) const;
1481 
1482   private:
1483     FilterType _filterType;
1484   };
1485 
1486   // Text rendering font
1487   class MagickPPExport fontImage
1488   {
1489   public:
1490     fontImage( const std::string &font_ );
1491 
1492     void operator()( Image &image_ ) const;
1493 
1494   private:
1495     std::string _font;
1496   };
1497 
1498   // Font point size
1499   class MagickPPExport fontPointsizeImage
1500   {
1501   public:
1502     fontPointsizeImage( const size_t pointsize_ );
1503 
1504     void operator()( Image &image_ ) const;
1505 
1506   private:
1507     size_t _pointsize;
1508   };
1509 
1510   // GIF disposal method
1511   class MagickPPExport gifDisposeMethodImage
1512   {
1513   public:
1514     gifDisposeMethodImage( const DisposeType disposeMethod_ );
1515 
1516     void operator()( Image &image_ ) const;
1517 
1518   private:
1519     DisposeType _disposeMethod;
1520   };
1521 
1522   // Type of interlacing to use
1523   class MagickPPExport interlaceTypeImage
1524   {
1525   public:
1526     interlaceTypeImage( const InterlaceType interlace_ );
1527 
1528     void operator()( Image &image_ ) const;
1529 
1530   private:
1531     InterlaceType _interlace;
1532   };
1533 
1534   // File type magick identifier (.e.g "GIF")
1535   class MagickPPExport magickImage
1536   {
1537   public:
1538     magickImage( const std::string &magick_ );
1539 
1540     void operator()( Image &image_ ) const;
1541 
1542   private:
1543     std::string _magick;
1544   };
1545 
1546   // Image supports transparent color
1547   class MagickPPExport alphaFlagImage
1548   {
1549   public:
1550     alphaFlagImage( const bool alphaFlag_ );
1551 
1552     void operator()( Image &image_ ) const;
1553 
1554   private:
1555     bool _alphaFlag;
1556   };
1557 
1558   // Transparent color
1559   class MagickPPExport matteColorImage
1560   {
1561   public:
1562     matteColorImage( const Color &matteColor_ );
1563 
1564     void operator()( Image &image_ ) const;
1565 
1566   private:
1567     Color _matteColor;
1568   };
1569 
1570   // Indicate that image is black and white
1571   class MagickPPExport monochromeImage
1572   {
1573   public:
1574     monochromeImage( const bool monochromeFlag_ );
1575 
1576     void operator()( Image &image_ ) const;
1577 
1578   private:
1579     bool _monochromeFlag;
1580   };
1581 
1582   // Pen color
1583   class MagickPPExport penColorImage
1584   {
1585   public:
1586     penColorImage( const Color &penColor_ );
1587 
1588     void operator()( Image &image_ ) const;
1589 
1590   private:
1591     Color _penColor;
1592   };
1593 
1594   // Pen texture image.
1595   class MagickPPExport penTextureImage
1596   {
1597   public:
1598     penTextureImage( const Image &penTexture_ );
1599 
1600     void operator()( Image &image_ ) const;
1601 
1602   private:
1603     Image _penTexture;
1604   };
1605 
1606   // Set pixel color at location x & y.
1607   class MagickPPExport pixelColorImage
1608   {
1609   public:
1610     pixelColorImage( const ::ssize_t x_,
1611                      const ::ssize_t y_,
1612          const Color &color_);
1613 
1614     void operator()( Image &image_ ) const;
1615 
1616   private:
1617     ::ssize_t    _x;
1618     ::ssize_t    _y;
1619     Color        _color;
1620   };
1621 
1622   // Postscript page size.
1623   class MagickPPExport pageImage
1624   {
1625   public:
1626     pageImage( const Geometry &pageSize_ );
1627 
1628     void operator()( Image &image_ ) const;
1629 
1630   private:
1631     Geometry _pageSize;
1632   };
1633 
1634   // JPEG/MIFF/PNG compression level (default 75).
1635   class MagickPPExport qualityImage
1636   {
1637   public:
1638     qualityImage( const size_t quality_ );
1639 
1640     void operator()( Image &image_ ) const;
1641 
1642   private:
1643     size_t _quality;
1644   };
1645 
1646   // Maximum number of colors to quantize to
1647   class MagickPPExport quantizeColorsImage
1648   {
1649   public:
1650     quantizeColorsImage( const size_t colors_ );
1651 
1652     void operator()( Image &image_ ) const;
1653 
1654   private:
1655     size_t _colors;
1656   };
1657 
1658   // Colorspace to quantize in.
1659   class MagickPPExport quantizeColorSpaceImage
1660   {
1661   public:
1662     quantizeColorSpaceImage( const ColorspaceType colorSpace_ );
1663 
1664     void operator()( Image &image_ ) const;
1665 
1666   private:
1667     ColorspaceType _colorSpace;
1668   };
1669 
1670   // Dither image during quantization (default true).
1671   class MagickPPExport quantizeDitherImage
1672   {
1673   public:
1674     quantizeDitherImage( const bool ditherFlag_ );
1675 
1676     void operator()( Image &image_ ) const;
1677 
1678   private:
1679     bool _ditherFlag;
1680   };
1681 
1682   // Quantization tree-depth
1683   class MagickPPExport quantizeTreeDepthImage
1684   {
1685   public:
1686     quantizeTreeDepthImage( const size_t treeDepth_ );
1687 
1688     void operator()( Image &image_ ) const;
1689 
1690   private:
1691     size_t _treeDepth;
1692   };
1693 
1694   // The type of rendering intent
1695   class MagickPPExport renderingIntentImage
1696   {
1697   public:
1698     renderingIntentImage( const RenderingIntent renderingIntent_ );
1699 
1700     void operator()( Image &image_ ) const;
1701 
1702   private:
1703     RenderingIntent _renderingIntent;
1704   };
1705 
1706   // Units of image resolution
1707   class MagickPPExport resolutionUnitsImage
1708   {
1709   public:
1710     resolutionUnitsImage( const ResolutionType resolutionUnits_ );
1711 
1712     void operator()( Image &image_ ) const;
1713 
1714   private:
1715     ResolutionType _resolutionUnits;
1716   };
1717 
1718   // Image scene number
1719   class MagickPPExport sceneImage
1720   {
1721   public:
1722     sceneImage( const size_t scene_ );
1723 
1724     void operator()( Image &image_ ) const;
1725 
1726   private:
1727     size_t _scene;
1728   };
1729 
1730   // adjust the image contrast with a non-linear sigmoidal contrast algorithm
1731   class MagickPPExport sigmoidalContrastImage
1732   {
1733   public:
1734     sigmoidalContrastImage( const size_t sharpen_,
1735       const double contrast,
1736       const double midpoint = QuantumRange / 2.0 );
1737 
1738     void operator()( Image &image_ ) const;
1739 
1740   private:
1741     size_t _sharpen;
1742     double contrast;
1743     double midpoint;
1744   };
1745 
1746   // Width and height of a raw image
1747   class MagickPPExport sizeImage
1748   {
1749   public:
1750     sizeImage( const Geometry &geometry_ );
1751 
1752     void operator()( Image &image_ ) const;
1753 
1754   private:
1755     Geometry _geometry;
1756   };
1757 
1758   // stripImage strips an image of all profiles and comments.
1759   class MagickPPExport stripImage
1760   {
1761   public:
1762     stripImage( void );
1763 
1764     void operator()( Image &image_ ) const;
1765 
1766   private:
1767   };
1768 
1769   // Subimage of an image sequence
1770   class MagickPPExport subImageImage
1771   {
1772   public:
1773     subImageImage( const size_t subImage_ );
1774 
1775     void operator()( Image &image_ ) const;
1776 
1777   private:
1778     size_t _subImage;
1779   };
1780 
1781   // Number of images relative to the base image
1782   class MagickPPExport subRangeImage
1783   {
1784   public:
1785     subRangeImage( const size_t subRange_ );
1786 
1787     void operator()( Image &image_ ) const;
1788 
1789   private:
1790     size_t _subRange;
1791   };
1792 
1793   // Anti-alias Postscript and TrueType fonts (default true)
1794   class MagickPPExport textAntiAliasImage
1795   {
1796   public:
1797     textAntiAliasImage( const bool flag_ );
1798 
1799     void operator()( Image &image_ ) const;
1800 
1801   private:
1802     bool _flag;
1803   };
1804 
1805   // Image storage type
1806   class MagickPPExport typeImage
1807   {
1808   public:
1809     typeImage( const ImageType type_ );
1810 
1811     void operator()( Image &image_ ) const;
1812 
1813   private:
1814     Magick::ImageType _type;
1815   };
1816 
1817 
1818   // Print detailed information about the image
1819   class MagickPPExport verboseImage
1820   {
1821   public:
1822     verboseImage( const bool verbose_ );
1823 
1824     void operator()( Image &image_ ) const;
1825 
1826   private:
1827     bool _verbose;
1828   };
1829 
1830   // X11 display to display to, obtain fonts from, or to capture
1831   // image from
1832   class MagickPPExport x11DisplayImage
1833   {
1834   public:
1835     x11DisplayImage( const std::string &display_ );
1836 
1837     void operator()( Image &image_ ) const;
1838 
1839   private:
1840     std::string _display;
1841   };
1842 
1843   //////////////////////////////////////////////////////////
1844   //
1845   // Implementation template definitions. Not for end-use.
1846   //
1847   //////////////////////////////////////////////////////////
1848 
1849   // Changes the channel mask of the images and places the old
1850   // values in the container.
1851   template<class InputIterator, class Container>
channelMaskImages(InputIterator first_,InputIterator last_,Container * container_,const ChannelType channel_)1852   void channelMaskImages(InputIterator first_,InputIterator last_,
1853     Container *container_,const ChannelType channel_)
1854   {
1855     MagickCore::ChannelType
1856       channel_mask;
1857 
1858     container_->clear();
1859     for (InputIterator iter = first_; iter != last_; ++iter)
1860     {
1861       iter->modifyImage();
1862       channel_mask=MagickCore::SetImageChannelMask(iter->image(),channel_);
1863       container_->push_back(channel_mask);
1864     }
1865   }
1866 
1867   // Insert images in image list into existing container (appending to container)
1868   // The images should not be deleted since only the image ownership is passed.
1869   // The options are copied into the object.
1870   template<class Container>
insertImages(Container * sequence_,MagickCore::Image * images_)1871   void insertImages(Container *sequence_,MagickCore::Image* images_)
1872   {
1873     MagickCore::Image
1874       *image,
1875       *next;
1876 
1877     image=images_;
1878     while (image != (MagickCore::Image *) NULL)
1879     {
1880       next=image->next;
1881       image->next=(MagickCore::Image *) NULL;
1882 
1883       if (next != (MagickCore::Image *) NULL)
1884         next->previous=(MagickCore::Image *) NULL;
1885 
1886       sequence_->push_back(Magick::Image(image));
1887 
1888       image=next;
1889     }
1890   }
1891 
1892   // Link images together into an image list based on the ordering of
1893   // the container implied by the iterator. This step is done in
1894   // preparation for use with ImageMagick functions which operate on
1895   // lists of images.
1896   // Images are selected by range, first_ to last_ so that a subset of
1897   // the container may be selected.  Specify first_ via the
1898   // container's begin() method and last_ via the container's end()
1899   // method in order to specify the entire container.
1900   template<class InputIterator>
linkImages(InputIterator first_,InputIterator last_)1901   bool linkImages(InputIterator first_,InputIterator last_)
1902   {
1903     MagickCore::Image
1904       *current,
1905       *previous;
1906 
1907     ::ssize_t
1908       scene;
1909 
1910     scene=0;
1911     previous=(MagickCore::Image *) NULL;
1912     for (InputIterator iter = first_; iter != last_; ++iter)
1913     {
1914       // Unless we reduce the reference count to one, the same image
1915       // structure may occur more than once in the container, causing
1916       // the linked list to fail.
1917       iter->modifyImage();
1918 
1919       current=iter->image();
1920 
1921       current->previous=previous;
1922       current->next=(MagickCore::Image *) NULL;
1923       current->scene=scene++;
1924 
1925       if (previous != (MagickCore::Image *) NULL)
1926         previous->next=current;
1927 
1928       previous=current;
1929     }
1930     return(scene > 0 ? true : false);
1931   }
1932 
1933   // Restores the channel mask of the images.
1934   template<class InputIterator, class Container>
restoreChannelMaskImages(InputIterator first_,InputIterator last_,Container * container_)1935   void restoreChannelMaskImages(InputIterator first_,InputIterator last_,
1936     Container *container_)
1937   {
1938     typename Container::iterator
1939       channel_mask;
1940 
1941     channel_mask=container_->begin();
1942     for (InputIterator iter = first_; iter != last_; ++iter)
1943     {
1944       iter->modifyImage();
1945       (void) MagickCore::SetImageChannelMask(iter->image(),
1946         (const MagickCore::ChannelType) *channel_mask);
1947       channel_mask++;
1948     }
1949   }
1950 
1951   // Remove links added by linkImages. This should be called after the
1952   // ImageMagick function call has completed to reset the image list
1953   // back to its pristine un-linked state.
1954   template<class InputIterator>
unlinkImages(InputIterator first_,InputIterator last_)1955   void unlinkImages(InputIterator first_,InputIterator last_)
1956   {
1957     MagickCore::Image
1958       *image;
1959 
1960     for (InputIterator iter = first_; iter != last_; ++iter)
1961     {
1962       image=iter->image();
1963       image->previous=(MagickCore::Image *) NULL;
1964       image->next=(MagickCore::Image *) NULL;
1965     }
1966   }
1967 
1968   ///////////////////////////////////////////////////////////////////
1969   //
1970   // Template definitions for documented API
1971   //
1972   ///////////////////////////////////////////////////////////////////
1973 
1974   template <class InputIterator>
animateImages(InputIterator first_,InputIterator last_)1975   void animateImages( InputIterator first_,InputIterator last_)
1976   {
1977     if (linkImages(first_,last_) == false)
1978       return;
1979     GetPPException;
1980     MagickCore::AnimateImages(first_->imageInfo(),first_->image(),
1981       exceptionInfo);
1982     unlinkImages(first_,last_);
1983     ThrowPPException(first_->quiet());
1984   }
1985 
1986   // Append images from list into single image in either horizontal or
1987   // vertical direction.
1988   template <class InputIterator>
1989   void appendImages( Image *appendedImage_,
1990          InputIterator first_,
1991          InputIterator last_,
1992          bool stack_ = false) {
1993     if (linkImages(first_,last_) == false)
1994       return;
1995     GetPPException;
1996     MagickCore::Image* image = MagickCore::AppendImages( first_->image(),
1997                    (MagickBooleanType) stack_,
1998                    exceptionInfo );
1999     unlinkImages( first_, last_ );
2000     appendedImage_->replaceImage( image );
2001     ThrowPPException(appendedImage_->quiet());
2002   }
2003 
2004   // Adds the names of the artifacts of the image to the container.
2005   template <class Container>
artifactNames(Container * names_,const Image * image_)2006   void artifactNames(Container *names_,const Image* image_)
2007   {
2008     const char*
2009       name;
2010 
2011     names_->clear();
2012 
2013     MagickCore::ResetImageArtifactIterator(image_->constImage());
2014     name=MagickCore::GetNextImageArtifact(image_->constImage());
2015     while (name != (const char *) NULL)
2016     {
2017       names_->push_back(std::string(name));
2018       name=MagickCore::GetNextImageArtifact(image_->constImage());
2019     }
2020   }
2021 
2022   // Adds the names of the attributes of the image to the container.
2023   template <class Container>
attributeNames(Container * names_,const Image * image_)2024   void attributeNames(Container *names_,const Image* image_)
2025   {
2026     const char*
2027       name;
2028 
2029     names_->clear();
2030 
2031     MagickCore::ResetImagePropertyIterator(image_->constImage());
2032     name=MagickCore::GetNextImageProperty(image_->constImage());
2033     while (name != (const char *) NULL)
2034     {
2035       names_->push_back(std::string(name));
2036       name=MagickCore::GetNextImageProperty(image_->constImage());
2037     }
2038   }
2039 
2040   // Average a set of images.
2041   // All the input images must be the same size in pixels.
2042   template <class InputIterator>
averageImages(Image * averagedImage_,InputIterator first_,InputIterator last_)2043   void averageImages( Image *averagedImage_,
2044           InputIterator first_,
2045           InputIterator last_ ) {
2046     if (linkImages(first_,last_) == false)
2047       return;
2048     GetPPException;
2049     MagickCore::Image* image = MagickCore::EvaluateImages( first_->image(),
2050       MagickCore::MeanEvaluateOperator, exceptionInfo );
2051     unlinkImages( first_, last_ );
2052     averagedImage_->replaceImage( image );
2053     ThrowPPException(averagedImage_->quiet());
2054   }
2055 
2056   // Merge a sequence of images.
2057   // This is useful for GIF animation sequences that have page
2058   // offsets and disposal methods. A container to contain
2059   // the updated image sequence is passed via the coalescedImages_
2060   // option.
2061   template <class InputIterator, class Container >
coalesceImages(Container * coalescedImages_,InputIterator first_,InputIterator last_)2062   void coalesceImages(Container *coalescedImages_,InputIterator first_,
2063     InputIterator last_)
2064   {
2065     bool
2066       quiet;
2067 
2068     MagickCore::Image
2069       *images;
2070 
2071     if (linkImages(first_,last_) == false)
2072       return;
2073 
2074     GetPPException;
2075     quiet=first_->quiet();
2076     images=MagickCore::CoalesceImages(first_->image(),exceptionInfo);
2077 
2078     // Unlink image list
2079     unlinkImages(first_,last_);
2080 
2081     // Ensure container is empty
2082     coalescedImages_->clear();
2083 
2084     // Move images to container
2085     insertImages(coalescedImages_,images);
2086 
2087     // Report any error
2088     ThrowPPException(quiet);
2089   }
2090 
2091   // Return format coders matching specified conditions.
2092   //
2093   // The default (if no match terms are supplied) is to return all
2094   // available format coders.
2095   //
2096   // For example, to return all readable formats:
2097   //  list<CoderInfo> coderList;
2098   //  coderInfoList( &coderList, CoderInfo::TrueMatch, CoderInfo::AnyMatch, CoderInfo::AnyMatch)
2099   //
2100   template <class Container >
2101   void coderInfoList( Container *container_,
2102                       CoderInfo::MatchType isReadable_ = CoderInfo::AnyMatch,
2103                       CoderInfo::MatchType isWritable_ = CoderInfo::AnyMatch,
2104                       CoderInfo::MatchType isMultiFrame_ = CoderInfo::AnyMatch
2105                       ) {
2106     // Obtain first entry in MagickInfo list
2107     size_t number_formats;
2108     GetPPException;
2109     char **coder_list =
2110       MagickCore::GetMagickList( "*", &number_formats, exceptionInfo );
2111     if( !coder_list )
2112       {
2113         throwException(exceptionInfo);
2114         throwExceptionExplicit(MagickCore::MissingDelegateError,
2115                              "Coder array not returned!", 0 );
2116       }
2117 
2118     // Clear out container
2119     container_->clear();
2120 
2121     for ( ::ssize_t i=0; i < (::ssize_t) number_formats; i++)
2122       {
2123         const MagickCore::MagickInfo *magick_info =
2124           MagickCore::GetMagickInfo( coder_list[i], exceptionInfo );
2125         coder_list[i]=(char *)
2126           MagickCore::RelinquishMagickMemory( coder_list[i] );
2127 
2128         // Skip stealth coders
2129         if ( MagickCore::GetMagickStealth(magick_info) )
2130           continue;
2131 
2132         try {
2133           CoderInfo coderInfo( magick_info->name );
2134 
2135           // Test isReadable_
2136           if ( isReadable_ != CoderInfo::AnyMatch &&
2137                (( coderInfo.isReadable() && isReadable_ != CoderInfo::TrueMatch ) ||
2138                 ( !coderInfo.isReadable() && isReadable_ != CoderInfo::FalseMatch )) )
2139             continue;
2140 
2141           // Test isWritable_
2142           if ( isWritable_ != CoderInfo::AnyMatch &&
2143                (( coderInfo.isWritable() && isWritable_ != CoderInfo::TrueMatch ) ||
2144                 ( !coderInfo.isWritable() && isWritable_ != CoderInfo::FalseMatch )) )
2145             continue;
2146 
2147           // Test isMultiFrame_
2148           if ( isMultiFrame_ != CoderInfo::AnyMatch &&
2149                (( coderInfo.isMultiFrame() && isMultiFrame_ != CoderInfo::TrueMatch ) ||
2150                 ( !coderInfo.isMultiFrame() && isMultiFrame_ != CoderInfo::FalseMatch )) )
2151             continue;
2152 
2153           // Append matches to container
2154           container_->push_back( coderInfo );
2155         }
2156         // Intentionally ignore missing module errors
catch(Magick::ErrorModule &)2157         catch ( Magick::ErrorModule& )
2158           {
2159             continue;
2160           }
2161       }
2162     coder_list=(char **) MagickCore::RelinquishMagickMemory( coder_list );
2163     ThrowPPException(false);
2164   }
2165 
2166   //
2167   // Fill container with color histogram.
2168   // Entries are of type "std::pair<Color,size_t>".  Use the pair
2169   // "first" member to access the Color and the "second" member to access
2170   // the number of times the color occurs in the image.
2171   //
2172   // For example:
2173   //
2174   //  Using <map>:
2175   //
2176   //  Image image("image.miff");
2177   //  map<Color,size_t> histogram;
2178   //  colorHistogram( &histogram, image );
2179   //  std::map<Color,size_t>::const_iterator p=histogram.begin();
2180   //  while (p != histogram.end())
2181   //    {
2182   //      cout << setw(10) << (int)p->second << ": ("
2183   //           << setw(quantum_width) << (int)p->first.redQuantum() << ","
2184   //           << setw(quantum_width) << (int)p->first.greenQuantum() << ","
2185   //           << setw(quantum_width) << (int)p->first.blueQuantum() << ")"
2186   //           << endl;
2187   //      p++;
2188   //    }
2189   //
2190   //  Using <vector>:
2191   //
2192   //  Image image("image.miff");
2193   //  std::vector<std::pair<Color,size_t> > histogram;
2194   //  colorHistogram( &histogram, image );
2195   //  std::vector<std::pair<Color,size_t> >::const_iterator p=histogram.begin();
2196   //  while (p != histogram.end())
2197   //    {
2198   //      cout << setw(10) << (int)p->second << ": ("
2199   //           << setw(quantum_width) << (int)p->first.redQuantum() << ","
2200   //           << setw(quantum_width) << (int)p->first.greenQuantum() << ","
2201   //           << setw(quantum_width) << (int)p->first.blueQuantum() << ")"
2202   //           << endl;
2203   //      p++;
2204   //    }
2205 
2206   template <class Container >
colorHistogram(Container * histogram_,const Image image)2207   void colorHistogram( Container *histogram_, const Image image)
2208   {
2209     GetPPException;
2210 
2211     // Obtain histogram array
2212     size_t colors;
2213     MagickCore::PixelInfo *histogram_array =
2214       MagickCore::GetImageHistogram( image.constImage(), &colors, exceptionInfo );
2215     ThrowPPException(image.quiet());
2216 
2217     // Clear out container
2218     histogram_->clear();
2219 
2220     // Transfer histogram array to container
2221     for ( size_t i=0; i < colors; i++)
2222       {
2223         histogram_->insert( histogram_->end(), std::pair<const Color,size_t>
2224           ( Color(histogram_array[i]), (size_t) histogram_array[i].count) );
2225       }
2226 
2227     // Deallocate histogram array
2228     histogram_array=(MagickCore::PixelInfo *)
2229       MagickCore::RelinquishMagickMemory(histogram_array);
2230   }
2231 
2232   // Combines one or more images into a single image. The grayscale value of
2233   // the pixels of each image in the sequence is assigned in order to the
2234   // specified channels of the combined image. The typical ordering would be
2235   // image 1 => Red, 2 => Green, 3 => Blue, etc.
2236   template<class InputIterator >
combineImages(Image * combinedImage_,InputIterator first_,InputIterator last_,const ChannelType channel_,const ColorspaceType colorspace_)2237   void combineImages(Image *combinedImage_,InputIterator first_,
2238     InputIterator last_,const ChannelType channel_,
2239     const ColorspaceType colorspace_)
2240   {
2241     MagickCore::Image
2242       *image;
2243 
2244     std::vector<ChannelType>
2245       channelMask;
2246 
2247     if (linkImages(first_,last_) == false)
2248       return;
2249     GetPPException;
2250     channelMaskImages(first_,last_,&channelMask,channel_);
2251     image=CombineImages(first_->image(),colorspace_,exceptionInfo);
2252     restoreChannelMaskImages(first_,last_,&channelMask);
2253     unlinkImages(first_,last_);
2254     combinedImage_->replaceImage(image);
2255     ThrowPPException(combinedImage_->quiet());
2256   }
2257 
2258   template <class Container>
cropToTiles(Container * tiledImages_,const Image image_,const Geometry & geometry_)2259   void cropToTiles(Container *tiledImages_,const Image image_,
2260     const Geometry &geometry_)
2261   {
2262     GetPPException;
2263     MagickCore::Image* images=CropImageToTiles(image_.constImage(),
2264       static_cast<std::string>(geometry_).c_str(),exceptionInfo);
2265     tiledImages_->clear();
2266     insertImages(tiledImages_,images);
2267     ThrowPPException(image_.quiet());
2268   }
2269 
2270   // Break down an image sequence into constituent parts.  This is
2271   // useful for creating GIF or MNG animation sequences.
2272   template<class InputIterator,class Container>
deconstructImages(Container * deconstructedImages_,InputIterator first_,InputIterator last_)2273   void deconstructImages(Container *deconstructedImages_,
2274     InputIterator first_,InputIterator last_)
2275   {
2276     bool
2277       quiet;
2278 
2279     MagickCore::Image
2280       *images;
2281 
2282     if (linkImages(first_,last_) == false)
2283       return;
2284     GetPPException;
2285     quiet=first_->quiet();
2286     images=CompareImagesLayers(first_->image(),CompareAnyLayer,exceptionInfo);
2287     unlinkImages(first_,last_);
2288 
2289     deconstructedImages_->clear();
2290     insertImages(deconstructedImages_,images);
2291 
2292     ThrowPPException(quiet);
2293   }
2294 
2295   //
2296   // Display an image sequence
2297   //
2298   template <class InputIterator>
displayImages(InputIterator first_,InputIterator last_)2299   void displayImages(InputIterator first_,InputIterator last_)
2300   {
2301     if (linkImages(first_,last_) == false)
2302       return;
2303     GetPPException;
2304     MagickCore::DisplayImages(first_->imageInfo(),first_->image(),
2305       exceptionInfo);
2306     unlinkImages(first_,last_);
2307     ThrowPPException(first_->quiet());
2308   }
2309 
2310   // Applies a value to the image with an arithmetic, relational,
2311   // or logical operator to an image. Use these operations to lighten or darken
2312   // an image, to increase or decrease contrast in an image, or to produce the
2313   // "negative" of an image.
2314   template <class InputIterator >
evaluateImages(Image * evaluatedImage_,InputIterator first_,InputIterator last_,const MagickEvaluateOperator operator_)2315   void evaluateImages( Image *evaluatedImage_,
2316                        InputIterator first_,
2317                        InputIterator last_,
2318                        const MagickEvaluateOperator operator_ ) {
2319     if (linkImages(first_,last_) == false)
2320       return;
2321     GetPPException;
2322     MagickCore::Image* image = EvaluateImages( first_->image(), operator_, exceptionInfo );
2323     unlinkImages( first_, last_ );
2324     evaluatedImage_->replaceImage( image );
2325     ThrowPPException(evaluatedImage_->quiet());
2326   }
2327 
2328   // Merge a sequence of image frames which represent image layers.
2329   // This is useful for combining Photoshop layers into a single image.
2330   template <class InputIterator>
flattenImages(Image * flattendImage_,InputIterator first_,InputIterator last_)2331   void flattenImages( Image *flattendImage_,
2332           InputIterator first_,
2333           InputIterator last_ ) {
2334     if (linkImages(first_,last_) == false)
2335       return;
2336     GetPPException;
2337     MagickCore::Image* image = MagickCore::MergeImageLayers( first_->image(),
2338       FlattenLayer,exceptionInfo );
2339     unlinkImages( first_, last_ );
2340     flattendImage_->replaceImage( image );
2341     ThrowPPException(flattendImage_->quiet());
2342   }
2343 
2344   // Implements the discrete Fourier transform (DFT) of the image either as a
2345   // magnitude / phase or real / imaginary image pair.
2346   template <class Container >
forwardFourierTransformImage(Container * fourierImages_,const Image & image_)2347   void forwardFourierTransformImage( Container *fourierImages_,
2348     const Image &image_ ) {
2349     GetPPException;
2350 
2351     // Build image list
2352     MagickCore::Image* images = ForwardFourierTransformImage(
2353       image_.constImage(), MagickTrue, exceptionInfo);
2354 
2355     // Ensure container is empty
2356     fourierImages_->clear();
2357 
2358     // Move images to container
2359     insertImages( fourierImages_, images );
2360 
2361     // Report any error
2362     ThrowPPException(image_.quiet());
2363   }
2364   template <class Container >
forwardFourierTransformImage(Container * fourierImages_,const Image & image_,const bool magnitude_)2365   void forwardFourierTransformImage( Container *fourierImages_,
2366     const Image &image_, const bool magnitude_ ) {
2367     GetPPException;
2368 
2369     // Build image list
2370     MagickCore::Image* images = ForwardFourierTransformImage(
2371       image_.constImage(), magnitude_ == true ? MagickTrue : MagickFalse,
2372       exceptionInfo);
2373 
2374     // Ensure container is empty
2375     fourierImages_->clear();
2376 
2377     // Move images to container
2378     insertImages( fourierImages_, images );
2379 
2380     // Report any error
2381     ThrowPPException(image_.quiet());
2382   }
2383 
2384   // Applies a mathematical expression to a sequence of images.
2385   template <class InputIterator>
fxImages(Image * fxImage_,InputIterator first_,InputIterator last_,const std::string expression)2386   void fxImages(Image *fxImage_,InputIterator first_,InputIterator last_,
2387     const std::string expression)
2388   {
2389     MagickCore::Image
2390       *image;
2391 
2392     if (linkImages(first_,last_) == false)
2393       return;
2394     GetPPException;
2395     image=FxImage(first_->constImage(),expression.c_str(),exceptionInfo);
2396     unlinkImages(first_,last_);
2397     fxImage_->replaceImage(image);
2398     ThrowPPException(fxImage_->quiet());
2399   }
2400 
2401   // Replace the colors of a sequence of images with the closest color
2402   // from a reference image.
2403   // Set dither_ to true to enable dithering.  Set measureError_ to
2404   // true in order to evaluate quantization error.
2405   template<class InputIterator>
2406   void mapImages(InputIterator first_,InputIterator last_,
2407     const Image& mapImage_,bool dither_=false,bool measureError_=false)
2408   {
2409     MagickCore::Image
2410       *image;
2411 
2412     MagickCore::QuantizeInfo
2413       quantizeInfo;
2414 
2415     if (linkImages(first_,last_) == false)
2416       return;
2417     GetPPException;
2418     MagickCore::GetQuantizeInfo(&quantizeInfo);
2419     quantizeInfo.dither_method = dither_ ? MagickCore::RiemersmaDitherMethod :
2420       MagickCore::NoDitherMethod;
2421     MagickCore::RemapImages(&quantizeInfo,first_->image(),
2422       (mapImage_.isValid() ? mapImage_.constImage() :
2423       (const MagickCore::Image*) NULL),exceptionInfo);
2424     unlinkImages(first_,last_);
2425     if (exceptionInfo->severity != MagickCore::UndefinedException)
2426       {
2427         unlinkImages(first_,last_);
2428         throwException(exceptionInfo,mapImage_.quiet());
2429       }
2430 
2431     image=first_->image();
2432     while(image != (MagickCore::Image *) NULL)
2433       {
2434         // Calculate quantization error
2435         if (measureError_)
2436           {
2437             MagickCore::GetImageQuantizeError(image,exceptionInfo);
2438             if (exceptionInfo->severity > MagickCore::UndefinedException)
2439               {
2440                 unlinkImages(first_,last_);
2441                 throwException(exceptionInfo,mapImage_.quiet());
2442               }
2443           }
2444 
2445         // Update DirectClass representation of pixels
2446         MagickCore::SyncImage(image,exceptionInfo);
2447         if (exceptionInfo->severity > MagickCore::UndefinedException)
2448           {
2449             unlinkImages(first_,last_);
2450             throwException(exceptionInfo,mapImage_.quiet());
2451           }
2452 
2453         // Next image
2454         image=image->next;
2455       }
2456 
2457     unlinkImages(first_,last_);
2458     (void) MagickCore::DestroyExceptionInfo(exceptionInfo);
2459   }
2460 
2461   // Composes all the image layers from the current given
2462   // image onward to produce a single image of the merged layers.
2463   template <class InputIterator >
mergeImageLayers(Image * mergedImage_,InputIterator first_,InputIterator last_,const LayerMethod method_)2464   void mergeImageLayers( Image *mergedImage_,
2465                          InputIterator first_,
2466                          InputIterator last_,
2467                          const LayerMethod method_ ) {
2468     if (linkImages(first_,last_) == false)
2469       return;
2470     GetPPException;
2471     MagickCore::Image* image = MergeImageLayers( first_->image(), method_, exceptionInfo );
2472     unlinkImages( first_, last_ );
2473     mergedImage_->replaceImage( image );
2474     ThrowPPException(mergedImage_->quiet());
2475   }
2476 
2477   // Create a composite image by combining several separate images.
2478   template <class Container, class InputIterator>
montageImages(Container * montageImages_,InputIterator first_,InputIterator last_,const Montage & options_)2479   void montageImages(Container *montageImages_,InputIterator first_,
2480     InputIterator last_,const Montage &options_)
2481   {
2482     bool
2483       quiet;
2484 
2485     MagickCore::Image
2486       *images;
2487 
2488     MagickCore::MontageInfo
2489       *montageInfo;
2490 
2491     if (linkImages(first_,last_) == false)
2492       return;
2493 
2494     montageInfo=static_cast<MagickCore::MontageInfo*>(
2495       MagickCore::AcquireMagickMemory(sizeof(MagickCore::MontageInfo)));
2496 
2497     // Update montage options with those set in montageOpts_
2498     options_.updateMontageInfo(*montageInfo);
2499 
2500     // Update options which must transfer to image options
2501     if (options_.label().length() != 0)
2502       first_->label(options_.label());
2503 
2504     // Do montage
2505     GetPPException;
2506     quiet=first_->quiet();
2507     images=MagickCore::MontageImages(first_->image(),montageInfo,
2508       exceptionInfo);
2509 
2510     // Unlink linked image list
2511     unlinkImages(first_,last_);
2512 
2513     // Reset output container to pristine state
2514     montageImages_->clear();
2515 
2516     if (images != (MagickCore::Image *) NULL)
2517       insertImages(montageImages_,images);
2518 
2519     // Clean up any allocated data in montageInfo
2520     MagickCore::DestroyMontageInfo(montageInfo);
2521 
2522     // Report any montage error
2523     ThrowPPException(quiet);
2524 
2525     // Apply transparency to montage images
2526     if (montageImages_->size() > 0 && options_.transparentColor().isValid())
2527       for_each(montageImages_->begin(),montageImages_->end(),transparentImage(
2528         options_.transparentColor()));
2529   }
2530 
2531   // Morph a set of images
2532   template <class InputIterator, class Container >
morphImages(Container * morphedImages_,InputIterator first_,InputIterator last_,size_t frames_)2533   void morphImages(Container *morphedImages_,InputIterator first_,
2534     InputIterator last_,size_t frames_)
2535   {
2536     bool
2537       quiet;
2538 
2539     MagickCore::Image
2540       *images;
2541 
2542     if (linkImages(first_,last_) == false)
2543       return;
2544 
2545     GetPPException;
2546     quiet=first_->quiet();
2547     images=MagickCore::MorphImages(first_->image(),frames_,exceptionInfo);
2548 
2549     // Unlink image list
2550     unlinkImages(first_,last_);
2551 
2552     // Ensure container is empty
2553     morphedImages_->clear();
2554 
2555     // Move images to container
2556     insertImages(morphedImages_,images);
2557 
2558     // Report any error
2559     ThrowPPException(quiet);
2560   }
2561 
2562   // Inlay a number of images to form a single coherent picture.
2563   template <class InputIterator>
mosaicImages(Image * mosaicImage_,InputIterator first_,InputIterator last_)2564   void mosaicImages( Image *mosaicImage_,
2565          InputIterator first_,
2566          InputIterator last_ ) {
2567     if (linkImages(first_,last_) == false)
2568       return;
2569     GetPPException;
2570     MagickCore::Image* image = MagickCore::MergeImageLayers( first_->image(),
2571        MosaicLayer,exceptionInfo );
2572     unlinkImages( first_, last_ );
2573     mosaicImage_->replaceImage( image );
2574     ThrowPPException(mosaicImage_->quiet());
2575   }
2576 
2577   // Compares each image the GIF disposed forms of the previous image in
2578   // the sequence. From this it attempts to select the smallest cropped
2579   // image to replace each frame, while preserving the results of the
2580   // GIF animation.
2581   template <class InputIterator, class Container >
optimizeImageLayers(Container * optimizedImages_,InputIterator first_,InputIterator last_)2582   void optimizeImageLayers(Container *optimizedImages_,InputIterator first_,
2583     InputIterator last_)
2584   {
2585     bool
2586       quiet;
2587 
2588     MagickCore::Image
2589       *images;
2590 
2591     if (linkImages(first_,last_) == false)
2592       return;
2593 
2594     GetPPException;
2595     quiet=first_->quiet();
2596     images=OptimizeImageLayers(first_->image(),exceptionInfo);
2597 
2598     unlinkImages(first_,last_);
2599 
2600     optimizedImages_->clear();
2601 
2602     insertImages(optimizedImages_,images);
2603 
2604     ThrowPPException(quiet);
2605   }
2606 
2607   // optimizeImagePlusLayers is exactly as optimizeImageLayers, but may
2608   // also add or even remove extra frames in the animation, if it improves
2609   // the total number of pixels in the resulting GIF animation.
2610   template <class InputIterator, class Container >
optimizePlusImageLayers(Container * optimizedImages_,InputIterator first_,InputIterator last_)2611   void optimizePlusImageLayers(Container *optimizedImages_,
2612     InputIterator first_,InputIterator last_ )
2613   {
2614     bool
2615       quiet;
2616 
2617     MagickCore::Image
2618       *images;
2619 
2620     if (linkImages(first_,last_) == false)
2621       return;
2622 
2623     GetPPException;
2624     quiet=first_->quiet();
2625     images=OptimizePlusImageLayers(first_->image(),exceptionInfo);
2626 
2627     unlinkImages(first_,last_);
2628 
2629     optimizedImages_->clear();
2630 
2631     insertImages(optimizedImages_,images);
2632 
2633     ThrowPPException(quiet);
2634   }
2635 
2636   // Compares each image the GIF disposed forms of the previous image in the
2637   // sequence. Any pixel that does not change the displayed result is replaced
2638   // with transparency.
2639   template<class InputIterator>
optimizeTransparency(InputIterator first_,InputIterator last_)2640   void optimizeTransparency(InputIterator first_,InputIterator last_)
2641   {
2642     if (linkImages(first_,last_) == false)
2643       return;
2644     GetPPException;
2645     OptimizeImageTransparency(first_->image(),exceptionInfo);
2646     unlinkImages(first_,last_ );
2647 
2648     ThrowPPException(first_->quiet());
2649   }
2650 
2651   // Ping images into existing container (appending to container)
2652   template<class Container>
pingImages(Container * sequence_,const std::string & imageSpec_,ReadOptions & options)2653   void pingImages(Container *sequence_,const std::string &imageSpec_,
2654     ReadOptions &options)
2655   {
2656     options.ping(true);
2657     readImages(sequence_,imageSpec_,options);
2658   }
2659 
2660   template<class Container>
pingImages(Container * sequence_,const std::string & imageSpec_)2661   void pingImages(Container *sequence_,const std::string &imageSpec_)
2662   {
2663     ReadOptions options;
2664     pingImages(sequence_,imageSpec_,options);
2665   }
2666 
2667   template<class Container>
pingImages(Container * sequence_,const Blob & blob_,ReadOptions & options)2668   void pingImages(Container *sequence_,const Blob &blob_,ReadOptions &options)
2669   {
2670     options.ping(true);
2671     readImages(sequence_,blob_,options);
2672   }
2673 
2674   template<class Container>
pingImages(Container * sequence_,const Blob & blob_)2675   void pingImages(Container *sequence_,const Blob &blob_)
2676   {
2677     ReadOptions options;
2678     pingImages(sequence_,blob_,options);
2679   }
2680 
2681   // Adds the names of the profiles of the image to the container.
2682   template <class Container>
profileNames(Container * names_,const Image * image_)2683   void profileNames(Container *names_,const Image* image_)
2684   {
2685     const char*
2686       name;
2687 
2688     names_->clear();
2689 
2690     MagickCore::ResetImageProfileIterator(image_->constImage());
2691     name=MagickCore::GetNextImageProfile(image_->constImage());
2692     while (name != (const char *) NULL)
2693     {
2694       names_->push_back(std::string(name));
2695       name=MagickCore::GetNextImageProfile(image_->constImage());
2696     }
2697   }
2698 
2699   // Quantize colors in images using current quantization settings
2700   // Set measureError_ to true in order to measure quantization error
2701   template <class InputIterator>
2702   void quantizeImages(InputIterator first_,InputIterator last_,
2703     bool measureError_ = false)
2704   {
2705     if (linkImages(first_,last_) == false)
2706       return;
2707     GetPPException;
2708     MagickCore::QuantizeImages(first_->quantizeInfo(),first_->image(),
2709       exceptionInfo);
2710     unlinkImages(first_,last_);
2711 
2712     MagickCore::Image *image=first_->image();
2713     while (image != (MagickCore::Image *) NULL)
2714     {
2715       // Calculate quantization error
2716       if (measureError_)
2717         MagickCore::GetImageQuantizeError(image,exceptionInfo);
2718 
2719       // Update DirectClass representation of pixels
2720       MagickCore::SyncImage(image,exceptionInfo);
2721 
2722       image=image->next;
2723     }
2724     unlinkImages(first_,last_);
2725     ThrowPPException(first_->quiet());
2726   }
2727 
2728   // Read images into existing container (appending to container)
2729   template<class Container>
readImages(Container * sequence_,const std::string & imageSpec_,ReadOptions & options)2730   void readImages(Container *sequence_,const std::string &imageSpec_,
2731     ReadOptions &options)
2732   {
2733     MagickCore::Image
2734       *images;
2735 
2736     MagickCore::ImageInfo
2737       *imageInfo;
2738 
2739     imageInfo=options.imageInfo();
2740     imageSpec_.copy(imageInfo->filename,MagickPathExtent-1);
2741     imageInfo->filename[imageSpec_.length()] = 0;
2742     GetPPException;
2743     images=MagickCore::ReadImage(imageInfo,exceptionInfo);
2744     insertImages(sequence_,images);
2745     ThrowPPException(options.quiet());
2746   }
2747 
2748   template<class Container>
readImages(Container * sequence_,const std::string & imageSpec_)2749   void readImages(Container *sequence_,const std::string &imageSpec_)
2750   {
2751     ReadOptions options;
2752     readImages(sequence_,imageSpec_,options);
2753   }
2754 
2755   template<class Container>
readImages(Container * sequence_,const Blob & blob_,ReadOptions & options)2756   void readImages(Container *sequence_,const Blob &blob_,ReadOptions &options)
2757   {
2758     MagickCore::Image
2759       *images;
2760 
2761     GetPPException;
2762     images=MagickCore::BlobToImage(options.imageInfo(),blob_.data(),
2763       blob_.length(),exceptionInfo);
2764     insertImages(sequence_,images);
2765     ThrowPPException(options.quiet());
2766   }
2767 
2768   template<class Container>
readImages(Container * sequence_,const Blob & blob_)2769   void readImages(Container *sequence_,const Blob &blob_)
2770   {
2771     ReadOptions options;
2772     readImages(sequence_,blob_,options);
2773   }
2774 
2775   // Returns a separate grayscale image for each channel specified.
2776   template<class Container>
separateImages(Container * separatedImages_,Image & image_,const ChannelType channel_)2777   void separateImages(Container *separatedImages_,Image &image_,
2778     const ChannelType channel_)
2779   {
2780     MagickCore::ChannelType
2781       channel_mask;
2782 
2783     MagickCore::Image
2784       *images;
2785 
2786     GetPPException;
2787     channel_mask=MagickCore::SetImageChannelMask(image_.image(),channel_);
2788     images=SeparateImages(image_.constImage(),exceptionInfo);
2789     MagickCore::SetPixelChannelMask(image_.image(),channel_mask);
2790 
2791     separatedImages_->clear();
2792     insertImages(separatedImages_,images);
2793 
2794     ThrowPPException(image_.quiet());
2795   }
2796 
2797   // Smush images from list into single image in either horizontal or
2798   // vertical direction.
2799   template<class InputIterator>
2800   void smushImages(Image *smushedImage_,InputIterator first_,
2801     InputIterator last_,const ssize_t offset_,bool stack_=false)
2802   {
2803     MagickCore::Image
2804       *newImage;
2805 
2806     if (linkImages(first_,last_) == false)
2807       return;
2808     GetPPException;
2809     newImage=MagickCore::SmushImages(first_->constImage(),
2810       (MagickBooleanType) stack_,offset_,exceptionInfo);
2811     unlinkImages(first_,last_);
2812     smushedImage_->replaceImage(newImage);
2813     ThrowPPException(smushedImage_->quiet());
2814   }
2815 
2816   // Write Images
2817   template <class InputIterator>
2818   void writeImages( InputIterator first_,
2819         InputIterator last_,
2820         const std::string &imageSpec_,
2821         bool adjoin_ = true ) {
2822 
2823     if (linkImages(first_,last_) == false)
2824       return;
2825 
2826     first_->adjoin( adjoin_ );
2827 
2828     GetPPException;
2829     ::ssize_t errorStat = MagickCore::WriteImages( first_->constImageInfo(),
2830                                             first_->image(),
2831                                             imageSpec_.c_str(),
2832                                             exceptionInfo );
2833     unlinkImages( first_, last_ );
2834 
2835     if ( errorStat != false )
2836       {
2837         (void) MagickCore::DestroyExceptionInfo( exceptionInfo );
2838         return;
2839       }
2840 
2841     ThrowPPException(first_->quiet());
2842   }
2843   // Write images to BLOB
2844   template <class InputIterator>
2845   void writeImages( InputIterator first_,
2846         InputIterator last_,
2847         Blob *blob_,
2848         bool adjoin_ = true) {
2849     if (linkImages(first_,last_) == false)
2850       return;
2851 
2852     first_->adjoin( adjoin_ );
2853 
2854     GetPPException;
2855     size_t length = 2048; // Efficient size for small images
2856     void* data = MagickCore::ImagesToBlob( first_->imageInfo(),
2857            first_->image(),
2858            &length,
2859            exceptionInfo);
2860     blob_->updateNoCopy( data, length, Magick::Blob::MallocAllocator );
2861 
2862     unlinkImages( first_, last_ );
2863 
2864     ThrowPPException(first_->quiet());
2865   }
2866 
2867 } // namespace Magick
2868 
2869 #endif // Magick_STL_header
2870