44#include "wand/studio.h"
45#include "wand/MagickWand.h"
46#include "wand/mogrify-private.h"
47#include "magick/image-private.h"
48#include "magick/string-private.h"
84static MagickBooleanType CompareUsage(
void)
88 " -debug events display copious debugging information\n"
89 " -help print program options\n"
90 " -list type print a list of supported option arguments\n"
91 " -log format format of debugging information",
93 " -brightness-contrast geometry\n"
94 " improve brightness / contrast of the image\n"
95 " -distort method args\n"
96 " distort images according to given method and args\n"
97 " -level value adjust the level of image contrast\n"
98 " -resize geometry resize the image\n"
99 " -rotate degrees apply Paeth rotation to the image\n"
100 " -sigmoidal-contrast geometry\n"
101 " increase the contrast without saturating highlights or\n"
102 " -trim trim image edges",
103 sequence_operators[] =
104 " -crop geometry cut out a rectangular region of the image\n"
105 " -separate separate an image channel into a grayscale image\n"
106 " -write filename write images to this file",
108 " -adjoin join images into a single multi-image file\n"
109 " -alpha option on, activate, off, deactivate, set, opaque, copy\n"
110 " transparent, extract, background, or shape\n"
111 " -authenticate password\n"
112 " decipher image with this password\n"
113 " -background color background color\n"
114 " -channel type apply option to select image channels\n"
115 " -colorspace type alternate image colorspace\n"
116 " -compose operator set image composite operator\n"
117 " -compress type type of pixel compression when writing the image\n"
118 " -decipher filename convert cipher pixels to plain pixels\n"
119 " -define format:option\n"
120 " define one or more image format options\n"
121 " -density geometry horizontal and vertical density of the image\n"
122 " -depth value image depth\n"
123 " -dissimilarity-threshold value\n"
124 " maximum distortion for (sub)image match\n"
125 " -encipher filename convert plain pixels to cipher pixels\n"
126 " -extract geometry extract area from image\n"
127 " -format \"string\" output formatted image characteristics\n"
128 " -fuzz distance colors within this distance are considered equal\n"
129 " -gravity type horizontal and vertical text placement\n"
130 " -highlight-color color\n"
131 " emphasize pixel differences with this color\n"
132 " -identify identify the format and characteristics of the image\n"
133 " -interlace type type of image interlacing scheme\n"
134 " -limit type value pixel cache resource limit\n"
135 " -lowlight-color color\n"
136 " de-emphasize pixel differences with this color\n"
137 " -mask filename associate a mask with the image\n"
138 " -metric type measure differences between images with this metric\n"
139 " -monitor monitor progress\n"
140 " -passphrase filename get the passphrase from this file\n"
141 " -precision value maximum number of significant digits to print\n"
142 " -profile filename add, delete, or apply an image profile\n"
143 " -quality value JPEG/MIFF/PNG compression level\n"
144 " -quiet suppress all warning messages\n"
145 " -quantize colorspace reduce colors in this colorspace\n"
146 " -regard-warnings pay attention to warning messages\n"
147 " -repage geometry size and location of an image canvas\n"
148 " -respect-parentheses settings remain in effect until parenthesis boundary\n"
149 " -sampling-factor geometry\n"
150 " horizontal and vertical sampling factor\n"
151 " -seed value seed a new sequence of pseudo-random numbers\n"
152 " -set attribute value set an image attribute\n"
153 " -quality value JPEG/MIFF/PNG compression level\n"
154 " -similarity-threshold value\n"
155 " minimum distortion for (sub)image match\n"
156 " -size geometry width and height of image\n"
157 " -subimage-search search for subimage\n"
158 " -synchronize synchronize image to storage device\n"
159 " -taint declare the image as modified\n"
160 " -transparent-color color\n"
161 " transparent color\n"
162 " -type type image type\n"
163 " -verbose print detailed information about the image\n"
164 " -version print version information\n"
165 " -virtual-pixel method\n"
166 " virtual pixel access method",
168 " -delete indexes delete the image from the image sequence";
170 ListMagickVersion(stdout);
171 (void) printf(
"Usage: %s [options ...] image reconstruct difference\n",
173 (void) printf(
"\nImage Settings:\n");
174 (void) puts(settings);
175 (void) printf(
"\nImage Operators:\n");
176 (void) puts(operators);
177 (void) printf(
"\nImage Sequence Operators:\n");
178 (void) puts(sequence_operators);
179 (void) printf(
"\nImage Stack Operators:\n");
180 (void) puts(stack_operators);
181 (void) printf(
"\nMiscellaneous Options:\n");
182 (void) puts(miscellaneous);
184 "\nBy default, the image format of `file' is determined by its magic\n");
186 "number. To specify a particular image format, precede the filename\n");
188 "with an image format name and a colon (i.e. ps:image) or specify the\n");
190 "image type as the filename suffix (i.e. image.ps). Specify 'file' as\n");
191 (void) printf(
"'-' for standard input or output.\n");
195WandExport MagickBooleanType CompareImageCommand(ImageInfo *image_info,
196 int argc,
char **argv,
char **metadata,ExceptionInfo *exception)
198#define CompareEpsilon (1.0e-06)
199#define DefaultDissimilarityThreshold (1.0/MagickPI)
200#define DefaultSimilarityThreshold (-1.0)
201#define DestroyCompare() \
203 if (similarity_image != (Image *) NULL) \
204 similarity_image=DestroyImageList(similarity_image); \
205 if (difference_image != (Image *) NULL) \
206 difference_image=DestroyImageList(difference_image); \
207 DestroyImageStack(); \
208 for (i=0; i < (ssize_t) argc; i++) \
209 argv[i]=DestroyString(argv[i]); \
210 argv=(char **) RelinquishMagickMemory(argv); \
212#define ThrowCompareException(asperity,tag,option) \
214 if (exception->severity < (asperity)) \
215 (void) ThrowMagickException(exception,GetMagickModule(),asperity,tag, \
218 return(MagickFalse); \
220#define ThrowCompareInvalidArgumentException(option,argument) \
222 (void) ThrowMagickException(exception,GetMagickModule(),OptionError, \
223 "InvalidArgument","`%s': %s",option,argument); \
225 return(MagickFalse); \
239 dissimilarity_threshold,
242 similarity_threshold;
246 *image = (Image *) NULL,
254 image_stack[MaxImageStackDepth+1];
260 similar = MagickTrue,
282 assert(image_info != (ImageInfo *) NULL);
283 assert(image_info->signature == MagickCoreSignature);
284 assert(exception != (ExceptionInfo *) NULL);
285 if (IsEventLogging() != MagickFalse)
286 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
290 if ((LocaleCompare(
"version",option+1) == 0) ||
291 (LocaleCompare(
"-version",option+1) == 0))
293 ListMagickVersion(stdout);
299 (void) ThrowMagickException(exception,GetMagickModule(),OptionError,
300 "MissingArgument",
"%s",
"");
301 (void) CompareUsage();
304 restore_info=image_info;
305 channels=DefaultChannels;
306 difference_image=NewImageList();
307 similarity_image=NewImageList();
308 dissimilarity_threshold=DefaultDissimilarityThreshold;
309 similarity_threshold=DefaultSimilarityThreshold;
311 format=(
char *) NULL;
314 metric=UndefinedErrorMetric;
316 option=(
char *) NULL;
318 reconstruct_image=NewImageList();
319 respect_parenthesis=MagickFalse;
321 subimage_search=MagickFalse;
325 ReadCommandlLine(argc,&argv);
326 status=ExpandFilenames(&argc,&argv);
327 if (status == MagickFalse)
328 ThrowCompareException(ResourceLimitError,
"MemoryAllocationFailed",
329 GetExceptionMessage(errno));
330 for (i=1; i < (ssize_t) (argc-1); i++)
333 if (LocaleCompare(option,
"(") == 0)
335 FireImageStack(MagickTrue,MagickTrue,pend);
336 if (k == MaxImageStackDepth)
337 ThrowCompareException(OptionError,
"ParenthesisNestedTooDeeply",
342 if (LocaleCompare(option,
")") == 0)
344 FireImageStack(MagickTrue,MagickTrue,MagickTrue);
346 ThrowCompareException(OptionError,
"UnableToParseExpression",option);
350 if (IsCommandOption(option) == MagickFalse)
358 FireImageStack(MagickFalse,MagickFalse,pend);
360 if ((LocaleCompare(filename,
"--") == 0) && (i < (ssize_t) (argc-1)))
362 (void) SetImageOption(image_info,
"filename",filename);
363 (void) CopyMagickString(image_info->filename,filename,MaxTextExtent);
364 images=ReadImages(image_info,exception);
365 status&=(images != (Image *) NULL) &&
366 (exception->severity < ErrorException);
367 if (images == (Image *) NULL)
369 AppendImageStack(images);
372 pend=image != (Image *) NULL ? MagickTrue : MagickFalse;
377 if (LocaleCompare(
"adjoin",option+1) == 0)
379 if (LocaleCompare(
"alpha",option+1) == 0)
387 if (i == (ssize_t) argc)
388 ThrowCompareException(OptionError,
"MissingArgument",option);
389 type=ParseCommandOption(MagickAlphaOptions,MagickFalse,argv[i]);
391 ThrowCompareException(OptionError,
"UnrecognizedAlphaChannelType",
395 if (LocaleCompare(
"authenticate",option+1) == 0)
400 if (i == (ssize_t) argc)
401 ThrowCompareException(OptionError,
"MissingArgument",option);
404 ThrowCompareException(OptionError,
"UnrecognizedOption",option);
408 if (LocaleCompare(
"background",option+1) == 0)
413 if (i == (ssize_t) argc)
414 ThrowCompareException(OptionError,
"MissingArgument",option);
417 if (LocaleCompare(
"brightness-contrast",option+1) == 0)
420 if (i == (ssize_t) argc)
421 ThrowCompareException(OptionError,
"MissingArgument",option);
422 if (IsGeometry(argv[i]) == MagickFalse)
423 ThrowCompareInvalidArgumentException(option,argv[i]);
426 ThrowCompareException(OptionError,
"UnrecognizedOption",option);
430 if (LocaleCompare(
"cache",option+1) == 0)
435 if (i == (ssize_t) argc)
436 ThrowCompareException(OptionError,
"MissingArgument",option);
437 if (IsGeometry(argv[i]) == MagickFalse)
438 ThrowCompareInvalidArgumentException(option,argv[i]);
441 if (LocaleCompare(
"channel",option+1) == 0)
449 if (i == (ssize_t) argc)
450 ThrowCompareException(OptionError,
"MissingArgument",option);
451 channel=ParseChannelOption(argv[i]);
453 ThrowCompareException(OptionError,
"UnrecognizedChannelType",
455 channels=(ChannelType) channel;
458 if (LocaleCompare(
"colorspace",option+1) == 0)
466 if (i == (ssize_t) argc)
467 ThrowCompareException(OptionError,
"MissingArgument",option);
468 colorspace=ParseCommandOption(MagickColorspaceOptions,MagickFalse,
471 ThrowCompareException(OptionError,
"UnrecognizedColorspace",
475 if (LocaleCompare(
"compose",option+1) == 0)
483 if (i == (ssize_t) argc)
484 ThrowCompareException(OptionError,
"MissingArgument",option);
485 compose=ParseCommandOption(MagickComposeOptions,MagickFalse,
488 ThrowCompareException(OptionError,
"UnrecognizedComposeOperator",
492 if (LocaleCompare(
"compress",option+1) == 0)
500 if (i == (ssize_t) argc)
501 ThrowCompareException(OptionError,
"MissingArgument",option);
502 compress=ParseCommandOption(MagickCompressOptions,MagickFalse,
505 ThrowCompareException(OptionError,
"UnrecognizedImageCompression",
509 if (LocaleCompare(
"concurrent",option+1) == 0)
511 if (LocaleCompare(
"crop",option+1) == 0)
516 if (i == (ssize_t) argc)
517 ThrowCompareException(OptionError,
"MissingArgument",option);
518 if (IsGeometry(argv[i]) == MagickFalse)
519 ThrowCompareInvalidArgumentException(option,argv[i]);
522 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
526 if (LocaleCompare(
"debug",option+1) == 0)
534 if (i == (ssize_t) argc)
535 ThrowCompareException(OptionError,
"MissingArgument",option);
536 event_mask=SetLogEventMask(argv[i]);
537 if (event_mask == UndefinedEvents)
538 ThrowCompareException(OptionError,
"UnrecognizedEventType",
542 if (LocaleCompare(
"decipher",option+1) == 0)
547 if (i == (ssize_t) argc)
548 ThrowCompareException(OptionError,
"MissingArgument",option);
551 if (LocaleCompare(
"define",option+1) == 0)
554 if (i == (ssize_t) argc)
555 ThrowCompareException(OptionError,
"MissingArgument",option);
561 define=GetImageOption(image_info,argv[i]);
562 if (define == (
const char *) NULL)
563 ThrowCompareException(OptionError,
"NoSuchOption",argv[i]);
568 if (LocaleCompare(
"delete",option+1) == 0)
573 if (i == (ssize_t) argc)
574 ThrowCompareException(OptionError,
"MissingArgument",option);
575 if (IsSceneGeometry(argv[i],MagickFalse) == MagickFalse)
576 ThrowCompareInvalidArgumentException(option,argv[i]);
579 if (LocaleCompare(
"density",option+1) == 0)
584 if (i == (ssize_t) argc)
585 ThrowCompareException(OptionError,
"MissingArgument",option);
586 if (IsGeometry(argv[i]) == MagickFalse)
587 ThrowCompareInvalidArgumentException(option,argv[i]);
590 if (LocaleCompare(
"depth",option+1) == 0)
595 if (i == (ssize_t) argc)
596 ThrowCompareException(OptionError,
"MissingArgument",option);
597 if (IsGeometry(argv[i]) == MagickFalse)
598 ThrowCompareInvalidArgumentException(option,argv[i]);
601 if (LocaleCompare(
"dissimilarity-threshold",option+1) == 0)
606 if (i == (ssize_t) argc)
607 ThrowCompareException(OptionError,
"MissingArgument",option);
608 if (IsGeometry(argv[i]) == MagickFalse)
609 ThrowCompareInvalidArgumentException(option,argv[i]);
611 dissimilarity_threshold=DefaultDissimilarityThreshold;
613 dissimilarity_threshold=StringToDouble(argv[i],(
char **) NULL);
616 if (LocaleCompare(
"distort",option+1) == 0)
622 if (i == (ssize_t) argc)
623 ThrowCompareException(OptionError,
"MissingArgument",option);
624 op=ParseCommandOption(MagickDistortOptions,MagickFalse,argv[i]);
626 ThrowCompareException(OptionError,
"UnrecognizedDistortMethod",
629 if (i == (ssize_t) argc)
630 ThrowCompareException(OptionError,
"MissingArgument",option);
633 if (LocaleCompare(
"duration",option+1) == 0)
638 if (i == (ssize_t) argc)
639 ThrowCompareException(OptionError,
"MissingArgument",option);
640 if (IsGeometry(argv[i]) == MagickFalse)
641 ThrowCompareInvalidArgumentException(option,argv[i]);
644 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
648 if (LocaleCompare(
"encipher",option+1) == 0)
653 if (i == (ssize_t) argc)
654 ThrowCompareException(OptionError,
"MissingArgument",option);
657 if (LocaleCompare(
"extract",option+1) == 0)
662 if (i == (ssize_t) argc)
663 ThrowCompareException(OptionError,
"MissingArgument",option);
664 if (IsGeometry(argv[i]) == MagickFalse)
665 ThrowCompareInvalidArgumentException(option,argv[i]);
668 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
672 if (LocaleCompare(
"format",option+1) == 0)
677 if (i == (ssize_t) argc)
678 ThrowCompareException(OptionError,
"MissingArgument",option);
682 if (LocaleCompare(
"fuzz",option+1) == 0)
687 if (i == (ssize_t) argc)
688 ThrowCompareException(OptionError,
"MissingArgument",option);
689 if (IsGeometry(argv[i]) == MagickFalse)
690 ThrowCompareInvalidArgumentException(option,argv[i]);
693 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
697 if (LocaleCompare(
"gravity",option+1) == 0)
705 if (i == (ssize_t) argc)
706 ThrowCompareException(OptionError,
"MissingArgument",option);
707 gravity=ParseCommandOption(MagickGravityOptions,MagickFalse,
710 ThrowCompareException(OptionError,
"UnrecognizedGravityType",
714 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
718 if ((LocaleCompare(
"help",option+1) == 0) ||
719 (LocaleCompare(
"-help",option+1) == 0))
722 return(CompareUsage());
724 if (LocaleCompare(
"highlight-color",option+1) == 0)
729 if (i == (ssize_t) argc)
730 ThrowCompareException(OptionError,
"MissingArgument",option);
733 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
737 if (LocaleCompare(
"identify",option+1) == 0)
739 if (LocaleCompare(
"interlace",option+1) == 0)
747 if (i == (ssize_t) argc)
748 ThrowCompareException(OptionError,
"MissingArgument",option);
749 interlace=ParseCommandOption(MagickInterlaceOptions,MagickFalse,
752 ThrowCompareException(OptionError,
"UnrecognizedInterlaceType",
756 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
760 if (LocaleCompare(
"level",option+1) == 0)
763 if (i == (ssize_t) argc)
764 ThrowCompareException(OptionError,
"MissingArgument",option);
765 if (IsGeometry(argv[i]) == MagickFalse)
766 ThrowCompareInvalidArgumentException(option,argv[i]);
769 if (LocaleCompare(
"limit",option+1) == 0)
783 if (i == (ssize_t) argc)
784 ThrowCompareException(OptionError,
"MissingArgument",option);
785 resource=ParseCommandOption(MagickResourceOptions,MagickFalse,
788 ThrowCompareException(OptionError,
"UnrecognizedResourceType",
791 if (i == (ssize_t) argc)
792 ThrowCompareException(OptionError,
"MissingArgument",option);
793 value=StringToDouble(argv[i],&p);
795 if ((p == argv[i]) && (LocaleCompare(
"unlimited",argv[i]) != 0))
796 ThrowCompareInvalidArgumentException(option,argv[i]);
799 if (LocaleCompare(
"list",option+1) == 0)
807 if (i == (ssize_t) argc)
808 ThrowCompareException(OptionError,
"MissingArgument",option);
809 list=ParseCommandOption(MagickListOptions,MagickFalse,argv[i]);
811 ThrowCompareException(OptionError,
"UnrecognizedListType",argv[i]);
812 status=MogrifyImageInfo(image_info,(
int) (i-j+1),(
const char **)
815 return(status == 0 ? MagickFalse : MagickTrue);
817 if (LocaleCompare(
"log",option+1) == 0)
822 if ((i == (ssize_t) argc) || (strchr(argv[i],
'%') == (
char *) NULL))
823 ThrowCompareException(OptionError,
"MissingArgument",option);
826 if (LocaleCompare(
"lowlight-color",option+1) == 0)
831 if (i == (ssize_t) argc)
832 ThrowCompareException(OptionError,
"MissingArgument",option);
835 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
839 if (LocaleCompare(
"mask",option+1) == 0)
844 if (i == (ssize_t) argc)
845 ThrowCompareException(OptionError,
"MissingArgument",option);
848 if (LocaleCompare(
"matte",option+1) == 0)
850 if (LocaleCompare(
"metric",option+1) == 0)
858 if (i == (ssize_t) argc)
859 ThrowCompareException(OptionError,
"MissingArgument",option);
860 type=ParseCommandOption(MagickMetricOptions,MagickTrue,argv[i]);
862 ThrowCompareException(OptionError,
"UnrecognizedMetricType",
864 metric=(MetricType) type;
867 if (LocaleCompare(
"monitor",option+1) == 0)
869 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
873 if (LocaleCompare(
"precision",option+1) == 0)
878 if (i == (ssize_t) argc)
879 ThrowCompareException(OptionError,
"MissingArgument",option);
880 if (IsGeometry(argv[i]) == MagickFalse)
881 ThrowCompareInvalidArgumentException(option,argv[i]);
884 if (LocaleCompare(
"passphrase",option+1) == 0)
889 if (i == (ssize_t) argc)
890 ThrowCompareException(OptionError,
"MissingArgument",option);
893 if (LocaleCompare(
"profile",option+1) == 0)
896 if (i == (ssize_t) argc)
897 ThrowCompareException(OptionError,
"MissingArgument",option);
900 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
904 if (LocaleCompare(
"quality",option+1) == 0)
909 if (i == (ssize_t) argc)
910 ThrowCompareException(OptionError,
"MissingArgument",option);
911 if (IsGeometry(argv[i]) == MagickFalse)
912 ThrowCompareInvalidArgumentException(option,argv[i]);
915 if (LocaleCompare(
"quantize",option+1) == 0)
923 if (i == (ssize_t) argc)
924 ThrowCompareException(OptionError,
"MissingArgument",option);
925 colorspace=ParseCommandOption(MagickColorspaceOptions,
926 MagickFalse,argv[i]);
928 ThrowCompareException(OptionError,
"UnrecognizedColorspace",
932 if (LocaleCompare(
"quiet",option+1) == 0)
934 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
938 if (LocaleCompare(
"regard-warnings",option+1) == 0)
940 if (LocaleCompare(
"repage",option+1) == 0)
945 if (i == (ssize_t) argc)
946 ThrowCompareException(OptionError,
"MissingArgument",option);
947 if (IsGeometry(argv[i]) == MagickFalse)
948 ThrowCompareInvalidArgumentException(option,argv[i]);
951 if (LocaleCompare(
"resize",option+1) == 0)
956 if (i == (ssize_t) argc)
957 ThrowCompareException(OptionError,
"MissingArgument",option);
958 if (IsGeometry(argv[i]) == MagickFalse)
959 ThrowCompareInvalidArgumentException(option,argv[i]);
962 if (LocaleCompare(
"rotate",option+1) == 0)
965 if (i == (ssize_t) argc)
966 ThrowCompareException(OptionError,
"MissingArgument",option);
967 if (IsGeometry(argv[i]) == MagickFalse)
968 ThrowCompareInvalidArgumentException(option,argv[i]);
971 if (LocaleNCompare(
"respect-parentheses",option+1,17) == 0)
973 respect_parenthesis=(*option ==
'-') ? MagickTrue : MagickFalse;
976 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
980 if (LocaleCompare(
"sampling-factor",option+1) == 0)
985 if (i == (ssize_t) argc)
986 ThrowCompareException(OptionError,
"MissingArgument",option);
987 if (IsGeometry(argv[i]) == MagickFalse)
988 ThrowCompareInvalidArgumentException(option,argv[i]);
991 if (LocaleCompare(
"seed",option+1) == 0)
996 if (i == (ssize_t) argc)
997 ThrowCompareException(OptionError,
"MissingArgument",option);
998 if (IsGeometry(argv[i]) == MagickFalse)
999 ThrowCompareInvalidArgumentException(option,argv[i]);
1002 if (LocaleCompare(
"separate",option+1) == 0)
1004 if (LocaleCompare(
"set",option+1) == 0)
1007 if (i == (ssize_t) argc)
1008 ThrowCompareException(OptionError,
"MissingArgument",option);
1012 if (i == (ssize_t) argc)
1013 ThrowCompareException(OptionError,
"MissingArgument",option);
1016 if (LocaleCompare(
"sigmoidal-contrast",option+1) == 0)
1019 if (i == (ssize_t) argc)
1020 ThrowCompareException(OptionError,
"MissingArgument",option);
1021 if (IsGeometry(argv[i]) == MagickFalse)
1022 ThrowCompareInvalidArgumentException(option,argv[i]);
1025 if (LocaleCompare(
"similarity-threshold",option+1) == 0)
1030 if (i == (ssize_t) argc)
1031 ThrowCompareException(OptionError,
"MissingArgument",option);
1032 if (IsGeometry(argv[i]) == MagickFalse)
1033 ThrowCompareInvalidArgumentException(option,argv[i]);
1035 similarity_threshold=DefaultSimilarityThreshold;
1037 similarity_threshold=StringToDouble(argv[i],(
char **) NULL);
1040 if (LocaleCompare(
"size",option+1) == 0)
1045 if (i == (ssize_t) argc)
1046 ThrowCompareException(OptionError,
"MissingArgument",option);
1047 if (IsGeometry(argv[i]) == MagickFalse)
1048 ThrowCompareInvalidArgumentException(option,argv[i]);
1051 if (LocaleCompare(
"subimage-search",option+1) == 0)
1055 subimage_search=MagickFalse;
1058 subimage_search=MagickTrue;
1061 if (LocaleCompare(
"synchronize",option+1) == 0)
1063 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1067 if (LocaleCompare(
"taint",option+1) == 0)
1069 if (LocaleCompare(
"transparent-color",option+1) == 0)
1074 if (i == (ssize_t) argc)
1075 ThrowCompareException(OptionError,
"MissingArgument",option);
1078 if (LocaleCompare(
"trim",option+1) == 0)
1080 if (LocaleCompare(
"type",option+1) == 0)
1088 if (i == (ssize_t) argc)
1089 ThrowCompareException(OptionError,
"MissingArgument",option);
1090 type=ParseCommandOption(MagickTypeOptions,MagickFalse,argv[i]);
1092 ThrowCompareException(OptionError,
"UnrecognizedImageType",
1096 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1100 if (LocaleCompare(
"verbose",option+1) == 0)
1102 if ((LocaleCompare(
"version",option+1) == 0) ||
1103 (LocaleCompare(
"-version",option+1) == 0))
1105 ListMagickVersion(stdout);
1108 if (LocaleCompare(
"virtual-pixel",option+1) == 0)
1116 if (i == (ssize_t) argc)
1117 ThrowCompareException(OptionError,
"MissingArgument",option);
1118 method=ParseCommandOption(MagickVirtualPixelOptions,MagickFalse,
1121 ThrowCompareException(OptionError,
1122 "UnrecognizedVirtualPixelMethod",argv[i]);
1125 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1129 if (LocaleCompare(
"write",option+1) == 0)
1132 if (i == (ssize_t) argc)
1133 ThrowCompareException(OptionError,
"MissingArgument",option);
1136 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1141 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1143 fire=(GetCommandOptionFlags(MagickCommandOptions,MagickFalse,option) &
1144 FireOptionFlag) == 0 ? MagickFalse : MagickTrue;
1145 if (fire != MagickFalse)
1146 FireImageStack(MagickTrue,MagickTrue,MagickTrue);
1149 ThrowCompareException(OptionError,
"UnbalancedParenthesis",argv[i]);
1150 if (i-- != (ssize_t) (argc-1))
1151 ThrowCompareException(OptionError,
"MissingAnImageFilename",argv[i]);
1152 if ((image == (Image *) NULL) || (GetImageListLength(image) < 2))
1153 ThrowCompareException(OptionError,
"MissingAnImageFilename",argv[i]);
1154 FinalizeImageSettings(image_info,image,MagickTrue);
1155 if ((image == (Image *) NULL) || (GetImageListLength(image) < 2))
1156 ThrowCompareException(OptionError,
"MissingAnImageFilename",argv[i]);
1157 image=GetImageFromList(image,0);
1158 reconstruct_image=GetImageFromList(image,1);
1161 if (subimage_search != MagickFalse)
1164 artifact[MaxTextExtent];
1166 (void) FormatLocaleString(artifact,MaxTextExtent,
"%g",
1167 similarity_threshold);
1168 (void) SetImageArtifact(image,
"compare:similarity-threshold",artifact);
1169 similarity_image=SimilarityMetricImage(image,reconstruct_image,metric,
1170 &offset,&similarity_metric,exception);
1171 if (similarity_metric > dissimilarity_threshold)
1172 (void) ThrowMagickException(exception,GetMagickModule(),ImageWarning,
1173 "ImagesTooDissimilar",
"`%s'",image->filename);
1175 if (similarity_image == (Image *) NULL)
1176 difference_image=CompareImageChannels(image,reconstruct_image,channels,
1177 metric,&distortion,exception);
1186 composite_image=CloneImage(image,0,0,MagickTrue,exception);
1187 if (composite_image == (Image *) NULL)
1188 difference_image=CompareImageChannels(image,reconstruct_image,
1189 channels,metric,&distortion,exception);
1198 (void) CompositeImage(composite_image,CopyCompositeOp,
1199 reconstruct_image,offset.x,offset.y);
1200 difference_image=CompareImageChannels(image,composite_image,
1201 channels,metric,&distortion,exception);
1202 if (difference_image != (Image *) NULL)
1204 difference_image->page.x=offset.x;
1205 difference_image->page.y=offset.y;
1207 composite_image=DestroyImage(composite_image);
1208 page.width=reconstruct_image->columns;
1209 page.height=reconstruct_image->rows;
1212 distort_image=CropImage(image,&page,exception);
1213 if (distort_image != (Image *) NULL)
1218 sans_image=CompareImageChannels(distort_image,reconstruct_image,
1219 channels,metric,&distortion,exception);
1220 distort_image=DestroyImage(distort_image);
1221 if (sans_image != (Image *) NULL)
1222 sans_image=DestroyImage(sans_image);
1225 if (difference_image != (Image *) NULL)
1227 AppendImageToList(&difference_image,similarity_image);
1228 similarity_image=(Image *) NULL;
1231 if (fabs(distortion) > CompareEpsilon)
1232 similar=MagickFalse;
1235 case NormalizedCrossCorrelationErrorMetric:
1237 distortion=1.0-distortion;
1238 similarity_metric=1.0-similarity_metric;
1240 case PeakSignalToNoiseRatioMetric:
1242 distortion=fabs(distortion);
1243 similarity_metric=fabs(similarity_metric);
1248 if (difference_image == (Image *) NULL)
1252 if (image_info->verbose != MagickFalse)
1253 (void) IsImagesEqual(image,reconstruct_image);
1254 if (*difference_image->magick ==
'\0')
1255 (void) CopyMagickString(difference_image->magick,image->magick,
1257 if (image_info->verbose == MagickFalse)
1261 case FuzzErrorMetric:
1262 case MeanAbsoluteErrorMetric:
1263 case MeanSquaredErrorMetric:
1264 case PeakAbsoluteErrorMetric:
1265 case RootMeanSquaredErrorMetric:
1266 case AbsoluteErrorMetric:
1267 case NormalizedCrossCorrelationErrorMetric:
1268 case PerceptualHashErrorMetric:
1270 (void) FormatLocaleFile(stderr,
"%.*g (%.*g)",GetMagickPrecision(),
1271 (
double) QuantumRange*distortion,GetMagickPrecision(),
1275 case PeakSignalToNoiseRatioMetric:
1277 (void) FormatLocaleFile(stderr,
"%.*g (%.*g)",GetMagickPrecision(),
1278 48.1647*distortion,GetMagickPrecision(),distortion);
1281 case MeanErrorPerPixelMetric:
1283 (void) FormatLocaleFile(stderr,
"%.*g (%.*g, %.*g)",
1284 GetMagickPrecision(),distortion,
1285 GetMagickPrecision(),image->error.normalized_mean_error,
1286 GetMagickPrecision(),image->error.normalized_maximum_error);
1289 case UndefinedErrorMetric:
1292 if (subimage_search != MagickFalse)
1293 (void) FormatLocaleFile(stderr,
" @ %.20g,%.20g [%.*g]",
1294 (
double) offset.x,(
double) offset.y,GetMagickPrecision(),
1300 *channel_distortion;
1302 channel_distortion=GetImageChannelDistortions(image,reconstruct_image,
1303 metric,&image->exception);
1304 (void) FormatLocaleFile(stderr,
"Image: %s\n",image->filename);
1305 if ((reconstruct_image->columns != image->columns) ||
1306 (reconstruct_image->rows != image->rows))
1307 (void) FormatLocaleFile(stderr,
"Offset: %.20g,%.20g\n",(
double)
1308 difference_image->page.x,(
double) difference_image->page.y);
1309 (void) FormatLocaleFile(stderr,
" Channel distortion: %s\n",
1310 CommandOptionToMnemonic(MagickMetricOptions,(ssize_t) metric));
1313 case FuzzErrorMetric:
1314 case MeanAbsoluteErrorMetric:
1315 case MeanSquaredErrorMetric:
1316 case PeakAbsoluteErrorMetric:
1317 case RootMeanSquaredErrorMetric:
1319 switch (image->colorspace)
1324 (void) FormatLocaleFile(stderr,
" red: %.*g (%.*g)\n",
1325 GetMagickPrecision(),(
double) QuantumRange*
1326 channel_distortion[RedChannel],GetMagickPrecision(),
1327 channel_distortion[RedChannel]);
1328 (void) FormatLocaleFile(stderr,
" green: %.*g (%.*g)\n",
1329 GetMagickPrecision(),(
double) QuantumRange*
1330 channel_distortion[GreenChannel],GetMagickPrecision(),
1331 channel_distortion[GreenChannel]);
1332 (void) FormatLocaleFile(stderr,
" blue: %.*g (%.*g)\n",
1333 GetMagickPrecision(),(
double) QuantumRange*
1334 channel_distortion[BlueChannel],GetMagickPrecision(),
1335 channel_distortion[BlueChannel]);
1336 if (image->matte != MagickFalse)
1337 (void) FormatLocaleFile(stderr,
" alpha: %.*g (%.*g)\n",
1338 GetMagickPrecision(),(
double) QuantumRange*
1339 channel_distortion[OpacityChannel],GetMagickPrecision(),
1340 channel_distortion[OpacityChannel]);
1343 case CMYKColorspace:
1345 (void) FormatLocaleFile(stderr,
" cyan: %.*g (%.*g)\n",
1346 GetMagickPrecision(),(
double) QuantumRange*
1347 channel_distortion[CyanChannel],GetMagickPrecision(),
1348 channel_distortion[CyanChannel]);
1349 (void) FormatLocaleFile(stderr,
" magenta: %.*g (%.*g)\n",
1350 GetMagickPrecision(),(
double) QuantumRange*
1351 channel_distortion[MagentaChannel],GetMagickPrecision(),
1352 channel_distortion[MagentaChannel]);
1353 (void) FormatLocaleFile(stderr,
" yellow: %.*g (%.*g)\n",
1354 GetMagickPrecision(),(
double) QuantumRange*
1355 channel_distortion[YellowChannel],GetMagickPrecision(),
1356 channel_distortion[YellowChannel]);
1357 (void) FormatLocaleFile(stderr,
" black: %.*g (%.*g)\n",
1358 GetMagickPrecision(),(
double) QuantumRange*
1359 channel_distortion[BlackChannel],GetMagickPrecision(),
1360 channel_distortion[BlackChannel]);
1361 if (image->matte != MagickFalse)
1362 (void) FormatLocaleFile(stderr,
" alpha: %.*g (%.*g)\n",
1363 GetMagickPrecision(),(
double) QuantumRange*
1364 channel_distortion[OpacityChannel],GetMagickPrecision(),
1365 channel_distortion[OpacityChannel]);
1368 case LinearGRAYColorspace:
1369 case GRAYColorspace:
1371 (void) FormatLocaleFile(stderr,
" gray: %.*g (%.*g)\n",
1372 GetMagickPrecision(),(
double) QuantumRange*
1373 channel_distortion[GrayChannel],GetMagickPrecision(),
1374 channel_distortion[GrayChannel]);
1375 if (image->matte != MagickFalse)
1376 (void) FormatLocaleFile(stderr,
" alpha: %.*g (%.*g)\n",
1377 GetMagickPrecision(),(
double) QuantumRange*
1378 channel_distortion[OpacityChannel],GetMagickPrecision(),
1379 channel_distortion[OpacityChannel]);
1383 (void) FormatLocaleFile(stderr,
" all: %.*g (%.*g)\n",
1384 GetMagickPrecision(),(
double) QuantumRange*
1385 channel_distortion[CompositeChannels],GetMagickPrecision(),
1386 channel_distortion[CompositeChannels]);
1389 case AbsoluteErrorMetric:
1390 case NormalizedCrossCorrelationErrorMetric:
1391 case PeakSignalToNoiseRatioMetric:
1392 case PerceptualHashErrorMetric:
1394 switch (image->colorspace)
1399 (void) FormatLocaleFile(stderr,
" red: %.*g\n",
1400 GetMagickPrecision(),channel_distortion[RedChannel]);
1401 (void) FormatLocaleFile(stderr,
" green: %.*g\n",
1402 GetMagickPrecision(),channel_distortion[GreenChannel]);
1403 (void) FormatLocaleFile(stderr,
" blue: %.*g\n",
1404 GetMagickPrecision(),channel_distortion[BlueChannel]);
1405 if (image->matte != MagickFalse)
1406 (void) FormatLocaleFile(stderr,
" alpha: %.*g\n",
1407 GetMagickPrecision(),channel_distortion[OpacityChannel]);
1410 case CMYKColorspace:
1412 (void) FormatLocaleFile(stderr,
" cyan: %.*g\n",
1413 GetMagickPrecision(),channel_distortion[CyanChannel]);
1414 (void) FormatLocaleFile(stderr,
" magenta: %.*g\n",
1415 GetMagickPrecision(),channel_distortion[MagentaChannel]);
1416 (void) FormatLocaleFile(stderr,
" yellow: %.*g\n",
1417 GetMagickPrecision(),channel_distortion[YellowChannel]);
1418 (void) FormatLocaleFile(stderr,
" black: %.*g\n",
1419 GetMagickPrecision(),channel_distortion[BlackChannel]);
1420 if (image->matte != MagickFalse)
1421 (void) FormatLocaleFile(stderr,
" alpha: %.*g\n",
1422 GetMagickPrecision(),channel_distortion[OpacityChannel]);
1425 case LinearGRAYColorspace:
1426 case GRAYColorspace:
1428 (void) FormatLocaleFile(stderr,
" gray: %.*g\n",
1429 GetMagickPrecision(),channel_distortion[GrayChannel]);
1430 if (image->matte != MagickFalse)
1431 (void) FormatLocaleFile(stderr,
" alpha: %.*g\n",
1432 GetMagickPrecision(),channel_distortion[OpacityChannel]);
1436 (void) FormatLocaleFile(stderr,
" all: %.*g\n",
1437 GetMagickPrecision(),channel_distortion[CompositeChannels]);
1440 case MeanErrorPerPixelMetric:
1442 (void) FormatLocaleFile(stderr,
" %.*g (%.*g, %.*g)\n",
1443 GetMagickPrecision(),channel_distortion[CompositeChannels],
1444 GetMagickPrecision(),image->error.normalized_mean_error,
1445 GetMagickPrecision(),image->error.normalized_maximum_error);
1448 case UndefinedErrorMetric:
1451 channel_distortion=(
double *) RelinquishMagickMemory(
1452 channel_distortion);
1453 if (subimage_search != MagickFalse)
1455 (void) FormatLocaleFile(stderr,
" Offset: %.20g,%.20g\n",
1456 (
double) difference_image->page.x,(
double)
1457 difference_image->page.y);
1458 (void) FormatLocaleFile(stderr,
" Similarity metric: %.*g\n",
1459 GetMagickPrecision(),similarity_metric);
1462 (void) ResetImagePage(difference_image,
"0x0+0+0");
1463 if (difference_image->next != (Image *) NULL)
1464 (void) ResetImagePage(difference_image->next,
"0x0+0+0");
1465 status&=WriteImages(image_info,difference_image,argv[argc-1],exception);
1466 if ((metadata != (
char **) NULL) && (format != (
char *) NULL))
1471 text=InterpretImageProperties(image_info,difference_image,format);
1472 InheritException(exception,&image->exception);
1473 if (text == (
char *) NULL)
1474 ThrowCompareException(ResourceLimitError,
"MemoryAllocationFailed",
1475 GetExceptionMessage(errno));
1476 (void) ConcatenateString(&(*metadata),text);
1477 text=DestroyString(text);
1479 difference_image=DestroyImageList(difference_image);
1482 image_info=restore_info;
1483 if (similar == MagickFalse)
1484 (void) SetImageOption(image_info,
"compare:dissimilar",
"true");
1485 return(status != 0 ? MagickTrue : MagickFalse);