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 " -alpha option on, activate, off, deactivate, set, opaque, copy\n"
109 " transparent, extract, background, or shape\n"
110 " -authenticate password\n"
111 " decipher image with this password\n"
112 " -background color background color\n"
113 " -channel type apply option to select image channels\n"
114 " -colorspace type alternate image colorspace\n"
115 " -compose operator set image composite operator\n"
116 " -compress type type of pixel compression when writing the image\n"
117 " -decipher filename convert cipher pixels to plain pixels\n"
118 " -define format:option\n"
119 " define one or more image format options\n"
120 " -density geometry horizontal and vertical density of the image\n"
121 " -depth value image depth\n"
122 " -dissimilarity-threshold value\n"
123 " maximum distortion for (sub)image match\n"
124 " -encipher filename convert plain pixels to cipher pixels\n"
125 " -extract geometry extract area from image\n"
126 " -format \"string\" output formatted image characteristics\n"
127 " -fuzz distance colors within this distance are considered equal\n"
128 " -gravity type horizontal and vertical text placement\n"
129 " -highlight-color color\n"
130 " emphasize pixel differences with this color\n"
131 " -identify identify the format and characteristics of the image\n"
132 " -interlace type type of image interlacing scheme\n"
133 " -limit type value pixel cache resource limit\n"
134 " -lowlight-color color\n"
135 " de-emphasize pixel differences with this color\n"
136 " -mask filename associate a mask with the image\n"
137 " -metric type measure differences between images with this metric\n"
138 " -monitor monitor progress\n"
139 " -passphrase filename get the passphrase from this file\n"
140 " -precision value maximum number of significant digits to print\n"
141 " -profile filename add, delete, or apply an image profile\n"
142 " -quality value JPEG/MIFF/PNG compression level\n"
143 " -quiet suppress all warning messages\n"
144 " -quantize colorspace reduce colors in this colorspace\n"
145 " -regard-warnings pay attention to warning messages\n"
146 " -repage geometry size and location of an image canvas\n"
147 " -respect-parentheses settings remain in effect until parenthesis boundary\n"
148 " -sampling-factor geometry\n"
149 " horizontal and vertical sampling factor\n"
150 " -seed value seed a new sequence of pseudo-random numbers\n"
151 " -set attribute value set an image attribute\n"
152 " -quality value JPEG/MIFF/PNG compression level\n"
153 " -similarity-threshold value\n"
154 " minimum distortion for (sub)image match\n"
155 " -size geometry width and height of image\n"
156 " -subimage-search search for subimage\n"
157 " -synchronize synchronize image to storage device\n"
158 " -taint declare the image as modified\n"
159 " -transparent-color color\n"
160 " transparent color\n"
161 " -type type image type\n"
162 " -verbose print detailed information about the image\n"
163 " -version print version information\n"
164 " -virtual-pixel method\n"
165 " virtual pixel access method",
167 " -delete indexes delete the image from the image sequence";
169 ListMagickVersion(stdout);
170 (void) printf(
"Usage: %s [options ...] image reconstruct difference\n",
172 (void) printf(
"\nImage Settings:\n");
173 (void) puts(settings);
174 (void) printf(
"\nImage Operators:\n");
175 (void) puts(operators);
176 (void) printf(
"\nImage Sequence Operators:\n");
177 (void) puts(sequence_operators);
178 (void) printf(
"\nImage Stack Operators:\n");
179 (void) puts(stack_operators);
180 (void) printf(
"\nMiscellaneous Options:\n");
181 (void) puts(miscellaneous);
183 "\nBy default, the image format of `file' is determined by its magic\n");
185 "number. To specify a particular image format, precede the filename\n");
187 "with an image format name and a colon (i.e. ps:image) or specify the\n");
189 "image type as the filename suffix (i.e. image.ps). Specify 'file' as\n");
190 (void) printf(
"'-' for standard input or output.\n");
194WandExport MagickBooleanType CompareImageCommand(ImageInfo *image_info,
195 int argc,
char **argv,
char **metadata,ExceptionInfo *exception)
197#define CompareEpsilon (1.0e-06)
198#define DefaultDissimilarityThreshold (1.0/MagickPI)
199#define DefaultSimilarityThreshold (-1.0)
200#define DestroyCompare() \
202 if (similarity_image != (Image *) NULL) \
203 similarity_image=DestroyImageList(similarity_image); \
204 if (difference_image != (Image *) NULL) \
205 difference_image=DestroyImageList(difference_image); \
206 DestroyImageStack(); \
207 for (i=0; i < (ssize_t) argc; i++) \
208 argv[i]=DestroyString(argv[i]); \
209 argv=(char **) RelinquishMagickMemory(argv); \
211#define ThrowCompareException(asperity,tag,option) \
213 if (exception->severity < (asperity)) \
214 (void) ThrowMagickException(exception,GetMagickModule(),asperity,tag, \
217 return(MagickFalse); \
219#define ThrowCompareInvalidArgumentException(option,argument) \
221 (void) ThrowMagickException(exception,GetMagickModule(),OptionError, \
222 "InvalidArgument","`%s': %s",option,argument); \
224 return(MagickFalse); \
238 dissimilarity_threshold,
241 similarity_threshold;
245 *image = (Image *) NULL,
253 image_stack[MaxImageStackDepth+1];
259 similar = MagickTrue,
281 assert(image_info != (ImageInfo *) NULL);
282 assert(image_info->signature == MagickCoreSignature);
283 assert(exception != (ExceptionInfo *) NULL);
284 if (IsEventLogging() != MagickFalse)
285 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
289 if ((LocaleCompare(
"version",option+1) == 0) ||
290 (LocaleCompare(
"-version",option+1) == 0))
292 ListMagickVersion(stdout);
298 (void) ThrowMagickException(exception,GetMagickModule(),OptionError,
299 "MissingArgument",
"%s",
"");
300 (void) CompareUsage();
303 restore_info=image_info;
304 channels=DefaultChannels;
305 difference_image=NewImageList();
306 similarity_image=NewImageList();
307 dissimilarity_threshold=DefaultDissimilarityThreshold;
308 similarity_threshold=DefaultSimilarityThreshold;
310 format=(
char *) NULL;
313 metric=UndefinedErrorMetric;
315 option=(
char *) NULL;
317 reconstruct_image=NewImageList();
318 respect_parenthesis=MagickFalse;
320 subimage_search=MagickFalse;
324 ReadCommandlLine(argc,&argv);
325 status=ExpandFilenames(&argc,&argv);
326 if (status == MagickFalse)
327 ThrowCompareException(ResourceLimitError,
"MemoryAllocationFailed",
328 GetExceptionMessage(errno));
329 for (i=1; i < (ssize_t) (argc-1); i++)
332 if (LocaleCompare(option,
"(") == 0)
334 FireImageStack(MagickTrue,MagickTrue,pend);
335 if (k == MaxImageStackDepth)
336 ThrowCompareException(OptionError,
"ParenthesisNestedTooDeeply",
341 if (LocaleCompare(option,
")") == 0)
343 FireImageStack(MagickTrue,MagickTrue,MagickTrue);
345 ThrowCompareException(OptionError,
"UnableToParseExpression",option);
349 if (IsCommandOption(option) == MagickFalse)
357 FireImageStack(MagickFalse,MagickFalse,pend);
359 if ((LocaleCompare(filename,
"--") == 0) && (i < (ssize_t) (argc-1)))
361 (void) SetImageOption(image_info,
"filename",filename);
362 (void) CopyMagickString(image_info->filename,filename,MaxTextExtent);
363 images=ReadImages(image_info,exception);
364 status&=(images != (Image *) NULL) &&
365 (exception->severity < ErrorException);
366 if (images == (Image *) NULL)
368 AppendImageStack(images);
371 pend=image != (Image *) NULL ? MagickTrue : MagickFalse;
376 if (LocaleCompare(
"alpha",option+1) == 0)
384 if (i == (ssize_t) argc)
385 ThrowCompareException(OptionError,
"MissingArgument",option);
386 type=ParseCommandOption(MagickAlphaOptions,MagickFalse,argv[i]);
388 ThrowCompareException(OptionError,
"UnrecognizedAlphaChannelType",
392 if (LocaleCompare(
"authenticate",option+1) == 0)
397 if (i == (ssize_t) argc)
398 ThrowCompareException(OptionError,
"MissingArgument",option);
401 ThrowCompareException(OptionError,
"UnrecognizedOption",option);
405 if (LocaleCompare(
"background",option+1) == 0)
410 if (i == (ssize_t) argc)
411 ThrowCompareException(OptionError,
"MissingArgument",option);
414 if (LocaleCompare(
"brightness-contrast",option+1) == 0)
417 if (i == (ssize_t) argc)
418 ThrowCompareException(OptionError,
"MissingArgument",option);
419 if (IsGeometry(argv[i]) == MagickFalse)
420 ThrowCompareInvalidArgumentException(option,argv[i]);
423 ThrowCompareException(OptionError,
"UnrecognizedOption",option);
427 if (LocaleCompare(
"cache",option+1) == 0)
432 if (i == (ssize_t) argc)
433 ThrowCompareException(OptionError,
"MissingArgument",option);
434 if (IsGeometry(argv[i]) == MagickFalse)
435 ThrowCompareInvalidArgumentException(option,argv[i]);
438 if (LocaleCompare(
"channel",option+1) == 0)
446 if (i == (ssize_t) argc)
447 ThrowCompareException(OptionError,
"MissingArgument",option);
448 channel=ParseChannelOption(argv[i]);
450 ThrowCompareException(OptionError,
"UnrecognizedChannelType",
452 channels=(ChannelType) channel;
455 if (LocaleCompare(
"colorspace",option+1) == 0)
463 if (i == (ssize_t) argc)
464 ThrowCompareException(OptionError,
"MissingArgument",option);
465 colorspace=ParseCommandOption(MagickColorspaceOptions,MagickFalse,
468 ThrowCompareException(OptionError,
"UnrecognizedColorspace",
472 if (LocaleCompare(
"compose",option+1) == 0)
480 if (i == (ssize_t) argc)
481 ThrowCompareException(OptionError,
"MissingArgument",option);
482 compose=ParseCommandOption(MagickComposeOptions,MagickFalse,
485 ThrowCompareException(OptionError,
"UnrecognizedComposeOperator",
489 if (LocaleCompare(
"compress",option+1) == 0)
497 if (i == (ssize_t) argc)
498 ThrowCompareException(OptionError,
"MissingArgument",option);
499 compress=ParseCommandOption(MagickCompressOptions,MagickFalse,
502 ThrowCompareException(OptionError,
"UnrecognizedImageCompression",
506 if (LocaleCompare(
"concurrent",option+1) == 0)
508 if (LocaleCompare(
"crop",option+1) == 0)
513 if (i == (ssize_t) argc)
514 ThrowCompareException(OptionError,
"MissingArgument",option);
515 if (IsGeometry(argv[i]) == MagickFalse)
516 ThrowCompareInvalidArgumentException(option,argv[i]);
519 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
523 if (LocaleCompare(
"debug",option+1) == 0)
531 if (i == (ssize_t) argc)
532 ThrowCompareException(OptionError,
"MissingArgument",option);
533 event_mask=SetLogEventMask(argv[i]);
534 if (event_mask == UndefinedEvents)
535 ThrowCompareException(OptionError,
"UnrecognizedEventType",
539 if (LocaleCompare(
"decipher",option+1) == 0)
544 if (i == (ssize_t) argc)
545 ThrowCompareException(OptionError,
"MissingArgument",option);
548 if (LocaleCompare(
"define",option+1) == 0)
551 if (i == (ssize_t) argc)
552 ThrowCompareException(OptionError,
"MissingArgument",option);
558 define=GetImageOption(image_info,argv[i]);
559 if (define == (
const char *) NULL)
560 ThrowCompareException(OptionError,
"NoSuchOption",argv[i]);
565 if (LocaleCompare(
"delete",option+1) == 0)
570 if (i == (ssize_t) argc)
571 ThrowCompareException(OptionError,
"MissingArgument",option);
572 if (IsSceneGeometry(argv[i],MagickFalse) == MagickFalse)
573 ThrowCompareInvalidArgumentException(option,argv[i]);
576 if (LocaleCompare(
"density",option+1) == 0)
581 if (i == (ssize_t) argc)
582 ThrowCompareException(OptionError,
"MissingArgument",option);
583 if (IsGeometry(argv[i]) == MagickFalse)
584 ThrowCompareInvalidArgumentException(option,argv[i]);
587 if (LocaleCompare(
"depth",option+1) == 0)
592 if (i == (ssize_t) argc)
593 ThrowCompareException(OptionError,
"MissingArgument",option);
594 if (IsGeometry(argv[i]) == MagickFalse)
595 ThrowCompareInvalidArgumentException(option,argv[i]);
598 if (LocaleCompare(
"dissimilarity-threshold",option+1) == 0)
603 if (i == (ssize_t) argc)
604 ThrowCompareException(OptionError,
"MissingArgument",option);
605 if (IsGeometry(argv[i]) == MagickFalse)
606 ThrowCompareInvalidArgumentException(option,argv[i]);
608 dissimilarity_threshold=DefaultDissimilarityThreshold;
610 dissimilarity_threshold=StringToDouble(argv[i],(
char **) NULL);
613 if (LocaleCompare(
"distort",option+1) == 0)
619 if (i == (ssize_t) argc)
620 ThrowCompareException(OptionError,
"MissingArgument",option);
621 op=ParseCommandOption(MagickDistortOptions,MagickFalse,argv[i]);
623 ThrowCompareException(OptionError,
"UnrecognizedDistortMethod",
626 if (i == (ssize_t) argc)
627 ThrowCompareException(OptionError,
"MissingArgument",option);
630 if (LocaleCompare(
"duration",option+1) == 0)
635 if (i == (ssize_t) argc)
636 ThrowCompareException(OptionError,
"MissingArgument",option);
637 if (IsGeometry(argv[i]) == MagickFalse)
638 ThrowCompareInvalidArgumentException(option,argv[i]);
641 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
645 if (LocaleCompare(
"encipher",option+1) == 0)
650 if (i == (ssize_t) argc)
651 ThrowCompareException(OptionError,
"MissingArgument",option);
654 if (LocaleCompare(
"extract",option+1) == 0)
659 if (i == (ssize_t) argc)
660 ThrowCompareException(OptionError,
"MissingArgument",option);
661 if (IsGeometry(argv[i]) == MagickFalse)
662 ThrowCompareInvalidArgumentException(option,argv[i]);
665 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
669 if (LocaleCompare(
"format",option+1) == 0)
674 if (i == (ssize_t) argc)
675 ThrowCompareException(OptionError,
"MissingArgument",option);
679 if (LocaleCompare(
"fuzz",option+1) == 0)
684 if (i == (ssize_t) argc)
685 ThrowCompareException(OptionError,
"MissingArgument",option);
686 if (IsGeometry(argv[i]) == MagickFalse)
687 ThrowCompareInvalidArgumentException(option,argv[i]);
690 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
694 if (LocaleCompare(
"gravity",option+1) == 0)
702 if (i == (ssize_t) argc)
703 ThrowCompareException(OptionError,
"MissingArgument",option);
704 gravity=ParseCommandOption(MagickGravityOptions,MagickFalse,
707 ThrowCompareException(OptionError,
"UnrecognizedGravityType",
711 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
715 if ((LocaleCompare(
"help",option+1) == 0) ||
716 (LocaleCompare(
"-help",option+1) == 0))
719 return(CompareUsage());
721 if (LocaleCompare(
"highlight-color",option+1) == 0)
726 if (i == (ssize_t) argc)
727 ThrowCompareException(OptionError,
"MissingArgument",option);
730 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
734 if (LocaleCompare(
"identify",option+1) == 0)
736 if (LocaleCompare(
"interlace",option+1) == 0)
744 if (i == (ssize_t) argc)
745 ThrowCompareException(OptionError,
"MissingArgument",option);
746 interlace=ParseCommandOption(MagickInterlaceOptions,MagickFalse,
749 ThrowCompareException(OptionError,
"UnrecognizedInterlaceType",
753 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
757 if (LocaleCompare(
"level",option+1) == 0)
760 if (i == (ssize_t) argc)
761 ThrowCompareException(OptionError,
"MissingArgument",option);
762 if (IsGeometry(argv[i]) == MagickFalse)
763 ThrowCompareInvalidArgumentException(option,argv[i]);
766 if (LocaleCompare(
"limit",option+1) == 0)
780 if (i == (ssize_t) argc)
781 ThrowCompareException(OptionError,
"MissingArgument",option);
782 resource=ParseCommandOption(MagickResourceOptions,MagickFalse,
785 ThrowCompareException(OptionError,
"UnrecognizedResourceType",
788 if (i == (ssize_t) argc)
789 ThrowCompareException(OptionError,
"MissingArgument",option);
790 value=StringToDouble(argv[i],&p);
792 if ((p == argv[i]) && (LocaleCompare(
"unlimited",argv[i]) != 0))
793 ThrowCompareInvalidArgumentException(option,argv[i]);
796 if (LocaleCompare(
"list",option+1) == 0)
804 if (i == (ssize_t) argc)
805 ThrowCompareException(OptionError,
"MissingArgument",option);
806 list=ParseCommandOption(MagickListOptions,MagickFalse,argv[i]);
808 ThrowCompareException(OptionError,
"UnrecognizedListType",argv[i]);
809 status=MogrifyImageInfo(image_info,(
int) (i-j+1),(
const char **)
812 return(status == 0 ? MagickFalse : MagickTrue);
814 if (LocaleCompare(
"log",option+1) == 0)
819 if ((i == (ssize_t) argc) || (strchr(argv[i],
'%') == (
char *) NULL))
820 ThrowCompareException(OptionError,
"MissingArgument",option);
823 if (LocaleCompare(
"lowlight-color",option+1) == 0)
828 if (i == (ssize_t) argc)
829 ThrowCompareException(OptionError,
"MissingArgument",option);
832 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
836 if (LocaleCompare(
"mask",option+1) == 0)
841 if (i == (ssize_t) argc)
842 ThrowCompareException(OptionError,
"MissingArgument",option);
845 if (LocaleCompare(
"matte",option+1) == 0)
847 if (LocaleCompare(
"metric",option+1) == 0)
855 if (i == (ssize_t) argc)
856 ThrowCompareException(OptionError,
"MissingArgument",option);
857 type=ParseCommandOption(MagickMetricOptions,MagickTrue,argv[i]);
859 ThrowCompareException(OptionError,
"UnrecognizedMetricType",
861 metric=(MetricType) type;
864 if (LocaleCompare(
"monitor",option+1) == 0)
866 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
870 if (LocaleCompare(
"precision",option+1) == 0)
875 if (i == (ssize_t) argc)
876 ThrowCompareException(OptionError,
"MissingArgument",option);
877 if (IsGeometry(argv[i]) == MagickFalse)
878 ThrowCompareInvalidArgumentException(option,argv[i]);
881 if (LocaleCompare(
"passphrase",option+1) == 0)
886 if (i == (ssize_t) argc)
887 ThrowCompareException(OptionError,
"MissingArgument",option);
890 if (LocaleCompare(
"profile",option+1) == 0)
893 if (i == (ssize_t) argc)
894 ThrowCompareException(OptionError,
"MissingArgument",option);
897 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
901 if (LocaleCompare(
"quality",option+1) == 0)
906 if (i == (ssize_t) argc)
907 ThrowCompareException(OptionError,
"MissingArgument",option);
908 if (IsGeometry(argv[i]) == MagickFalse)
909 ThrowCompareInvalidArgumentException(option,argv[i]);
912 if (LocaleCompare(
"quantize",option+1) == 0)
920 if (i == (ssize_t) argc)
921 ThrowCompareException(OptionError,
"MissingArgument",option);
922 colorspace=ParseCommandOption(MagickColorspaceOptions,
923 MagickFalse,argv[i]);
925 ThrowCompareException(OptionError,
"UnrecognizedColorspace",
929 if (LocaleCompare(
"quiet",option+1) == 0)
931 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
935 if (LocaleCompare(
"regard-warnings",option+1) == 0)
937 if (LocaleCompare(
"repage",option+1) == 0)
942 if (i == (ssize_t) argc)
943 ThrowCompareException(OptionError,
"MissingArgument",option);
944 if (IsGeometry(argv[i]) == MagickFalse)
945 ThrowCompareInvalidArgumentException(option,argv[i]);
948 if (LocaleCompare(
"resize",option+1) == 0)
953 if (i == (ssize_t) argc)
954 ThrowCompareException(OptionError,
"MissingArgument",option);
955 if (IsGeometry(argv[i]) == MagickFalse)
956 ThrowCompareInvalidArgumentException(option,argv[i]);
959 if (LocaleCompare(
"rotate",option+1) == 0)
962 if (i == (ssize_t) argc)
963 ThrowCompareException(OptionError,
"MissingArgument",option);
964 if (IsGeometry(argv[i]) == MagickFalse)
965 ThrowCompareInvalidArgumentException(option,argv[i]);
968 if (LocaleNCompare(
"respect-parentheses",option+1,17) == 0)
970 respect_parenthesis=(*option ==
'-') ? MagickTrue : MagickFalse;
973 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
977 if (LocaleCompare(
"sampling-factor",option+1) == 0)
982 if (i == (ssize_t) argc)
983 ThrowCompareException(OptionError,
"MissingArgument",option);
984 if (IsGeometry(argv[i]) == MagickFalse)
985 ThrowCompareInvalidArgumentException(option,argv[i]);
988 if (LocaleCompare(
"seed",option+1) == 0)
993 if (i == (ssize_t) argc)
994 ThrowCompareException(OptionError,
"MissingArgument",option);
995 if (IsGeometry(argv[i]) == MagickFalse)
996 ThrowCompareInvalidArgumentException(option,argv[i]);
999 if (LocaleCompare(
"separate",option+1) == 0)
1001 if (LocaleCompare(
"set",option+1) == 0)
1004 if (i == (ssize_t) argc)
1005 ThrowCompareException(OptionError,
"MissingArgument",option);
1009 if (i == (ssize_t) argc)
1010 ThrowCompareException(OptionError,
"MissingArgument",option);
1013 if (LocaleCompare(
"sigmoidal-contrast",option+1) == 0)
1016 if (i == (ssize_t) argc)
1017 ThrowCompareException(OptionError,
"MissingArgument",option);
1018 if (IsGeometry(argv[i]) == MagickFalse)
1019 ThrowCompareInvalidArgumentException(option,argv[i]);
1022 if (LocaleCompare(
"similarity-threshold",option+1) == 0)
1027 if (i == (ssize_t) argc)
1028 ThrowCompareException(OptionError,
"MissingArgument",option);
1029 if (IsGeometry(argv[i]) == MagickFalse)
1030 ThrowCompareInvalidArgumentException(option,argv[i]);
1032 similarity_threshold=DefaultSimilarityThreshold;
1034 similarity_threshold=StringToDouble(argv[i],(
char **) NULL);
1037 if (LocaleCompare(
"size",option+1) == 0)
1042 if (i == (ssize_t) argc)
1043 ThrowCompareException(OptionError,
"MissingArgument",option);
1044 if (IsGeometry(argv[i]) == MagickFalse)
1045 ThrowCompareInvalidArgumentException(option,argv[i]);
1048 if (LocaleCompare(
"subimage-search",option+1) == 0)
1052 subimage_search=MagickFalse;
1055 subimage_search=MagickTrue;
1058 if (LocaleCompare(
"synchronize",option+1) == 0)
1060 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1064 if (LocaleCompare(
"taint",option+1) == 0)
1066 if (LocaleCompare(
"transparent-color",option+1) == 0)
1071 if (i == (ssize_t) argc)
1072 ThrowCompareException(OptionError,
"MissingArgument",option);
1075 if (LocaleCompare(
"trim",option+1) == 0)
1077 if (LocaleCompare(
"type",option+1) == 0)
1085 if (i == (ssize_t) argc)
1086 ThrowCompareException(OptionError,
"MissingArgument",option);
1087 type=ParseCommandOption(MagickTypeOptions,MagickFalse,argv[i]);
1089 ThrowCompareException(OptionError,
"UnrecognizedImageType",
1093 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1097 if (LocaleCompare(
"verbose",option+1) == 0)
1099 if ((LocaleCompare(
"version",option+1) == 0) ||
1100 (LocaleCompare(
"-version",option+1) == 0))
1102 ListMagickVersion(stdout);
1105 if (LocaleCompare(
"virtual-pixel",option+1) == 0)
1113 if (i == (ssize_t) argc)
1114 ThrowCompareException(OptionError,
"MissingArgument",option);
1115 method=ParseCommandOption(MagickVirtualPixelOptions,MagickFalse,
1118 ThrowCompareException(OptionError,
1119 "UnrecognizedVirtualPixelMethod",argv[i]);
1122 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1126 if (LocaleCompare(
"write",option+1) == 0)
1129 if (i == (ssize_t) argc)
1130 ThrowCompareException(OptionError,
"MissingArgument",option);
1133 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1138 ThrowCompareException(OptionError,
"UnrecognizedOption",option)
1140 fire=(GetCommandOptionFlags(MagickCommandOptions,MagickFalse,option) &
1141 FireOptionFlag) == 0 ? MagickFalse : MagickTrue;
1142 if (fire != MagickFalse)
1143 FireImageStack(MagickTrue,MagickTrue,MagickTrue);
1146 ThrowCompareException(OptionError,
"UnbalancedParenthesis",argv[i]);
1147 if (i-- != (ssize_t) (argc-1))
1148 ThrowCompareException(OptionError,
"MissingAnImageFilename",argv[i]);
1149 if ((image == (Image *) NULL) || (GetImageListLength(image) < 2))
1150 ThrowCompareException(OptionError,
"MissingAnImageFilename",argv[i]);
1151 FinalizeImageSettings(image_info,image,MagickTrue);
1152 if ((image == (Image *) NULL) || (GetImageListLength(image) < 2))
1153 ThrowCompareException(OptionError,
"MissingAnImageFilename",argv[i]);
1154 image=GetImageFromList(image,0);
1155 reconstruct_image=GetImageFromList(image,1);
1158 if (subimage_search != MagickFalse)
1161 artifact[MaxTextExtent];
1163 (void) FormatLocaleString(artifact,MaxTextExtent,
"%g",
1164 similarity_threshold);
1165 (void) SetImageArtifact(image,
"compare:similarity-threshold",artifact);
1166 similarity_image=SimilarityMetricImage(image,reconstruct_image,metric,
1167 &offset,&similarity_metric,exception);
1168 if (similarity_metric > dissimilarity_threshold)
1169 (void) ThrowMagickException(exception,GetMagickModule(),ImageWarning,
1170 "ImagesTooDissimilar",
"`%s'",image->filename);
1172 if (similarity_image == (Image *) NULL)
1173 difference_image=CompareImageChannels(image,reconstruct_image,channels,
1174 metric,&distortion,exception);
1183 composite_image=CloneImage(image,0,0,MagickTrue,exception);
1184 if (composite_image == (Image *) NULL)
1185 difference_image=CompareImageChannels(image,reconstruct_image,
1186 channels,metric,&distortion,exception);
1195 (void) CompositeImage(composite_image,CopyCompositeOp,
1196 reconstruct_image,offset.x,offset.y);
1197 difference_image=CompareImageChannels(image,composite_image,
1198 channels,metric,&distortion,exception);
1199 if (difference_image != (Image *) NULL)
1201 difference_image->page.x=offset.x;
1202 difference_image->page.y=offset.y;
1204 composite_image=DestroyImage(composite_image);
1205 page.width=reconstruct_image->columns;
1206 page.height=reconstruct_image->rows;
1209 distort_image=CropImage(image,&page,exception);
1210 if (distort_image != (Image *) NULL)
1215 sans_image=CompareImageChannels(distort_image,reconstruct_image,
1216 channels,metric,&distortion,exception);
1217 distort_image=DestroyImage(distort_image);
1218 if (sans_image != (Image *) NULL)
1219 sans_image=DestroyImage(sans_image);
1222 if (difference_image != (Image *) NULL)
1224 AppendImageToList(&difference_image,similarity_image);
1225 similarity_image=(Image *) NULL;
1228 if (fabs(distortion) > CompareEpsilon)
1229 similar=MagickFalse;
1230 if (metric == NormalizedCrossCorrelationErrorMetric)
1232 distortion=1.0-distortion;
1233 similarity_metric=1.0-similarity_metric;
1235 if (difference_image == (Image *) NULL)
1239 if (image_info->verbose != MagickFalse)
1240 (void) IsImagesEqual(image,reconstruct_image);
1241 if (*difference_image->magick ==
'\0')
1242 (void) CopyMagickString(difference_image->magick,image->magick,
1244 if (image_info->verbose == MagickFalse)
1248 case FuzzErrorMetric:
1249 case MeanAbsoluteErrorMetric:
1250 case MeanSquaredErrorMetric:
1251 case PeakAbsoluteErrorMetric:
1252 case RootMeanSquaredErrorMetric:
1253 case AbsoluteErrorMetric:
1254 case NormalizedCrossCorrelationErrorMetric:
1255 case PerceptualHashErrorMetric:
1257 (void) FormatLocaleFile(stderr,
"%.*g (%.*g)",GetMagickPrecision(),
1258 (
double) QuantumRange*distortion,GetMagickPrecision(),
1262 case PeakSignalToNoiseRatioMetric:
1264 (void) FormatLocaleFile(stderr,
"%.*g (%.*g)",GetMagickPrecision(),
1265 (
double) QuantumRange*distortion,GetMagickPrecision(),
1269 case MeanErrorPerPixelMetric:
1271 (void) FormatLocaleFile(stderr,
"%.*g (%.*g, %.*g)",
1272 GetMagickPrecision(),distortion,
1273 GetMagickPrecision(),image->error.normalized_mean_error,
1274 GetMagickPrecision(),image->error.normalized_maximum_error);
1277 case UndefinedErrorMetric:
1280 if (subimage_search != MagickFalse)
1281 (void) FormatLocaleFile(stderr,
" @ %.20g,%.20g [%.*g]",(
double)
1282 difference_image->page.x,(
double) difference_image->page.y,
1283 GetMagickPrecision(),similarity_metric);
1288 *channel_distortion;
1290 channel_distortion=GetImageChannelDistortions(image,reconstruct_image,
1291 metric,&image->exception);
1292 (void) FormatLocaleFile(stderr,
"Image: %s\n",image->filename);
1293 if ((reconstruct_image->columns != image->columns) ||
1294 (reconstruct_image->rows != image->rows))
1295 (void) FormatLocaleFile(stderr,
"Offset: %.20g,%.20g\n",(
double)
1296 difference_image->page.x,(
double) difference_image->page.y);
1297 (void) FormatLocaleFile(stderr,
" Channel distortion: %s\n",
1298 CommandOptionToMnemonic(MagickMetricOptions,(ssize_t) metric));
1301 case FuzzErrorMetric:
1302 case MeanAbsoluteErrorMetric:
1303 case MeanSquaredErrorMetric:
1304 case PeakAbsoluteErrorMetric:
1305 case RootMeanSquaredErrorMetric:
1307 switch (image->colorspace)
1312 (void) FormatLocaleFile(stderr,
" red: %.*g (%.*g)\n",
1313 GetMagickPrecision(),(
double) QuantumRange*
1314 channel_distortion[RedChannel],GetMagickPrecision(),
1315 channel_distortion[RedChannel]);
1316 (void) FormatLocaleFile(stderr,
" green: %.*g (%.*g)\n",
1317 GetMagickPrecision(),(
double) QuantumRange*
1318 channel_distortion[GreenChannel],GetMagickPrecision(),
1319 channel_distortion[GreenChannel]);
1320 (void) FormatLocaleFile(stderr,
" blue: %.*g (%.*g)\n",
1321 GetMagickPrecision(),(
double) QuantumRange*
1322 channel_distortion[BlueChannel],GetMagickPrecision(),
1323 channel_distortion[BlueChannel]);
1324 if (image->matte != MagickFalse)
1325 (void) FormatLocaleFile(stderr,
" alpha: %.*g (%.*g)\n",
1326 GetMagickPrecision(),(
double) QuantumRange*
1327 channel_distortion[OpacityChannel],GetMagickPrecision(),
1328 channel_distortion[OpacityChannel]);
1331 case CMYKColorspace:
1333 (void) FormatLocaleFile(stderr,
" cyan: %.*g (%.*g)\n",
1334 GetMagickPrecision(),(
double) QuantumRange*
1335 channel_distortion[CyanChannel],GetMagickPrecision(),
1336 channel_distortion[CyanChannel]);
1337 (void) FormatLocaleFile(stderr,
" magenta: %.*g (%.*g)\n",
1338 GetMagickPrecision(),(
double) QuantumRange*
1339 channel_distortion[MagentaChannel],GetMagickPrecision(),
1340 channel_distortion[MagentaChannel]);
1341 (void) FormatLocaleFile(stderr,
" yellow: %.*g (%.*g)\n",
1342 GetMagickPrecision(),(
double) QuantumRange*
1343 channel_distortion[YellowChannel],GetMagickPrecision(),
1344 channel_distortion[YellowChannel]);
1345 (void) FormatLocaleFile(stderr,
" black: %.*g (%.*g)\n",
1346 GetMagickPrecision(),(
double) QuantumRange*
1347 channel_distortion[BlackChannel],GetMagickPrecision(),
1348 channel_distortion[BlackChannel]);
1349 if (image->matte != MagickFalse)
1350 (void) FormatLocaleFile(stderr,
" alpha: %.*g (%.*g)\n",
1351 GetMagickPrecision(),(
double) QuantumRange*
1352 channel_distortion[OpacityChannel],GetMagickPrecision(),
1353 channel_distortion[OpacityChannel]);
1356 case LinearGRAYColorspace:
1357 case GRAYColorspace:
1359 (void) FormatLocaleFile(stderr,
" gray: %.*g (%.*g)\n",
1360 GetMagickPrecision(),(
double) QuantumRange*
1361 channel_distortion[GrayChannel],GetMagickPrecision(),
1362 channel_distortion[GrayChannel]);
1363 if (image->matte != MagickFalse)
1364 (void) FormatLocaleFile(stderr,
" alpha: %.*g (%.*g)\n",
1365 GetMagickPrecision(),(
double) QuantumRange*
1366 channel_distortion[OpacityChannel],GetMagickPrecision(),
1367 channel_distortion[OpacityChannel]);
1371 (void) FormatLocaleFile(stderr,
" all: %.*g (%.*g)\n",
1372 GetMagickPrecision(),(
double) QuantumRange*
1373 channel_distortion[CompositeChannels],GetMagickPrecision(),
1374 channel_distortion[CompositeChannels]);
1377 case AbsoluteErrorMetric:
1378 case NormalizedCrossCorrelationErrorMetric:
1379 case PeakSignalToNoiseRatioMetric:
1380 case PerceptualHashErrorMetric:
1382 switch (image->colorspace)
1387 (void) FormatLocaleFile(stderr,
" red: %.*g\n",
1388 GetMagickPrecision(),channel_distortion[RedChannel]);
1389 (void) FormatLocaleFile(stderr,
" green: %.*g\n",
1390 GetMagickPrecision(),channel_distortion[GreenChannel]);
1391 (void) FormatLocaleFile(stderr,
" blue: %.*g\n",
1392 GetMagickPrecision(),channel_distortion[BlueChannel]);
1393 if (image->matte != MagickFalse)
1394 (void) FormatLocaleFile(stderr,
" alpha: %.*g\n",
1395 GetMagickPrecision(),channel_distortion[OpacityChannel]);
1398 case CMYKColorspace:
1400 (void) FormatLocaleFile(stderr,
" cyan: %.*g\n",
1401 GetMagickPrecision(),channel_distortion[CyanChannel]);
1402 (void) FormatLocaleFile(stderr,
" magenta: %.*g\n",
1403 GetMagickPrecision(),channel_distortion[MagentaChannel]);
1404 (void) FormatLocaleFile(stderr,
" yellow: %.*g\n",
1405 GetMagickPrecision(),channel_distortion[YellowChannel]);
1406 (void) FormatLocaleFile(stderr,
" black: %.*g\n",
1407 GetMagickPrecision(),channel_distortion[BlackChannel]);
1408 if (image->matte != MagickFalse)
1409 (void) FormatLocaleFile(stderr,
" alpha: %.*g\n",
1410 GetMagickPrecision(),channel_distortion[OpacityChannel]);
1413 case LinearGRAYColorspace:
1414 case GRAYColorspace:
1416 (void) FormatLocaleFile(stderr,
" gray: %.*g\n",
1417 GetMagickPrecision(),channel_distortion[GrayChannel]);
1418 if (image->matte != MagickFalse)
1419 (void) FormatLocaleFile(stderr,
" alpha: %.*g\n",
1420 GetMagickPrecision(),channel_distortion[OpacityChannel]);
1424 (void) FormatLocaleFile(stderr,
" all: %.*g\n",
1425 GetMagickPrecision(),channel_distortion[CompositeChannels]);
1428 case MeanErrorPerPixelMetric:
1430 (void) FormatLocaleFile(stderr,
" %.*g (%.*g, %.*g)\n",
1431 GetMagickPrecision(),channel_distortion[CompositeChannels],
1432 GetMagickPrecision(),image->error.normalized_mean_error,
1433 GetMagickPrecision(),image->error.normalized_maximum_error);
1436 case UndefinedErrorMetric:
1439 channel_distortion=(
double *) RelinquishMagickMemory(
1440 channel_distortion);
1441 if (subimage_search != MagickFalse)
1443 (void) FormatLocaleFile(stderr,
" Offset: %.20g,%.20g\n",
1444 (
double) difference_image->page.x,(
double)
1445 difference_image->page.y);
1446 (void) FormatLocaleFile(stderr,
" Similarity metric: %.*g\n",
1447 GetMagickPrecision(),similarity_metric);
1450 (void) ResetImagePage(difference_image,
"0x0+0+0");
1451 if (difference_image->next != (Image *) NULL)
1452 (void) ResetImagePage(difference_image->next,
"0x0+0+0");
1453 status&=WriteImages(image_info,difference_image,argv[argc-1],exception);
1454 if ((metadata != (
char **) NULL) && (format != (
char *) NULL))
1459 text=InterpretImageProperties(image_info,difference_image,format);
1460 InheritException(exception,&image->exception);
1461 if (text == (
char *) NULL)
1462 ThrowCompareException(ResourceLimitError,
"MemoryAllocationFailed",
1463 GetExceptionMessage(errno));
1464 (void) ConcatenateString(&(*metadata),text);
1465 text=DestroyString(text);
1467 difference_image=DestroyImageList(difference_image);
1470 image_info=restore_info;
1471 if (similar == MagickFalse)
1472 (void) SetImageOption(image_info,
"compare:dissimilar",
"true");
1473 return(status != 0 ? MagickTrue : MagickFalse);