43#include "MagickCore/studio.h"
44#include "MagickCore/artifact.h"
45#include "MagickCore/cache.h"
46#include "MagickCore/cache-private.h"
47#include "MagickCore/cache-view.h"
48#include "MagickCore/channel.h"
49#include "MagickCore/client.h"
50#include "MagickCore/color.h"
51#include "MagickCore/color-private.h"
52#include "MagickCore/colorspace.h"
53#include "MagickCore/colorspace-private.h"
54#include "MagickCore/composite.h"
55#include "MagickCore/composite-private.h"
56#include "MagickCore/constitute.h"
57#include "MagickCore/draw.h"
58#include "MagickCore/exception-private.h"
59#include "MagickCore/fx.h"
60#include "MagickCore/gem.h"
61#include "MagickCore/geometry.h"
62#include "MagickCore/image.h"
63#include "MagickCore/image-private.h"
64#include "MagickCore/list.h"
65#include "MagickCore/log.h"
66#include "MagickCore/memory_.h"
67#include "MagickCore/monitor.h"
68#include "MagickCore/monitor-private.h"
69#include "MagickCore/morphology.h"
70#include "MagickCore/option.h"
71#include "MagickCore/pixel-accessor.h"
72#include "MagickCore/property.h"
73#include "MagickCore/quantum.h"
74#include "MagickCore/resample.h"
75#include "MagickCore/resource_.h"
76#include "MagickCore/string_.h"
77#include "MagickCore/string-private.h"
78#include "MagickCore/thread-private.h"
79#include "MagickCore/threshold.h"
80#include "MagickCore/token.h"
81#include "MagickCore/transform.h"
82#include "MagickCore/utility.h"
83#include "MagickCore/utility-private.h"
84#include "MagickCore/version.h"
185static Image *BlendConvolveImage(
const Image *image,
const char *kernel,
198 kernel_info=AcquireKernelInfo(kernel,exception);
200 return((
Image *) NULL);
201 clone_image=CloneImage(image,0,0,MagickTrue,exception);
202 if (clone_image == (
Image *) NULL)
204 kernel_info=DestroyKernelInfo(kernel_info);
205 return((
Image *) NULL);
207 (void) SetImageAlphaChannel(clone_image,OffAlphaChannel,exception);
208 convolve_image=ConvolveImage(clone_image,kernel_info,exception);
209 kernel_info=DestroyKernelInfo(kernel_info);
210 clone_image=DestroyImage(clone_image);
211 return(convolve_image);
214static Image *BlendMagnitudeImage(
const Image *dx_image,
const Image *dy_image,
234 magnitude_image=CloneImage(dx_image,0,0,MagickTrue,exception);
235 if (magnitude_image == (
Image *) NULL)
236 return(magnitude_image);
237 dx_view=AcquireVirtualCacheView(dx_image,exception);
238 dy_view=AcquireVirtualCacheView(dy_image,exception);
239 magnitude_view=AcquireAuthenticCacheView(magnitude_image,exception);
240#if defined(MAGICKCORE_OPENMP_SUPPORT)
241 #pragma omp parallel for schedule(static) shared(status) \
242 magick_number_threads(dx_image,magnitude_image,dx_image->rows,1)
244 for (y=0; y < (ssize_t) dx_image->rows; y++)
256 if (status == MagickFalse)
258 p=GetCacheViewVirtualPixels(dx_view,0,y,dx_image->columns,1,exception);
259 q=GetCacheViewVirtualPixels(dy_view,0,y,dx_image->columns,1,exception);
260 r=GetCacheViewAuthenticPixels(magnitude_view,0,y,dx_image->columns,1,
262 if ((p == (
const Quantum *) NULL) || (q == (
const Quantum *) NULL) ||
263 (r == (Quantum *) NULL))
268 for (x=0; x < (ssize_t) dx_image->columns; x++)
273 for (i=0; i < (ssize_t) GetPixelChannels(dx_image); i++)
275 PixelChannel channel = GetPixelChannelChannel(dx_image,i);
276 PixelTrait traits = GetPixelChannelTraits(dx_image,channel);
277 PixelTrait dy_traits = GetPixelChannelTraits(dy_image,channel);
278 if ((traits == UndefinedPixelTrait) ||
279 (dy_traits == UndefinedPixelTrait) ||
280 ((dy_traits & UpdatePixelTrait) == 0))
282 r[i]=ClampToQuantum(hypot((
double) p[i],(
double)
283 GetPixelChannel(dy_image,channel,q)));
285 p+=(ptrdiff_t) GetPixelChannels(dx_image);
286 q+=(ptrdiff_t) GetPixelChannels(dy_image);
287 r+=(ptrdiff_t) GetPixelChannels(magnitude_image);
289 if (SyncCacheViewAuthenticPixels(magnitude_view,exception) == MagickFalse)
292 magnitude_view=DestroyCacheView(magnitude_view);
293 dy_view=DestroyCacheView(dy_view);
294 dx_view=DestroyCacheView(dx_view);
295 if (status == MagickFalse)
296 magnitude_image=DestroyImage(magnitude_image);
297 return(magnitude_image);
300static Image *BlendMaxMagnitudeImage(
const Image *alpha_image,
323 magnitude_image=CloneImage(alpha_image,0,0,MagickTrue,exception);
324 if (magnitude_image == (
Image *) NULL)
325 return(magnitude_image);
326 alpha_view=AcquireVirtualCacheView(alpha_image,exception);
327 beta_view=AcquireVirtualCacheView(beta_image,exception);
328 dx_view=AcquireVirtualCacheView(dx_image,exception);
329 dy_view=AcquireVirtualCacheView(dy_image,exception);
330 magnitude_view=AcquireAuthenticCacheView(magnitude_image,exception);
331#if defined(MAGICKCORE_OPENMP_SUPPORT)
332 #pragma omp parallel for schedule(static) shared(status) \
333 magick_number_threads(alpha_image,magnitude_image,alpha_image->rows,1)
335 for (y=0; y < (ssize_t) alpha_image->rows; y++)
349 if (status == MagickFalse)
351 p=GetCacheViewVirtualPixels(alpha_view,0,y,alpha_image->columns,1,
353 q=GetCacheViewVirtualPixels(beta_view,0,y,alpha_image->columns,1,exception);
354 r=GetCacheViewVirtualPixels(dx_view,0,y,alpha_image->columns,1,exception);
355 s=GetCacheViewVirtualPixels(dy_view,0,y,alpha_image->columns,1,exception);
356 t=GetCacheViewAuthenticPixels(magnitude_view,0,y,alpha_image->columns,1,
358 if ((p == (
const Quantum *) NULL) || (q == (
const Quantum *) NULL) ||
359 (r == (
const Quantum *) NULL) || (s == (
const Quantum *) NULL) ||
360 (t == (Quantum *) NULL))
365 for (x=0; x < (ssize_t) alpha_image->columns; x++)
370 for (i=0; i < (ssize_t) GetPixelChannels(alpha_image); i++)
372 PixelChannel channel = GetPixelChannelChannel(alpha_image,i);
373 PixelTrait traits = GetPixelChannelTraits(alpha_image,channel);
374 PixelTrait beta_traits = GetPixelChannelTraits(beta_image,channel);
375 if ((traits == UndefinedPixelTrait) ||
376 (beta_traits == UndefinedPixelTrait) ||
377 ((beta_traits & UpdatePixelTrait) == 0))
379 if (p[i] > GetPixelChannel(beta_image,channel,q))
380 t[i]=GetPixelChannel(dx_image,channel,r);
382 t[i]=GetPixelChannel(dy_image,channel,s);
384 p+=(ptrdiff_t) GetPixelChannels(alpha_image);
385 q+=(ptrdiff_t) GetPixelChannels(beta_image);
386 r+=(ptrdiff_t) GetPixelChannels(dx_image);
387 s+=(ptrdiff_t) GetPixelChannels(dy_image);
388 t+=(ptrdiff_t) GetPixelChannels(magnitude_image);
390 if (SyncCacheViewAuthenticPixels(magnitude_view,exception) == MagickFalse)
393 magnitude_view=DestroyCacheView(magnitude_view);
394 dy_view=DestroyCacheView(dy_view);
395 dx_view=DestroyCacheView(dx_view);
396 beta_view=DestroyCacheView(beta_view);
397 alpha_view=DestroyCacheView(alpha_view);
398 if (status == MagickFalse)
399 magnitude_image=DestroyImage(magnitude_image);
400 return(magnitude_image);
403static Image *BlendSumImage(
const Image *alpha_image,
const Image *beta_image,
404 const double attenuate,
const double sign,
ExceptionInfo *exception)
423 sum_image=CloneImage(alpha_image,0,0,MagickTrue,exception);
424 if (sum_image == (
Image *) NULL)
426 alpha_view=AcquireVirtualCacheView(alpha_image,exception);
427 beta_view=AcquireVirtualCacheView(beta_image,exception);
428 sum_view=AcquireAuthenticCacheView(sum_image,exception);
429#if defined(MAGICKCORE_OPENMP_SUPPORT)
430 #pragma omp parallel for schedule(static) shared(status) \
431 magick_number_threads(alpha_image,sum_image,alpha_image->rows,1)
433 for (y=0; y < (ssize_t) alpha_image->rows; y++)
445 if (status == MagickFalse)
447 p=GetCacheViewVirtualPixels(alpha_view,0,y,alpha_image->columns,1,
449 q=GetCacheViewVirtualPixels(beta_view,0,y,alpha_image->columns,1,exception);
450 r=GetCacheViewAuthenticPixels(sum_view,0,y,alpha_image->columns,1,
452 if ((p == (
const Quantum *) NULL) || (q == (
const Quantum *) NULL) ||
453 (r == (Quantum *) NULL))
458 for (x=0; x < (ssize_t) alpha_image->columns; x++)
463 for (i=0; i < (ssize_t) GetPixelChannels(alpha_image); i++)
465 PixelChannel channel = GetPixelChannelChannel(alpha_image,i);
466 PixelTrait traits = GetPixelChannelTraits(alpha_image,channel);
467 PixelTrait beta_traits = GetPixelChannelTraits(beta_image,channel);
468 if ((traits == UndefinedPixelTrait) ||
469 (beta_traits == UndefinedPixelTrait) ||
470 ((beta_traits & UpdatePixelTrait) == 0))
472 r[i]=ClampToQuantum(attenuate*((
double) p[i]+sign*
473 (
double) GetPixelChannel(beta_image,channel,q)));
475 p+=(ptrdiff_t) GetPixelChannels(alpha_image);
476 q+=(ptrdiff_t) GetPixelChannels(beta_image);
477 r+=(ptrdiff_t) GetPixelChannels(sum_image);
479 if (SyncCacheViewAuthenticPixels(sum_view,exception) == MagickFalse)
482 sum_view=DestroyCacheView(sum_view);
483 beta_view=DestroyCacheView(beta_view);
484 alpha_view=DestroyCacheView(alpha_view);
485 if (status == MagickFalse)
486 sum_image=DestroyImage(sum_image);
490static Image *BlendDivergentImage(
const Image *alpha_image,
493#define FreeDivergentResources() \
495 if (dy_image != (Image *) NULL) \
496 dy_image=DestroyImage(dy_image); \
497 if (dx_image != (Image *) NULL) \
498 dx_image=DestroyImage(dx_image); \
499 if (magnitude_beta != (Image *) NULL) \
500 magnitude_beta=DestroyImage(magnitude_beta); \
501 if (dy_beta != (Image *) NULL) \
502 dy_beta=DestroyImage(dy_beta); \
503 if (dx_beta != (Image *) NULL) \
504 dx_beta=DestroyImage(dx_beta); \
505 if (magnitude_alpha != (Image *) NULL) \
506 magnitude_alpha=DestroyImage(magnitude_alpha); \
507 if (dy_alpha != (Image *) NULL) \
508 dy_alpha=DestroyImage(dy_alpha); \
509 if (dx_alpha != (Image *) NULL) \
510 dx_alpha=DestroyImage(dx_alpha); \
514 *divergent_image = (
Image *) NULL,
515 *dx_alpha = (
Image *) NULL,
516 *dx_beta = (
Image *) NULL,
517 *dx_divergent = (
Image *) NULL,
518 *dx_image = (
Image *) NULL,
519 *dy_alpha = (
Image *) NULL,
520 *dy_beta = (
Image *) NULL,
521 *dy_divergent = (
Image *) NULL,
522 *dy_image = (
Image *) NULL,
523 *magnitude_alpha = (
Image *) NULL,
524 *magnitude_beta = (
Image *) NULL;
529 dx_alpha=BlendConvolveImage(alpha_image,
"3x1:-0.5,0.0,0.5",exception);
530 if (dx_alpha == (
Image *) NULL)
532 FreeDivergentResources();
533 return((
Image *) NULL);
535 dy_alpha=BlendConvolveImage(alpha_image,
"1x3:-0.5,0.0,0.5",exception);
536 if (dy_alpha == (
Image *) NULL)
538 FreeDivergentResources();
539 return((
Image *) NULL);
541 magnitude_alpha=BlendMagnitudeImage(dx_alpha,dy_alpha,exception);
542 if (magnitude_alpha == (
Image *) NULL)
544 FreeDivergentResources();
545 return((
Image *) NULL);
550 dx_beta=BlendConvolveImage(beta_image,
"3x1:-0.5,0.0,0.5",exception);
551 if (dx_beta == (
Image *) NULL)
553 FreeDivergentResources();
554 return((
Image *) NULL);
556 dy_beta=BlendConvolveImage(beta_image,
"1x3:-0.5,0.0,0.5",exception);
557 if (dy_beta == (
Image *) NULL)
559 FreeDivergentResources();
560 return((
Image *) NULL);
562 magnitude_beta=BlendMagnitudeImage(dx_beta,dy_beta,exception);
563 if (magnitude_beta == (
Image *) NULL)
565 FreeDivergentResources();
566 return((
Image *) NULL);
571 dx_image=BlendMaxMagnitudeImage(magnitude_alpha,magnitude_beta,dx_alpha,
573 if (dx_image == (
Image *) NULL)
575 FreeDivergentResources();
576 return((
Image *) NULL);
578 dy_image=BlendMaxMagnitudeImage(magnitude_alpha,magnitude_beta,dy_alpha,
580 if (dy_image == (
Image *) NULL)
582 FreeDivergentResources();
583 return((
Image *) NULL);
585 dx_beta=DestroyImage(dx_beta);
586 dx_alpha=DestroyImage(dx_alpha);
587 magnitude_beta=DestroyImage(magnitude_beta);
588 magnitude_alpha=DestroyImage(magnitude_alpha);
592 dx_divergent=BlendConvolveImage(dx_image,
"3x1:-0.5,0.0,0.5",exception);
593 if (dx_divergent == (
Image *) NULL)
595 FreeDivergentResources();
596 return((
Image *) NULL);
598 dy_divergent=BlendConvolveImage(dy_image,
"1x3:-0.5,0.0,0.5",exception);
599 if (dy_divergent == (
Image *) NULL)
601 FreeDivergentResources();
602 return((
Image *) NULL);
604 divergent_image=BlendSumImage(dx_divergent,dy_divergent,0.25,1.0,exception);
605 dy_divergent=DestroyImage(dy_divergent);
606 dx_divergent=DestroyImage(dx_divergent);
607 if (divergent_image == (
Image *) NULL)
609 FreeDivergentResources();
610 return((
Image *) NULL);
612 FreeDivergentResources();
613 return(divergent_image);
616static MagickBooleanType BlendMaskAlphaChannel(
Image *image,
632 if (SetImageAlpha(image,OpaqueAlpha,exception) == MagickFalse)
634 image_view=AcquireAuthenticCacheView(image,exception);
635 mask_view=AcquireVirtualCacheView(mask_image,exception);
636#if defined(MAGICKCORE_OPENMP_SUPPORT)
637 #pragma omp parallel for schedule(static) shared(status) \
638 magick_number_threads(image,image,image->rows,2)
640 for (y=0; y < (ssize_t) image->rows; y++)
651 if (status == MagickFalse)
653 p=GetCacheViewVirtualPixels(mask_view,0,y,image->columns,1,exception);
654 q=GetCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
655 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
660 for (x=0; x < (ssize_t) image->columns; x++)
663 alpha = GetPixelAlpha(mask_image,p);
666 i = GetPixelChannelOffset(image,AlphaPixelChannel);
668 if (fabs((
double) alpha) >= MagickEpsilon)
670 p+=(ptrdiff_t) GetPixelChannels(mask_image);
671 q+=(ptrdiff_t) GetPixelChannels(image);
673 if (SyncCacheViewAuthenticPixels(image_view,exception) == MagickFalse)
676 mask_view=DestroyCacheView(mask_view);
677 image_view=DestroyCacheView(image_view);
690 mean[MaxPixelChannels];
705 (void) memset(mean,0,MaxPixelChannels*
sizeof(*mean));
706 alpha_view=AcquireVirtualCacheView(image,exception);
707 for (y=0; y < (ssize_t) image->rows; y++)
715 p=GetCacheViewVirtualPixels(alpha_view,0,y,image->columns,1,
717 if (p == (
const Quantum *) NULL)
719 for (x=0; x < (ssize_t) image->columns; x++)
724 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
726 PixelChannel channel = GetPixelChannelChannel(image,i);
727 PixelTrait traits = GetPixelChannelTraits(image,channel);
728 if (traits == UndefinedPixelTrait)
730 mean[i]+=QuantumScale*(double) p[i];
732 p+=(ptrdiff_t) GetPixelChannels(image);
735 alpha_view=DestroyCacheView(alpha_view);
736 if (y < (ssize_t) image->rows)
737 return((
Image *) NULL);
738 for (j=0; j < (ssize_t) GetPixelChannels(image); j++)
739 mean[j]=(
double) QuantumRange*mean[j]/image->columns/
744 mean_image=CloneImage(image,0,0,MagickTrue,exception);
745 if (mean_image == (
Image *) NULL)
747 mask_view=AcquireVirtualCacheView(mask_image,exception);
748 mean_view=AcquireAuthenticCacheView(mean_image,exception);
749#if defined(MAGICKCORE_OPENMP_SUPPORT)
750 #pragma omp parallel for schedule(static) shared(status) \
751 magick_number_threads(mask_image,mean_image,mean_image->rows,4)
753 for (y=0; y < (ssize_t) mean_image->rows; y++)
764 if (status == MagickFalse)
766 p=GetCacheViewVirtualPixels(mask_view,0,y,mean_image->columns,1,exception);
767 q=GetCacheViewAuthenticPixels(mean_view,0,y,mean_image->columns,1,
769 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
774 for (x=0; x < (ssize_t) mean_image->columns; x++)
777 alpha = GetPixelAlpha(mask_image,p),
778 mask = GetPixelReadMask(mask_image,p);
783 for (i=0; i < (ssize_t) GetPixelChannels(mean_image); i++)
785 PixelChannel channel = GetPixelChannelChannel(mean_image,i);
786 PixelTrait traits = GetPixelChannelTraits(mean_image,channel);
787 if (traits == UndefinedPixelTrait)
789 if (mask <= (QuantumRange/2))
792 if (fabs((
double) alpha) >= MagickEpsilon)
793 q[i]=ClampToQuantum(mean[i]);
795 p+=(ptrdiff_t) GetPixelChannels(mask_image);
796 q+=(ptrdiff_t) GetPixelChannels(mean_image);
798 if (SyncCacheViewAuthenticPixels(mean_view,exception) == MagickFalse)
801 mask_view=DestroyCacheView(mask_view);
802 mean_view=DestroyCacheView(mean_view);
803 if (status == MagickFalse)
804 mean_image=DestroyImage(mean_image);
808static MagickBooleanType BlendRMSEResidual(
const Image *alpha_image,
822 columns = MagickMax(alpha_image->columns,beta_image->columns),
823 rows = MagickMax(alpha_image->rows,beta_image->rows);
829 alpha_view=AcquireVirtualCacheView(alpha_image,exception);
830 beta_view=AcquireVirtualCacheView(beta_image,exception);
831#if defined(MAGICKCORE_OPENMP_SUPPORT)
832 #pragma omp parallel for schedule(static) shared(status) \
833 magick_number_threads(alpha_image,alpha_image,rows,1)
835 for (y=0; y < (ssize_t) rows; y++)
850 if (status == MagickFalse)
852 p=GetCacheViewVirtualPixels(alpha_view,0,y,columns,1,exception);
853 q=GetCacheViewVirtualPixels(beta_view,0,y,columns,1,exception);
854 if ((p == (
const Quantum *) NULL) || (q == (
const Quantum *) NULL))
859 channel_residual=0.0;
860 for (x=0; x < (ssize_t) columns; x++)
869 if ((GetPixelReadMask(alpha_image,p) <= (QuantumRange/2)) ||
870 (GetPixelReadMask(beta_image,q) <= (QuantumRange/2)))
872 p+=(ptrdiff_t) GetPixelChannels(alpha_image);
873 q+=(ptrdiff_t) GetPixelChannels(beta_image);
876 Sa=QuantumScale*(double) GetPixelAlpha(alpha_image,p);
877 Da=QuantumScale*(double) GetPixelAlpha(beta_image,q);
878 for (i=0; i < (ssize_t) GetPixelChannels(alpha_image); i++)
883 PixelChannel channel = GetPixelChannelChannel(alpha_image,i);
884 PixelTrait traits = GetPixelChannelTraits(alpha_image,channel);
885 PixelTrait beta_traits = GetPixelChannelTraits(beta_image,channel);
886 if ((traits == UndefinedPixelTrait) ||
887 (beta_traits == UndefinedPixelTrait) ||
888 ((beta_traits & UpdatePixelTrait) == 0))
890 if (channel == AlphaPixelChannel)
891 distance=QuantumScale*((double) p[i]-(
double) GetPixelChannel(
892 beta_image,channel,q));
894 distance=QuantumScale*(Sa*(double) p[i]-Da*(
double) GetPixelChannel(
895 beta_image,channel,q));
896 channel_residual+=distance*distance;
899 p+=(ptrdiff_t) GetPixelChannels(alpha_image);
900 q+=(ptrdiff_t) GetPixelChannels(beta_image);
902#if defined(MAGICKCORE_OPENMP_SUPPORT)
903 #pragma omp critical (MagickCore_BlendRMSEResidual)
907 *residual+=channel_residual;
910 area=PerceptibleReciprocal(area);
911 beta_view=DestroyCacheView(beta_view);
912 alpha_view=DestroyCacheView(alpha_view);
913 *residual=sqrt(*residual*area/(
double) GetImageChannels(alpha_image));
917static MagickBooleanType CompositeOverImage(
Image *image,
918 const Image *source_image,
const MagickBooleanType clip_to_self,
919 const ssize_t x_offset,
const ssize_t y_offset,
ExceptionInfo *exception)
921#define CompositeImageTag "Composite/Image"
946 value=GetImageArtifact(image,
"compose:clamp");
947 if (value != (
const char *) NULL)
948 clamp=IsStringTrue(value);
951 source_view=AcquireVirtualCacheView(source_image,exception);
952 image_view=AcquireAuthenticCacheView(image,exception);
953#if defined(MAGICKCORE_OPENMP_SUPPORT)
954 #pragma omp parallel for schedule(static) shared(progress,status) \
955 magick_number_threads(source_image,image,image->rows,1)
957 for (y=0; y < (ssize_t) image->rows; y++)
975 if (status == MagickFalse)
977 if (clip_to_self != MagickFalse)
981 if ((y-(
double) y_offset) >= (double) source_image->rows)
987 pixels=(Quantum *) NULL;
989 if ((y >= y_offset) &&
990 ((y-(double) y_offset) < (double) source_image->rows))
992 p=GetCacheViewVirtualPixels(source_view,0,
993 CastDoubleToLong(y-(
double) y_offset),source_image->columns,1,
995 if (p == (
const Quantum *) NULL)
1002 p-=(ptrdiff_t) CastDoubleToSsizeT((
double) x_offset*
1003 GetPixelChannels(source_image));
1005 q=GetCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
1006 if (q == (Quantum *) NULL)
1011 GetPixelInfo(image,&canvas_pixel);
1012 GetPixelInfo(source_image,&source_pixel);
1013 for (x=0; x < (ssize_t) image->columns; x++)
1033 if (clip_to_self != MagickFalse)
1037 q+=(ptrdiff_t) GetPixelChannels(image);
1040 if ((x-(
double) x_offset) >= (
double) source_image->columns)
1043 if ((pixels == (Quantum *) NULL) || (x < x_offset) ||
1044 ((x-(double) x_offset) >= (double) source_image->columns))
1047 source[MaxPixelChannels];
1054 (void) GetOneVirtualPixel(source_image,
1055 CastDoubleToLong(x-(
double) x_offset),
1056 CastDoubleToLong(y-(
double) y_offset),source,exception);
1057 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
1062 PixelChannel channel = GetPixelChannelChannel(image,i);
1063 PixelTrait traits = GetPixelChannelTraits(image,channel);
1064 PixelTrait source_traits=GetPixelChannelTraits(source_image,
1066 if ((traits == UndefinedPixelTrait) ||
1067 (source_traits == UndefinedPixelTrait))
1069 if (channel == AlphaPixelChannel)
1070 pixel=(MagickRealType) TransparentAlpha;
1072 pixel=(MagickRealType) q[i];
1073 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
1074 ClampToQuantum(pixel);
1076 q+=(ptrdiff_t) GetPixelChannels(image);
1084 Sa=QuantumScale*(double) GetPixelAlpha(source_image,p);
1085 Da=QuantumScale*(double) GetPixelAlpha(image,q);
1087 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
1092 PixelChannel channel = GetPixelChannelChannel(image,i);
1093 PixelTrait traits = GetPixelChannelTraits(image,channel);
1094 PixelTrait source_traits=GetPixelChannelTraits(source_image,channel);
1095 if (traits == UndefinedPixelTrait)
1097 if ((source_traits == UndefinedPixelTrait) &&
1098 (channel != AlphaPixelChannel))
1100 if (channel == AlphaPixelChannel)
1105 pixel=(double) QuantumRange*alpha;
1106 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
1107 ClampToQuantum(pixel);
1114 Sc=(MagickRealType) GetPixelChannel(source_image,channel,p);
1115 Dc=(MagickRealType) q[i];
1116 if ((traits & CopyPixelTrait) != 0)
1121 q[i]=ClampToQuantum(Sc);
1129 Sca=QuantumScale*Sa*Sc;
1130 Dca=QuantumScale*Da*Dc;
1131 gamma=PerceptibleReciprocal(alpha);
1132 pixel=(double) QuantumRange*gamma*(Sca+Dca*(1.0-Sa));
1133 q[i]=clamp != MagickFalse ? ClampPixel(pixel) : ClampToQuantum(pixel);
1135 p+=(ptrdiff_t) GetPixelChannels(source_image);
1136 channels=GetPixelChannels(source_image);
1137 if (p >= (pixels+channels*source_image->columns))
1139 q+=(ptrdiff_t) GetPixelChannels(image);
1141 if (SyncCacheViewAuthenticPixels(image_view,exception) == MagickFalse)
1143 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1148#if defined(MAGICKCORE_OPENMP_SUPPORT)
1152 proceed=SetImageProgress(image,CompositeImageTag,progress,image->rows);
1153 if (proceed == MagickFalse)
1157 source_view=DestroyCacheView(source_view);
1158 image_view=DestroyCacheView(image_view);
1162static MagickBooleanType SaliencyBlendImage(
Image *image,
1163 const Image *source_image,
const ssize_t x_offset,
const ssize_t y_offset,
1164 const double iterations,
const double residual_threshold,
const size_t tick,
1171 *residual_image = (
Image *) NULL;
1177 status = MagickTrue,
1178 verbose = MagickFalse;
1182 source_image->columns,
1194 crop_image=CropImage(image,&crop_info,exception);
1195 if (crop_image == (
Image *) NULL)
1196 return(MagickFalse);
1197 DisableCompositeClampUnlessSpecified(crop_image);
1198 divergent_image=BlendDivergentImage(crop_image,source_image,exception);
1199 if (divergent_image == (
Image *) NULL)
1201 crop_image=DestroyImage(crop_image);
1202 return(MagickFalse);
1204 (void) ResetImagePage(crop_image,
"0x0+0+0");
1205 relax_image=BlendMeanImage(crop_image,source_image,exception);
1206 if (relax_image == (
Image *) NULL)
1208 crop_image=DestroyImage(crop_image);
1209 divergent_image=DestroyImage(divergent_image);
1210 return(MagickFalse);
1212 status=BlendMaskAlphaChannel(crop_image,source_image,exception);
1213 if (status == MagickFalse)
1215 crop_image=DestroyImage(crop_image);
1216 divergent_image=DestroyImage(divergent_image);
1217 return(MagickFalse);
1219 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1220 if (residual_image == (
Image *) NULL)
1222 crop_image=DestroyImage(crop_image);
1223 relax_image=DestroyImage(relax_image);
1224 return(MagickFalse);
1229 kernel_info=AcquireKernelInfo(
"3x3:0,0.25,0,0.25,0,0.25,0,0.25,0",exception);
1232 crop_image=DestroyImage(crop_image);
1233 residual_image=DestroyImage(residual_image);
1234 relax_image=DestroyImage(relax_image);
1235 return(MagickFalse);
1237 verbose=IsStringTrue(GetImageArtifact(image,
"verbose"));
1238 if (verbose != MagickFalse)
1239 (void) FormatLocaleFile(stderr,
"saliency blending:\n");
1240 for (i=0; i < iterations; i++)
1249 convolve_image=ConvolveImage(relax_image,kernel_info,exception);
1250 if (convolve_image == (
Image *) NULL)
1252 relax_image=DestroyImage(relax_image);
1253 relax_image=convolve_image;
1254 sum_image=BlendSumImage(relax_image,divergent_image,1.0,-1.0,exception);
1255 if (sum_image == (
Image *) NULL)
1257 relax_image=DestroyImage(relax_image);
1258 relax_image=sum_image;
1259 status=CompositeOverImage(relax_image,crop_image,MagickTrue,0,0,exception);
1260 if (status == MagickFalse)
1262 status=BlendRMSEResidual(relax_image,residual_image,&residual,exception);
1263 if (status == MagickFalse)
1265 if ((verbose != MagickFalse) && ((i % MagickMax(tick,1)) == 0))
1266 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(double) residual);
1267 if (residual < residual_threshold)
1269 if (verbose != MagickFalse)
1270 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(double)
1274 residual_image=DestroyImage(residual_image);
1275 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1276 if (residual_image == (
Image *) NULL)
1279 kernel_info=DestroyKernelInfo(kernel_info);
1280 crop_image=DestroyImage(crop_image);
1281 divergent_image=DestroyImage(divergent_image);
1282 residual_image=DestroyImage(residual_image);
1286 status=CompositeOverImage(image,relax_image,MagickTrue,x_offset,y_offset,
1288 relax_image=DestroyImage(relax_image);
1292static MagickBooleanType SeamlessBlendImage(
Image *image,
1293 const Image *source_image,
const ssize_t x_offset,
const ssize_t y_offset,
1294 const double iterations,
const double residual_threshold,
const size_t tick,
1309 status = MagickTrue,
1310 verbose = MagickFalse;
1314 source_image->columns,
1326 crop_image=CropImage(image,&crop_info,exception);
1327 if (crop_image == (
Image *) NULL)
1328 return(MagickFalse);
1329 DisableCompositeClampUnlessSpecified(crop_image);
1330 (void) ResetImagePage(crop_image,
"0x0+0+0");
1331 sum_image=BlendSumImage(crop_image,source_image,1.0,-1.0,exception);
1332 crop_image=DestroyImage(crop_image);
1333 if (sum_image == (
Image *) NULL)
1334 return(MagickFalse);
1335 mean_image=BlendMeanImage(sum_image,source_image,exception);
1336 sum_image=DestroyImage(sum_image);
1337 if (mean_image == (
Image *) NULL)
1338 return(MagickFalse);
1339 relax_image=CloneImage(mean_image,0,0,MagickTrue,exception);
1340 if (relax_image == (
Image *) NULL)
1342 mean_image=DestroyImage(mean_image);
1343 return(MagickFalse);
1345 status=BlendMaskAlphaChannel(mean_image,source_image,exception);
1346 if (status == MagickFalse)
1348 relax_image=DestroyImage(relax_image);
1349 mean_image=DestroyImage(mean_image);
1350 return(MagickFalse);
1352 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1353 if (residual_image == (
Image *) NULL)
1355 relax_image=DestroyImage(relax_image);
1356 mean_image=DestroyImage(mean_image);
1357 return(MagickFalse);
1362 kernel_info=AcquireKernelInfo(
"3x3:0,0.25,0,0.25,0,0.25,0,0.25,0",exception);
1365 residual_image=DestroyImage(residual_image);
1366 relax_image=DestroyImage(relax_image);
1367 mean_image=DestroyImage(mean_image);
1368 return(MagickFalse);
1370 verbose=IsStringTrue(GetImageArtifact(image,
"verbose"));
1371 if (verbose != MagickFalse)
1372 (void) FormatLocaleFile(stderr,
"seamless blending:\n");
1373 for (i=0; i < iterations; i++)
1381 convolve_image=ConvolveImage(relax_image,kernel_info,exception);
1382 if (convolve_image == (
Image *) NULL)
1384 relax_image=DestroyImage(relax_image);
1385 relax_image=convolve_image;
1386 status=CompositeOverImage(relax_image,mean_image,MagickTrue,0,0,exception);
1387 if (status == MagickFalse)
1389 status=BlendRMSEResidual(relax_image,residual_image,&residual,exception);
1390 if (status == MagickFalse)
1392 if ((verbose != MagickFalse) && ((i % MagickMax(tick,1)) == 0))
1393 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(double) residual);
1394 if (residual < residual_threshold)
1396 if (verbose != MagickFalse)
1397 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(double)
1401 if (residual_image != (
Image *) NULL)
1402 residual_image=DestroyImage(residual_image);
1403 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1404 if (residual_image == (
Image *) NULL)
1407 kernel_info=DestroyKernelInfo(kernel_info);
1408 mean_image=DestroyImage(mean_image);
1409 residual_image=DestroyImage(residual_image);
1413 foreground_image=BlendSumImage(source_image,relax_image,1.0,1.0,exception);
1414 relax_image=DestroyImage(relax_image);
1415 if (foreground_image == (
Image *) NULL)
1416 return(MagickFalse);
1417 (void) SetImageMask(foreground_image,ReadPixelMask,(
const Image *) NULL,
1419 status=CompositeOverImage(image,foreground_image,MagickTrue,x_offset,y_offset,
1421 foreground_image=DestroyImage(foreground_image);
1425MagickExport MagickBooleanType CompositeImage(
Image *image,
1426 const Image *composite,
const CompositeOperator compose,
1427 const MagickBooleanType clip_to_self,
const ssize_t x_offset,
1430#define CompositeImageTag "Composite/Image"
1437 colorspace = HCLColorspace;
1443 white_luminance = 10000.0;
1449 illuminant = D65Illuminant;
1478 assert(image != (
Image *) NULL);
1479 assert(image->signature == MagickCoreSignature);
1480 assert(composite != (
Image *) NULL);
1481 assert(composite->signature == MagickCoreSignature);
1482 if (IsEventLogging() != MagickFalse)
1483 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1484 if (SetImageStorageClass(image,DirectClass,exception) == MagickFalse)
1485 return(MagickFalse);
1486 source_image=CloneImage(composite,0,0,MagickTrue,exception);
1487 if (source_image == (
const Image *) NULL)
1488 return(MagickFalse);
1489 (void) SetImageColorspace(source_image,image->colorspace,exception);
1490 if ((compose == OverCompositeOp) || (compose == SrcOverCompositeOp))
1492 status=CompositeOverImage(image,source_image,clip_to_self,x_offset,
1493 y_offset,exception);
1494 source_image=DestroyImage(source_image);
1498 canvas_image=(
Image *) NULL;
1499 canvas_dissolve=1.0;
1500 white_luminance=10000.0;
1501 artifact=GetImageArtifact(image,
"compose:white-luminance");
1502 if (artifact != (
const char *) NULL)
1503 white_luminance=StringToDouble(artifact,(
char **) NULL);
1504 artifact=GetImageArtifact(image,
"compose:illuminant");
1505 if (artifact != (
const char *) NULL)
1510 illuminant_type=ParseCommandOption(MagickIlluminantOptions,MagickFalse,
1512 if (illuminant_type < 0)
1513 illuminant=UndefinedIlluminant;
1515 illuminant=(IlluminantType) illuminant_type;
1517 artifact=GetImageArtifact(image,
"compose:colorspace");
1518 if (artifact != (
const char *) NULL)
1523 colorspace_type=ParseCommandOption(MagickColorspaceOptions,MagickFalse,
1525 if (colorspace_type < 0)
1526 colorspace=UndefinedColorspace;
1528 colorspace=(ColorspaceType) colorspace_type;
1531 artifact=GetImageArtifact(image,
"compose:clamp");
1532 if (artifact != (
const char *) NULL)
1533 clamp=IsStringTrue(artifact);
1534 compose_sync=MagickTrue;
1535 artifact=GetImageArtifact(image,
"compose:sync");
1536 if (artifact != (
const char *) NULL)
1537 compose_sync=IsStringTrue(artifact);
1538 SetGeometryInfo(&geometry_info);
1540 percent_chroma=100.0;
1541 source_dissolve=1.0;
1545 case CopyCompositeOp:
1547 if ((x_offset < 0) || (y_offset < 0))
1549 if ((x_offset+(ssize_t) source_image->columns) > (ssize_t) image->columns)
1551 if ((y_offset+(ssize_t) source_image->rows) > (ssize_t) image->rows)
1553 if ((source_image->alpha_trait == UndefinedPixelTrait) &&
1554 (image->alpha_trait != UndefinedPixelTrait))
1555 (void) SetImageAlphaChannel(source_image,OpaqueAlphaChannel,exception);
1557 source_view=AcquireVirtualCacheView(source_image,exception);
1558 image_view=AcquireAuthenticCacheView(image,exception);
1559#if defined(MAGICKCORE_OPENMP_SUPPORT)
1560 #pragma omp parallel for schedule(static) shared(status) \
1561 magick_number_threads(source_image,image,source_image->rows,4)
1563 for (y=0; y < (ssize_t) source_image->rows; y++)
1577 if (status == MagickFalse)
1579 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
1581 q=GetCacheViewAuthenticPixels(image_view,x_offset,y+y_offset,
1582 source_image->columns,1,exception);
1583 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
1588 for (x=0; x < (ssize_t) source_image->columns; x++)
1593 if (GetPixelReadMask(source_image,p) <= (QuantumRange/2))
1595 p+=(ptrdiff_t) GetPixelChannels(source_image);
1596 q+=(ptrdiff_t) GetPixelChannels(image);
1599 for (i=0; i < (ssize_t) GetPixelChannels(source_image); i++)
1601 PixelChannel channel = GetPixelChannelChannel(source_image,i);
1602 PixelTrait source_traits = GetPixelChannelTraits(source_image,
1604 PixelTrait traits = GetPixelChannelTraits(image,channel);
1605 if ((source_traits == UndefinedPixelTrait) ||
1606 (traits == UndefinedPixelTrait))
1608 SetPixelChannel(image,channel,p[i],q);
1610 p+=(ptrdiff_t) GetPixelChannels(source_image);
1611 q+=(ptrdiff_t) GetPixelChannels(image);
1613 sync=SyncCacheViewAuthenticPixels(image_view,exception);
1614 if (sync == MagickFalse)
1616 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1621 proceed=SetImageProgress(image,CompositeImageTag,(MagickOffsetType)
1623 if (proceed == MagickFalse)
1627 source_view=DestroyCacheView(source_view);
1628 image_view=DestroyCacheView(image_view);
1629 source_image=DestroyImage(source_image);
1632 case IntensityCompositeOp:
1634 if ((x_offset < 0) || (y_offset < 0))
1636 if ((x_offset+(ssize_t) source_image->columns) > (ssize_t) image->columns)
1638 if ((y_offset+(ssize_t) source_image->rows) > (ssize_t) image->rows)
1641 source_view=AcquireVirtualCacheView(source_image,exception);
1642 image_view=AcquireAuthenticCacheView(image,exception);
1643#if defined(MAGICKCORE_OPENMP_SUPPORT)
1644 #pragma omp parallel for schedule(static) shared(status) \
1645 magick_number_threads(source_image,image,source_image->rows,4)
1647 for (y=0; y < (ssize_t) source_image->rows; y++)
1661 if (status == MagickFalse)
1663 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
1665 q=GetCacheViewAuthenticPixels(image_view,x_offset,y+y_offset,
1666 source_image->columns,1,exception);
1667 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
1672 for (x=0; x < (ssize_t) source_image->columns; x++)
1674 if (GetPixelReadMask(source_image,p) <= (QuantumRange/2))
1676 p+=(ptrdiff_t) GetPixelChannels(source_image);
1677 q+=(ptrdiff_t) GetPixelChannels(image);
1680 SetPixelAlpha(image,clamp != MagickFalse ?
1681 ClampPixel(GetPixelIntensity(source_image,p)) :
1682 ClampToQuantum(GetPixelIntensity(source_image,p)),q);
1683 p+=(ptrdiff_t) GetPixelChannels(source_image);
1684 q+=(ptrdiff_t) GetPixelChannels(image);
1686 sync=SyncCacheViewAuthenticPixels(image_view,exception);
1687 if (sync == MagickFalse)
1689 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1694 proceed=SetImageProgress(image,CompositeImageTag,(MagickOffsetType)
1696 if (proceed == MagickFalse)
1700 source_view=DestroyCacheView(source_view);
1701 image_view=DestroyCacheView(image_view);
1702 source_image=DestroyImage(source_image);
1705 case CopyAlphaCompositeOp:
1706 case ChangeMaskCompositeOp:
1712 if ((image->alpha_trait & BlendPixelTrait) == 0)
1713 (void) SetImageAlphaChannel(image,OpaqueAlphaChannel,exception);
1716 case BlurCompositeOp:
1741 canvas_image=CloneImage(image,0,0,MagickTrue,exception);
1742 if (canvas_image == (
Image *) NULL)
1744 source_image=DestroyImage(source_image);
1745 return(MagickFalse);
1751 artifact=GetImageArtifact(image,
"compose:args");
1752 if (artifact != (
const char *) NULL)
1753 flags=ParseGeometry(artifact,&geometry_info);
1754 if ((flags & WidthValue) == 0)
1756 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
1757 "InvalidSetting",
"'%s' '%s'",
"compose:args",artifact);
1758 source_image=DestroyImage(source_image);
1759 canvas_image=DestroyImage(canvas_image);
1760 return(MagickFalse);
1767 width=2.0*geometry_info.rho;
1769 if ((flags & HeightValue) != 0)
1770 height=2.0*geometry_info.sigma;
1778 if ((flags & XValue) != 0 )
1786 angle=DegreesToRadians(geometry_info.xi);
1787 blur.x1=width*cos(angle);
1788 blur.x2=width*sin(angle);
1789 blur.y1=(-height*sin(angle));
1790 blur.y2=height*cos(angle);
1794 if ((flags & YValue) != 0 )
1799 angle_start=DegreesToRadians(geometry_info.xi);
1800 angle_range=DegreesToRadians(geometry_info.psi)-angle_start;
1811 resample_filter=AcquireResampleFilter(image,exception);
1812 SetResampleFilter(resample_filter,GaussianFilter);
1816 GetPixelInfo(image,&pixel);
1817 source_view=AcquireVirtualCacheView(source_image,exception);
1818 canvas_view=AcquireAuthenticCacheView(canvas_image,exception);
1819 for (y=0; y < (ssize_t) source_image->rows; y++)
1833 if (((y+y_offset) < 0) || ((y+y_offset) >= (ssize_t) image->rows))
1835 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
1837 q=QueueCacheViewAuthenticPixels(canvas_view,0,y,canvas_image->columns,1,
1839 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
1841 for (x=0; x < (ssize_t) source_image->columns; x++)
1843 if (((x_offset+x) < 0) || ((x_offset+x) >= (ssize_t) image->columns))
1845 p+=(ptrdiff_t) GetPixelChannels(source_image);
1848 if (fabs(angle_range) > MagickEpsilon)
1853 angle=angle_start+angle_range*QuantumScale*(double)
1854 GetPixelBlue(source_image,p);
1855 blur.x1=width*cos(angle);
1856 blur.x2=width*sin(angle);
1857 blur.y1=(-height*sin(angle));
1858 blur.y2=height*cos(angle);
1860 ScaleResampleFilter(resample_filter,
1861 blur.x1*QuantumScale*(
double) GetPixelRed(source_image,p),
1862 blur.y1*QuantumScale*(
double) GetPixelGreen(source_image,p),
1863 blur.x2*QuantumScale*(
double) GetPixelRed(source_image,p),
1864 blur.y2*QuantumScale*(
double) GetPixelGreen(source_image,p) );
1865 (void) ResamplePixelColor(resample_filter,(
double) x_offset+x,
1866 (double) y_offset+y,&pixel,exception);
1867 SetPixelViaPixelInfo(canvas_image,&pixel,q);
1868 p+=(ptrdiff_t) GetPixelChannels(source_image);
1869 q+=(ptrdiff_t) GetPixelChannels(canvas_image);
1871 sync=SyncCacheViewAuthenticPixels(canvas_view,exception);
1872 if (sync == MagickFalse)
1875 resample_filter=DestroyResampleFilter(resample_filter);
1876 source_view=DestroyCacheView(source_view);
1877 canvas_view=DestroyCacheView(canvas_view);
1878 source_image=DestroyImage(source_image);
1879 source_image=canvas_image;
1882 case DisplaceCompositeOp:
1883 case DistortCompositeOp:
1904 canvas_image=CloneImage(image,0,0,MagickTrue,exception);
1905 if (canvas_image == (
Image *) NULL)
1907 source_image=DestroyImage(source_image);
1908 return(MagickFalse);
1910 SetGeometryInfo(&geometry_info);
1912 artifact=GetImageArtifact(image,
"compose:args");
1913 if (artifact != (
char *) NULL)
1914 flags=ParseGeometry(artifact,&geometry_info);
1915 if ((flags & (WidthValue | HeightValue)) == 0 )
1917 if ((flags & AspectValue) == 0)
1919 horizontal_scale=(MagickRealType) (source_image->columns-1)/2.0;
1920 vertical_scale=(MagickRealType) (source_image->rows-1)/2.0;
1924 horizontal_scale=(MagickRealType) (image->columns-1)/2.0;
1925 vertical_scale=(MagickRealType) (image->rows-1)/2.0;
1930 horizontal_scale=geometry_info.rho;
1931 vertical_scale=geometry_info.sigma;
1932 if ((flags & PercentValue) != 0)
1934 if ((flags & AspectValue) == 0)
1936 horizontal_scale*=(source_image->columns-1)/200.0;
1937 vertical_scale*=(source_image->rows-1)/200.0;
1941 horizontal_scale*=(image->columns-1)/200.0;
1942 vertical_scale*=(image->rows-1)/200.0;
1945 if ((flags & HeightValue) == 0)
1946 vertical_scale=horizontal_scale;
1956 center.x=(MagickRealType) x_offset;
1957 center.y=(MagickRealType) y_offset;
1958 if (compose == DistortCompositeOp)
1960 if ((flags & XValue) == 0)
1961 if ((flags & AspectValue) != 0)
1962 center.x=(MagickRealType) ((image->columns-1)/2.0);
1964 center.x=(MagickRealType) (x_offset+(source_image->columns-1)/
1967 if ((flags & AspectValue) != 0)
1968 center.x=geometry_info.xi;
1970 center.x=(MagickRealType) (x_offset+geometry_info.xi);
1971 if ((flags & YValue) == 0)
1972 if ((flags & AspectValue) != 0)
1973 center.y=(MagickRealType) ((image->rows-1)/2.0);
1975 center.y=(MagickRealType) (y_offset+(source_image->rows-1)/2.0);
1977 if ((flags & AspectValue) != 0)
1978 center.y=geometry_info.psi;
1980 center.y=(MagickRealType) (y_offset+geometry_info.psi);
1986 GetPixelInfo(image,&pixel);
1987 image_view=AcquireVirtualCacheView(image,exception);
1988 source_view=AcquireVirtualCacheView(source_image,exception);
1989 canvas_view=AcquireAuthenticCacheView(canvas_image,exception);
1990 for (y=0; y < (ssize_t) source_image->rows; y++)
2004 if (((y+y_offset) < 0) || ((y+y_offset) >= (ssize_t) image->rows))
2006 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
2008 q=QueueCacheViewAuthenticPixels(canvas_view,0,y,canvas_image->columns,1,
2010 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
2012 for (x=0; x < (ssize_t) source_image->columns; x++)
2014 if (((x_offset+x) < 0) || ((x_offset+x) >= (ssize_t) image->columns))
2016 p+=(ptrdiff_t) GetPixelChannels(source_image);
2022 offset.x=(double) (horizontal_scale*((
double) GetPixelRed(
2023 source_image,p)-(((MagickRealType) QuantumRange+1.0)/2.0)))/
2024 (((MagickRealType) QuantumRange+1.0)/2.0)+center.x+
2025 ((compose == DisplaceCompositeOp) ? x : 0);
2026 offset.y=(double) (vertical_scale*((
double) GetPixelGreen(
2027 source_image,p)-(((MagickRealType) QuantumRange+1.0)/2.0)))/
2028 (((MagickRealType) QuantumRange+1.0)/2.0)+center.y+
2029 ((compose == DisplaceCompositeOp) ? y : 0);
2030 status=InterpolatePixelInfo(image,image_view,
2031 UndefinedInterpolatePixel,(
double) offset.x,(
double) offset.y,
2033 if (status == MagickFalse)
2038 pixel.alpha=(MagickRealType) QuantumRange*(QuantumScale*pixel.alpha)*
2039 (QuantumScale*(double) GetPixelAlpha(source_image,p));
2040 SetPixelViaPixelInfo(canvas_image,&pixel,q);
2041 p+=(ptrdiff_t) GetPixelChannels(source_image);
2042 q+=(ptrdiff_t) GetPixelChannels(canvas_image);
2044 if (x < (ssize_t) source_image->columns)
2046 sync=SyncCacheViewAuthenticPixels(canvas_view,exception);
2047 if (sync == MagickFalse)
2050 canvas_view=DestroyCacheView(canvas_view);
2051 source_view=DestroyCacheView(source_view);
2052 image_view=DestroyCacheView(image_view);
2053 source_image=DestroyImage(source_image);
2054 source_image=canvas_image;
2057 case DissolveCompositeOp:
2062 artifact=GetImageArtifact(image,
"compose:args");
2063 if (artifact != (
char *) NULL)
2065 flags=ParseGeometry(artifact,&geometry_info);
2066 source_dissolve=geometry_info.rho/100.0;
2067 canvas_dissolve=1.0;
2068 if ((source_dissolve-MagickEpsilon) < 0.0)
2069 source_dissolve=0.0;
2070 if ((source_dissolve+MagickEpsilon) > 1.0)
2072 canvas_dissolve=2.0-source_dissolve;
2073 source_dissolve=1.0;
2075 if ((flags & SigmaValue) != 0)
2076 canvas_dissolve=geometry_info.sigma/100.0;
2077 if ((canvas_dissolve-MagickEpsilon) < 0.0)
2078 canvas_dissolve=0.0;
2079 if ((canvas_dissolve+MagickEpsilon) > 1.0)
2080 canvas_dissolve=1.0;
2084 case BlendCompositeOp:
2086 artifact=GetImageArtifact(image,
"compose:args");
2087 if (artifact != (
char *) NULL)
2089 flags=ParseGeometry(artifact,&geometry_info);
2090 source_dissolve=geometry_info.rho/100.0;
2091 canvas_dissolve=1.0-source_dissolve;
2092 if ((flags & SigmaValue) != 0)
2093 canvas_dissolve=geometry_info.sigma/100.0;
2097 case SaliencyBlendCompositeOp:
2100 residual_threshold = 0.0002,
2106 artifact=GetImageArtifact(image,
"compose:args");
2107 if (artifact != (
char *) NULL)
2109 flags=ParseGeometry(artifact,&geometry_info);
2110 iterations=geometry_info.rho;
2111 if ((flags & SigmaValue) != 0)
2112 residual_threshold=geometry_info.sigma;
2113 if ((flags & XiValue) != 0)
2114 tick=(size_t) geometry_info.xi;
2116 status=SaliencyBlendImage(image,composite,x_offset,y_offset,iterations,
2117 residual_threshold,tick,exception);
2118 source_image=DestroyImage(source_image);
2121 case SeamlessBlendCompositeOp:
2124 residual_threshold = 0.0002,
2130 artifact=GetImageArtifact(image,
"compose:args");
2131 if (artifact != (
char *) NULL)
2133 flags=ParseGeometry(artifact,&geometry_info);
2134 iterations=geometry_info.rho;
2135 if ((flags & SigmaValue) != 0)
2136 residual_threshold=geometry_info.sigma;
2137 if ((flags & XiValue) != 0)
2138 tick=(size_t) geometry_info.xi;
2140 status=SeamlessBlendImage(image,composite,x_offset,y_offset,iterations,
2141 residual_threshold,tick,exception);
2142 source_image=DestroyImage(source_image);
2145 case MathematicsCompositeOp:
2155 SetGeometryInfo(&geometry_info);
2156 artifact=GetImageArtifact(image,
"compose:args");
2157 if (artifact != (
char *) NULL)
2159 flags=ParseGeometry(artifact,&geometry_info);
2160 if (flags == NoValue)
2161 (void) ThrowMagickException(exception,GetMagickModule(),OptionError,
2162 "InvalidGeometry",
"`%s'",artifact);
2166 case ModulateCompositeOp:
2171 artifact=GetImageArtifact(image,
"compose:args");
2172 if (artifact != (
char *) NULL)
2174 flags=ParseGeometry(artifact,&geometry_info);
2175 percent_luma=geometry_info.rho;
2176 if ((flags & SigmaValue) != 0)
2177 percent_chroma=geometry_info.sigma;
2181 case ThresholdCompositeOp:
2186 artifact=GetImageArtifact(image,
"compose:args");
2187 if (artifact != (
char *) NULL)
2189 flags=ParseGeometry(artifact,&geometry_info);
2190 amount=geometry_info.rho;
2191 threshold=geometry_info.sigma;
2192 if ((flags & SigmaValue) == 0)
2195 threshold*=(double) QuantumRange;
2206 midpoint=((MagickRealType) QuantumRange+1.0)/2;
2207 source_view=AcquireVirtualCacheView(source_image,exception);
2208 image_view=AcquireAuthenticCacheView(image,exception);
2209#if defined(MAGICKCORE_OPENMP_SUPPORT)
2210 #pragma omp parallel for schedule(static) shared(progress,status) \
2211 magick_number_threads(source_image,image,image->rows,1)
2213 for (y=0; y < (ssize_t) image->rows; y++)
2239 if (status == MagickFalse)
2241 if (clip_to_self != MagickFalse)
2245 if ((y-(
double) y_offset) >= (double) source_image->rows)
2251 pixels=(Quantum *) NULL;
2253 if ((y >= y_offset) &&
2254 ((y-(double) y_offset) < (double) source_image->rows))
2256 p=GetCacheViewVirtualPixels(source_view,0,
2257 CastDoubleToLong(y-(
double) y_offset),source_image->columns,1,
2259 if (p == (
const Quantum *) NULL)
2266 p-=(ptrdiff_t)CastDoubleToLong((
double) x_offset*GetPixelChannels(source_image));
2268 q=GetCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
2269 if (q == (Quantum *) NULL)
2274 GetPixelInfo(image,&canvas_pixel);
2275 GetPixelInfo(source_image,&source_pixel);
2276 for (x=0; x < (ssize_t) image->columns; x++)
2300 if (clip_to_self != MagickFalse)
2304 q+=(ptrdiff_t) GetPixelChannels(image);
2307 if ((x-(
double) x_offset) >= (
double) source_image->columns)
2310 if ((pixels == (Quantum *) NULL) || (x < x_offset) ||
2311 ((x-(double) x_offset) >= (double) source_image->columns))
2314 source[MaxPixelChannels];
2321 (void) GetOneVirtualPixel(source_image,
2322 CastDoubleToLong(x-(
double) x_offset),
2323 CastDoubleToLong(y-(
double) y_offset),source,exception);
2324 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
2329 PixelChannel channel = GetPixelChannelChannel(image,i);
2330 PixelTrait traits = GetPixelChannelTraits(image,channel);
2331 PixelTrait source_traits = GetPixelChannelTraits(source_image,
2333 if ((traits == UndefinedPixelTrait) ||
2334 (source_traits == UndefinedPixelTrait))
2338 case AlphaCompositeOp:
2339 case ChangeMaskCompositeOp:
2340 case CopyAlphaCompositeOp:
2341 case DstAtopCompositeOp:
2342 case DstInCompositeOp:
2344 case OutCompositeOp:
2345 case SrcInCompositeOp:
2346 case SrcOutCompositeOp:
2348 if (channel == AlphaPixelChannel)
2349 pixel=(MagickRealType) TransparentAlpha;
2351 pixel=(MagickRealType) q[i];
2354 case ClearCompositeOp:
2355 case CopyCompositeOp:
2356 case ReplaceCompositeOp:
2357 case SrcCompositeOp:
2359 if (channel == AlphaPixelChannel)
2360 pixel=(MagickRealType) TransparentAlpha;
2365 case BlendCompositeOp:
2366 case DissolveCompositeOp:
2368 if (channel == AlphaPixelChannel)
2369 pixel=canvas_dissolve*(double) GetPixelAlpha(source_image,
2372 pixel=(MagickRealType) source[channel];
2377 pixel=(MagickRealType) source[channel];
2381 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
2382 ClampToQuantum(pixel);
2384 q+=(ptrdiff_t) GetPixelChannels(image);
2392 Sa=QuantumScale*(double) GetPixelAlpha(source_image,p);
2393 Da=QuantumScale*(double) GetPixelAlpha(image,q);
2396 case BumpmapCompositeOp:
2397 case ColorBurnCompositeOp:
2398 case ColorDodgeCompositeOp:
2399 case DarkenCompositeOp:
2400 case DifferenceCompositeOp:
2401 case DivideDstCompositeOp:
2402 case DivideSrcCompositeOp:
2403 case ExclusionCompositeOp:
2404 case FreezeCompositeOp:
2405 case HardLightCompositeOp:
2406 case HardMixCompositeOp:
2407 case InterpolateCompositeOp:
2408 case LightenCompositeOp:
2409 case LinearBurnCompositeOp:
2410 case LinearDodgeCompositeOp:
2411 case LinearLightCompositeOp:
2412 case MathematicsCompositeOp:
2413 case MinusDstCompositeOp:
2414 case MinusSrcCompositeOp:
2415 case MultiplyCompositeOp:
2416 case NegateCompositeOp:
2417 case OverlayCompositeOp:
2418 case PegtopLightCompositeOp:
2419 case PinLightCompositeOp:
2420 case ReflectCompositeOp:
2421 case ScreenCompositeOp:
2422 case SoftBurnCompositeOp:
2423 case SoftDodgeCompositeOp:
2424 case SoftLightCompositeOp:
2425 case StampCompositeOp:
2426 case VividLightCompositeOp:
2428 alpha=RoundToUnity(Sa+Da-Sa*Da);
2431 case DstAtopCompositeOp:
2432 case DstInCompositeOp:
2434 case SrcInCompositeOp:
2439 case DissolveCompositeOp:
2441 alpha=source_dissolve*Sa*(-canvas_dissolve*Da)+source_dissolve*Sa+
2445 case DstOverCompositeOp:
2446 case OverCompositeOp:
2447 case SrcOverCompositeOp:
2452 case DstOutCompositeOp:
2457 case OutCompositeOp:
2458 case SrcOutCompositeOp:
2463 case BlendCompositeOp:
2464 case PlusCompositeOp:
2466 alpha=RoundToUnity(source_dissolve*Sa+canvas_dissolve*Da);
2469 case XorCompositeOp:
2471 alpha=Sa+Da-2.0*Sa*Da;
2474 case ModulusAddCompositeOp:
2481 alpha=((Sa+Da)-1.0);
2484 case ModulusSubtractCompositeOp:
2491 alpha=((Sa-Da)+1.0);
2502 case ColorizeCompositeOp:
2503 case HueCompositeOp:
2504 case LuminizeCompositeOp:
2505 case ModulateCompositeOp:
2506 case RMSECompositeOp:
2507 case SaturateCompositeOp:
2509 Si=GetPixelIntensity(source_image,p);
2510 GetPixelInfoPixel(source_image,p,&source_pixel);
2511 GetPixelInfoPixel(image,q,&canvas_pixel);
2514 case BumpmapCompositeOp:
2515 case CopyAlphaCompositeOp:
2516 case DarkenIntensityCompositeOp:
2517 case LightenIntensityCompositeOp:
2519 Si=GetPixelIntensity(source_image,p);
2520 Di=GetPixelIntensity(image,q);
2526 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
2532 PixelChannel channel = GetPixelChannelChannel(image,i);
2533 PixelTrait traits = GetPixelChannelTraits(image,channel);
2534 PixelTrait source_traits = GetPixelChannelTraits(source_image,channel);
2535 if (traits == UndefinedPixelTrait)
2537 if ((channel == AlphaPixelChannel) &&
2538 ((traits & UpdatePixelTrait) != 0))
2545 case AlphaCompositeOp:
2547 pixel=(double) QuantumRange*Sa;
2550 case AtopCompositeOp:
2551 case CopyBlackCompositeOp:
2552 case CopyBlueCompositeOp:
2553 case CopyCyanCompositeOp:
2554 case CopyGreenCompositeOp:
2555 case CopyMagentaCompositeOp:
2556 case CopyRedCompositeOp:
2557 case CopyYellowCompositeOp:
2558 case SrcAtopCompositeOp:
2559 case DstCompositeOp:
2562 pixel=(double) QuantumRange*Da;
2565 case BumpmapCompositeOp:
2570 case ChangeMaskCompositeOp:
2572 if (IsFuzzyEquivalencePixel(source_image,p,image,q) != MagickFalse)
2573 pixel=(MagickRealType) TransparentAlpha;
2575 pixel=(double) QuantumRange*Da;
2578 case ClearCompositeOp:
2580 pixel=(MagickRealType) TransparentAlpha;
2583 case ColorizeCompositeOp:
2584 case HueCompositeOp:
2585 case LuminizeCompositeOp:
2586 case RMSECompositeOp:
2587 case SaturateCompositeOp:
2589 if (fabs((
double) QuantumRange*Sa-(double) TransparentAlpha) < MagickEpsilon)
2591 pixel=(double) QuantumRange*Da;
2594 if (fabs((
double) QuantumRange*Da-(double) TransparentAlpha) < MagickEpsilon)
2596 pixel=(double) QuantumRange*Sa;
2601 pixel=(double) QuantumRange*Da;
2604 pixel=(double) QuantumRange*Sa;
2607 case CopyAlphaCompositeOp:
2609 if (source_image->alpha_trait == UndefinedPixelTrait)
2612 pixel=(double) QuantumRange*Sa;
2615 case BlurCompositeOp:
2616 case CopyCompositeOp:
2617 case DisplaceCompositeOp:
2618 case DistortCompositeOp:
2619 case DstAtopCompositeOp:
2620 case ReplaceCompositeOp:
2621 case SrcCompositeOp:
2623 pixel=(double) QuantumRange*Sa;
2626 case DarkenIntensityCompositeOp:
2628 if (compose_sync == MagickFalse)
2630 pixel=Si < Di? Sa : Da;
2633 pixel=Sa*Si < Da*Di ? Sa : Da;
2636 case DifferenceCompositeOp:
2638 pixel=(double) QuantumRange*fabs((
double) (Sa-Da));
2641 case FreezeCompositeOp:
2643 pixel=(double) QuantumRange*(1.0-(1.0-Sa)*(1.0-Sa)*
2644 PerceptibleReciprocal(Da));
2649 case InterpolateCompositeOp:
2651 pixel=(double) QuantumRange*(0.5-0.25*cos(MagickPI*Sa)-0.25*
2655 case LightenIntensityCompositeOp:
2657 if (compose_sync == MagickFalse)
2659 pixel=Si > Di ? Sa : Da;
2662 pixel=Sa*Si > Da*Di ? Sa : Da;
2665 case ModulateCompositeOp:
2667 pixel=(double) QuantumRange*Da;
2670 case MultiplyCompositeOp:
2672 if (compose_sync == MagickFalse)
2674 pixel=(double) QuantumRange*Sa*Da;
2677 pixel=(double) QuantumRange*alpha;
2680 case NegateCompositeOp:
2682 pixel=(double) QuantumRange*((1.0-Sa-Da));
2685 case ReflectCompositeOp:
2687 pixel=(double) QuantumRange*(Sa*Sa*
2688 PerceptibleReciprocal(1.0-Da));
2689 if (pixel > (
double) QuantumRange)
2690 pixel=(
double) QuantumRange;
2693 case StampCompositeOp:
2695 pixel=(double) QuantumRange*(Sa+Da*Da-1.0);
2698 case StereoCompositeOp:
2700 pixel=(double) QuantumRange*(Sa+Da)/2;
2705 pixel=(double) QuantumRange*alpha;
2709 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
2710 ClampToQuantum(pixel);
2713 if (source_traits == UndefinedPixelTrait)
2719 Sc=(MagickRealType) GetPixelChannel(source_image,channel,p);
2720 Dc=(MagickRealType) q[i];
2721 if ((traits & CopyPixelTrait) != 0)
2726 q[i]=ClampToQuantum(Dc);
2734 Sca=QuantumScale*Sa*Sc;
2735 Dca=QuantumScale*Da*Dc;
2736 SaSca=Sa*PerceptibleReciprocal(Sca);
2737 DcaDa=Dca*PerceptibleReciprocal(Da);
2740 case DarkenCompositeOp:
2741 case LightenCompositeOp:
2742 case ModulusSubtractCompositeOp:
2744 gamma=PerceptibleReciprocal(1.0-alpha);
2749 gamma=PerceptibleReciprocal(alpha);
2756 case AlphaCompositeOp:
2758 pixel=(double) QuantumRange*Sa;
2761 case AtopCompositeOp:
2762 case SrcAtopCompositeOp:
2764 pixel=(double) QuantumRange*(Sca*Da+Dca*(1.0-Sa));
2767 case BlendCompositeOp:
2769 pixel=gamma*(source_dissolve*Sa*Sc+canvas_dissolve*Da*Dc);
2772 case CopyCompositeOp:
2773 case ReplaceCompositeOp:
2774 case SrcCompositeOp:
2776 pixel=(double) QuantumRange*Sca;
2779 case BlurCompositeOp:
2780 case DisplaceCompositeOp:
2781 case DistortCompositeOp:
2786 case BumpmapCompositeOp:
2788 if (fabs((
double) QuantumRange*Sa-(double) TransparentAlpha) < MagickEpsilon)
2793 pixel=(double) QuantumScale*Si*Dc;
2796 case ChangeMaskCompositeOp:
2801 case ClearCompositeOp:
2806 case ColorBurnCompositeOp:
2808 if ((Sca == 0.0) && (Dca == Da))
2810 pixel=(double) QuantumRange*gamma*(Sa*Da+Dca*(1.0-Sa));
2815 pixel=(double) QuantumRange*gamma*(Dca*(1.0-Sa));
2818 pixel=(double) QuantumRange*gamma*(Sa*Da-Sa*Da*MagickMin(1.0,
2819 (1.0-DcaDa)*SaSca)+Sca*(1.0-Da)+Dca*(1.0-Sa));
2822 case ColorDodgeCompositeOp:
2824 if ((Sca*Da+Dca*Sa) >= Sa*Da)
2825 pixel=(
double) QuantumRange*gamma*(Sa*Da+Sca*(1.0-Da)+Dca*
2828 pixel=(double) QuantumRange*gamma*(Dca*Sa*Sa*
2829 PerceptibleReciprocal(Sa-Sca)+Sca*(1.0-Da)+Dca*(1.0-Sa));
2832 case ColorizeCompositeOp:
2834 if (fabs((
double) QuantumRange*Sa-(double) TransparentAlpha) < MagickEpsilon)
2839 if (fabs((
double) QuantumRange*Da-(double) TransparentAlpha) < MagickEpsilon)
2844 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
2845 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
2846 white_luminance,illuminant,&sans,&sans,&luma);
2847 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
2848 (
double) source_pixel.green,(
double) source_pixel.blue,
2849 white_luminance,illuminant,&hue,&chroma,&sans);
2850 ConvertGenericToRGB(colorspace,hue,chroma,luma,
2851 white_luminance,illuminant,&red,&green,&blue);
2854 case RedPixelChannel: pixel=red;
break;
2855 case GreenPixelChannel: pixel=green;
break;
2856 case BluePixelChannel: pixel=blue;
break;
2857 default: pixel=Dc;
break;
2861 case CopyAlphaCompositeOp:
2866 case CopyBlackCompositeOp:
2868 if (channel == BlackPixelChannel)
2869 pixel=(MagickRealType) GetPixelBlack(source_image,p);
2872 case CopyBlueCompositeOp:
2873 case CopyYellowCompositeOp:
2875 if (channel == BluePixelChannel)
2876 pixel=(MagickRealType) GetPixelBlue(source_image,p);
2879 case CopyGreenCompositeOp:
2880 case CopyMagentaCompositeOp:
2882 if (channel == GreenPixelChannel)
2883 pixel=(MagickRealType) GetPixelGreen(source_image,p);
2886 case CopyRedCompositeOp:
2887 case CopyCyanCompositeOp:
2889 if (channel == RedPixelChannel)
2890 pixel=(MagickRealType) GetPixelRed(source_image,p);
2893 case DarkenCompositeOp:
2900 if (compose_sync == MagickFalse)
2902 pixel=MagickMin(Sc,Dc);
2905 if ((Sca*Da) < (Dca*Sa))
2907 pixel=(double) QuantumRange*(Sca+Dca*(1.0-Sa));
2910 pixel=(double) QuantumRange*(Dca+Sca*(1.0-Da));
2913 case DarkenIntensityCompositeOp:
2915 if (compose_sync == MagickFalse)
2917 pixel=Si < Di ? Sc : Dc;
2920 pixel=Sa*Si < Da*Di ? Sc : Dc;
2923 case DifferenceCompositeOp:
2925 if (compose_sync == MagickFalse)
2927 pixel=fabs((
double) Sc-Dc);
2930 pixel=(double) QuantumRange*gamma*(Sca+Dca-2.0*MagickMin(Sca*Da,
2934 case DissolveCompositeOp:
2936 pixel=gamma*(source_dissolve*Sa*Sc-source_dissolve*Sa*
2937 canvas_dissolve*Da*Dc+canvas_dissolve*Da*Dc);
2940 case DivideDstCompositeOp:
2942 if (compose_sync == MagickFalse)
2944 pixel=(double) QuantumRange*(Sc/PerceptibleReciprocal(Dc));
2947 if ((fabs((
double) Sca) < MagickEpsilon) &&
2948 (fabs((
double) Dca) < MagickEpsilon))
2950 pixel=(double) QuantumRange*gamma*(Sca*(1.0-Da)+Dca*(1.0-Sa));
2953 if (fabs((
double) Dca) < MagickEpsilon)
2955 pixel=(double) QuantumRange*gamma*(Sa*Da+Sca*(1.0-Da)+Dca*
2959 pixel=(double) QuantumRange*gamma*(Sca*Da*Da/Dca+Sca*(1.0-Da)+Dca*
2963 case DivideSrcCompositeOp:
2965 if (compose_sync == MagickFalse)
2967 pixel=(double) QuantumRange*(Dc/PerceptibleReciprocal(Sc));
2970 if ((fabs((
double) Dca) < MagickEpsilon) &&
2971 (fabs((
double) Sca) < MagickEpsilon))
2973 pixel=(double) QuantumRange*gamma*(Dca*(1.0-Sa)+Sca*(1.0-Da));
2976 if (fabs((
double) Sca) < MagickEpsilon)
2978 pixel=(double) QuantumRange*gamma*(Da*Sa+Dca*(1.0-Sa)+Sca*
2982 pixel=(double) QuantumRange*gamma*(Dca*Sa*SaSca+Dca*(1.0-Sa)+Sca*
2986 case DstAtopCompositeOp:
2988 pixel=(double) QuantumRange*(Dca*Sa+Sca*(1.0-Da));
2991 case DstCompositeOp:
2994 pixel=(double) QuantumRange*Dca;
2997 case DstInCompositeOp:
2999 pixel=(double) QuantumRange*gamma*(Dca*Sa);
3002 case DstOutCompositeOp:
3004 pixel=(double) QuantumRange*gamma*(Dca*(1.0-Sa));
3007 case DstOverCompositeOp:
3009 pixel=(double) QuantumRange*gamma*(Dca+Sca*(1.0-Da));
3012 case ExclusionCompositeOp:
3014 pixel=(double) QuantumRange*gamma*(Sca*Da+Dca*Sa-2.0*Sca*Dca+Sca*
3015 (1.0-Da)+Dca*(1.0-Sa));
3018 case FreezeCompositeOp:
3020 pixel=(double) QuantumRange*gamma*(1.0-(1.0-Sca)*(1.0-Sca)*
3021 PerceptibleReciprocal(Dca));
3026 case HardLightCompositeOp:
3030 pixel=(double) QuantumRange*gamma*(2.0*Sca*Dca+Sca*(1.0-Da)+Dca*
3034 pixel=(double) QuantumRange*gamma*(Sa*Da-2.0*(Da-Dca)*(Sa-Sca)+Sca*
3035 (1.0-Da)+Dca*(1.0-Sa));
3038 case HardMixCompositeOp:
3040 pixel=gamma*(((Sca+Dca) < 1.0) ? 0.0 : (double) QuantumRange);
3043 case HueCompositeOp:
3045 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
3050 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
3055 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3056 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3057 white_luminance,illuminant,&hue,&chroma,&luma);
3058 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
3059 (
double) source_pixel.green,(
double) source_pixel.blue,
3060 white_luminance,illuminant,&hue,&sans,&sans);
3061 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3062 white_luminance,illuminant,&red,&green,&blue);
3065 case RedPixelChannel: pixel=red;
break;
3066 case GreenPixelChannel: pixel=green;
break;
3067 case BluePixelChannel: pixel=blue;
break;
3068 default: pixel=Dc;
break;
3073 case SrcInCompositeOp:
3075 pixel=(double) QuantumRange*(Sca*Da);
3078 case InterpolateCompositeOp:
3080 pixel=(double) QuantumRange*(0.5-0.25*cos(MagickPI*Sca)-0.25*
3084 case LinearBurnCompositeOp:
3092 pixel=(double) QuantumRange*gamma*(Sca+Dca-Sa*Da);
3095 case LinearDodgeCompositeOp:
3097 pixel=gamma*(Sa*Sc+Da*Dc);
3100 case LinearLightCompositeOp:
3108 pixel=(double) QuantumRange*gamma*((Sca-Sa)*Da+Sca+Dca);
3111 case LightenCompositeOp:
3113 if (compose_sync == MagickFalse)
3115 pixel=MagickMax(Sc,Dc);
3118 if ((Sca*Da) > (Dca*Sa))
3120 pixel=(double) QuantumRange*(Sca+Dca*(1.0-Sa));
3123 pixel=(double) QuantumRange*(Dca+Sca*(1.0-Da));
3126 case LightenIntensityCompositeOp:
3133 if (compose_sync == MagickFalse)
3135 pixel=Si > Di ? Sc : Dc;
3138 pixel=Sa*Si > Da*Di ? Sc : Dc;
3141 case LuminizeCompositeOp:
3143 if (fabs((
double) QuantumRange*Sa-(double) TransparentAlpha) < MagickEpsilon)
3148 if (fabs((
double) QuantumRange*Da-(double) TransparentAlpha) < MagickEpsilon)
3153 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3154 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3155 white_luminance,illuminant,&hue,&chroma,&luma);
3156 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
3157 (
double) source_pixel.green,(
double) source_pixel.blue,
3158 white_luminance,illuminant,&sans,&sans,&luma);
3159 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3160 white_luminance,illuminant,&red,&green,&blue);
3163 case RedPixelChannel: pixel=red;
break;
3164 case GreenPixelChannel: pixel=green;
break;
3165 case BluePixelChannel: pixel=blue;
break;
3166 default: pixel=Dc;
break;
3170 case MathematicsCompositeOp:
3191 if (compose_sync == MagickFalse)
3193 pixel=geometry_info.rho*Sc*Dc+geometry_info.sigma*Sc+
3194 geometry_info.xi*Dc+geometry_info.psi;
3197 pixel=(double) QuantumRange*gamma*(geometry_info.rho*Sca*Dca+
3198 geometry_info.sigma*Sca*Da+geometry_info.xi*Dca*Sa+
3199 geometry_info.psi*Sa*Da+Sca*(1.0-Da)+Dca*(1.0-Sa));
3202 case MinusDstCompositeOp:
3204 if (compose_sync == MagickFalse)
3209 pixel=gamma*(Sa*Sc+Da*Dc-2.0*Da*Dc*Sa);
3212 case MinusSrcCompositeOp:
3219 if (compose_sync == MagickFalse)
3224 pixel=gamma*(Da*Dc+Sa*Sc-2.0*Sa*Sc*Da);
3227 case ModulateCompositeOp:
3232 if (fabs((
double) QuantumRange*Sa-(double) TransparentAlpha) < MagickEpsilon)
3237 offset=(ssize_t) (Si-midpoint);
3243 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3244 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3245 white_luminance,illuminant,&hue,&chroma,&luma);
3246 luma+=(0.01*percent_luma*offset)/midpoint;
3247 chroma*=0.01*percent_chroma;
3248 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3249 white_luminance,illuminant,&red,&green,&blue);
3252 case RedPixelChannel: pixel=red;
break;
3253 case GreenPixelChannel: pixel=green;
break;
3254 case BluePixelChannel: pixel=blue;
break;
3255 default: pixel=Dc;
break;
3259 case ModulusAddCompositeOp:
3261 if (compose_sync == MagickFalse)
3266 if ((Sca+Dca) <= 1.0)
3268 pixel=(double) QuantumRange*(Sca+Dca);
3271 pixel=(double) QuantumRange*((Sca+Dca)-1.0);
3274 case ModulusSubtractCompositeOp:
3276 if (compose_sync == MagickFalse)
3281 if ((Sca-Dca) >= 0.0)
3283 pixel=(double) QuantumRange*(Sca-Dca);
3286 pixel=(double) QuantumRange*((Sca-Dca)+1.0);
3289 case MultiplyCompositeOp:
3291 if (compose_sync == MagickFalse)
3293 pixel=(double) QuantumScale*Dc*Sc;
3296 pixel=(double) QuantumRange*gamma*(Sca*Dca+Sca*(1.0-Da)+Dca*
3300 case NegateCompositeOp:
3302 pixel=(double) QuantumRange*(1.0-fabs(1.0-Sca-Dca));
3305 case OutCompositeOp:
3306 case SrcOutCompositeOp:
3308 pixel=(double) QuantumRange*(Sca*(1.0-Da));
3311 case OverCompositeOp:
3312 case SrcOverCompositeOp:
3314 pixel=(double) QuantumRange*gamma*(Sca+Dca*(1.0-Sa));
3317 case OverlayCompositeOp:
3321 pixel=(double) QuantumRange*gamma*(2.0*Dca*Sca+Dca*(1.0-Sa)+
3325 pixel=(double) QuantumRange*gamma*(Da*Sa-2.0*(Sa-Sca)*(Da-Dca)+Dca*
3326 (1.0-Sa)+Sca*(1.0-Da));
3329 case PegtopLightCompositeOp:
3339 if (fabs((
double) Da) < MagickEpsilon)
3341 pixel=(double) QuantumRange*gamma*Sca;
3344 pixel=(double) QuantumRange*gamma*(Dca*Dca*(Sa-2.0*Sca)/Da+Sca*
3345 (2.0*Dca+1.0-Da)+Dca*(1.0-Sa));
3348 case PinLightCompositeOp:
3356 if ((Dca*Sa) < (Da*(2.0*Sca-Sa)))
3358 pixel=(double) QuantumRange*gamma*(Sca*(Da+1.0)-Sa*Da+Dca*
3362 if ((Dca*Sa) > (2.0*Sca*Da))
3364 pixel=(double) QuantumRange*gamma*(Sca*Da+Sca+Dca*(1.0-Sa));
3367 pixel=(double) QuantumRange*gamma*(Sca*(1.0-Da)+Dca);
3370 case PlusCompositeOp:
3372 if (compose_sync == MagickFalse)
3377 pixel=(double) QuantumRange*(Sca+Dca);
3380 case ReflectCompositeOp:
3382 pixel=(double) QuantumRange*gamma*(Sca*Sca*
3383 PerceptibleReciprocal(1.0-Dca));
3384 if (pixel > (
double) QuantumRange)
3385 pixel=(
double) QuantumRange;
3388 case RMSECompositeOp:
3393 if (fabs((
double) QuantumRange*Sa-(double) TransparentAlpha) < MagickEpsilon)
3398 if (fabs((
double) QuantumRange*Da-(double) TransparentAlpha) < MagickEpsilon)
3404 (canvas_pixel.red-source_pixel.red)*
3405 (canvas_pixel.red-source_pixel.red)+
3406 (canvas_pixel.green-source_pixel.green)*
3407 (canvas_pixel.green-source_pixel.green)+
3408 (canvas_pixel.blue-source_pixel.blue)*
3409 (canvas_pixel.blue-source_pixel.blue)/3.0);
3412 case RedPixelChannel: pixel=gray;
break;
3413 case GreenPixelChannel: pixel=gray;
break;
3414 case BluePixelChannel: pixel=gray;
break;
3415 default: pixel=Dc;
break;
3419 case SaturateCompositeOp:
3421 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
3426 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
3431 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3432 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3433 white_luminance,illuminant,&hue,&chroma,&luma);
3434 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
3435 (
double) source_pixel.green,(
double) source_pixel.blue,
3436 white_luminance,illuminant,&sans,&chroma,&sans);
3437 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3438 white_luminance,illuminant,&red,&green,&blue);
3441 case RedPixelChannel: pixel=red;
break;
3442 case GreenPixelChannel: pixel=green;
break;
3443 case BluePixelChannel: pixel=blue;
break;
3444 default: pixel=Dc;
break;
3448 case ScreenCompositeOp:
3455 if (compose_sync == MagickFalse)
3460 pixel=(double) QuantumRange*gamma*(Sca+Dca-Sca*Dca);
3463 case SoftBurnCompositeOp:
3465 if ((Sca+Dca) < 1.0)
3466 pixel=(
double) QuantumRange*gamma*(0.5*Dca*
3467 PerceptibleReciprocal(1.0-Sca));
3469 pixel=(double) QuantumRange*gamma*(1.0-0.5*(1.0-Sca)*
3470 PerceptibleReciprocal(Dca));
3473 case SoftDodgeCompositeOp:
3475 if ((Sca+Dca) < 1.0)
3476 pixel=(
double) QuantumRange*gamma*(0.5*Sca*
3477 PerceptibleReciprocal(1.0-Dca));
3479 pixel=(double) QuantumRange*gamma*(1.0-0.5*(1.0-Dca)*
3480 PerceptibleReciprocal(Sca));
3483 case SoftLightCompositeOp:
3487 pixel=(double) QuantumRange*gamma*(Dca*(Sa+(2.0*Sca-Sa)*
3488 (1.0-DcaDa))+Sca*(1.0-Da)+Dca*(1.0-Sa));
3491 if (((2.0*Sca) > Sa) && ((4.0*Dca) <= Da))
3493 pixel=(double) QuantumRange*gamma*(Dca*Sa+Da*(2.0*Sca-Sa)*
3494 (4.0*DcaDa*(4.0*DcaDa+1.0)*(DcaDa-1.0)+7.0*DcaDa)+Sca*
3495 (1.0-Da)+Dca*(1.0-Sa));
3498 pixel=(double) QuantumRange*gamma*(Dca*Sa+Da*(2.0*Sca-Sa)*
3499 (pow(DcaDa,0.5)-DcaDa)+Sca*(1.0-Da)+Dca*(1.0-Sa));
3502 case StampCompositeOp:
3504 pixel=(double) QuantumRange*(Sca+Dca*Dca-1.0);
3507 case StereoCompositeOp:
3509 if (channel == RedPixelChannel)
3510 pixel=(MagickRealType) GetPixelRed(source_image,p);
3513 case ThresholdCompositeOp:
3519 if ((MagickRealType) fabs((
double) (2.0*delta)) < threshold)
3524 pixel=gamma*(Dc+delta*amount);
3527 case VividLightCompositeOp:
3535 if ((fabs((
double) Sa) < MagickEpsilon) ||
3536 (fabs((
double) (Sca-Sa)) < MagickEpsilon))
3538 pixel=(double) QuantumRange*gamma*(Sa*Da+Sca*(1.0-Da)+Dca*
3542 if ((2.0*Sca) <= Sa)
3544 pixel=(double) QuantumRange*gamma*(Sa*(Da+Sa*(Dca-Da)*
3545 PerceptibleReciprocal(2.0*Sca))+Sca*(1.0-Da)+Dca*(1.0-Sa));
3548 pixel=(double) QuantumRange*gamma*(Dca*Sa*Sa*
3549 PerceptibleReciprocal(2.0*(Sa-Sca))+Sca*(1.0-Da)+Dca*(1.0-Sa));
3552 case XorCompositeOp:
3554 pixel=(double) QuantumRange*(Sca*(1.0-Da)+Dca*(1.0-Sa));
3563 q[i]=clamp != MagickFalse ? ClampPixel(pixel) : ClampToQuantum(pixel);
3565 p+=(ptrdiff_t) GetPixelChannels(source_image);
3566 channels=GetPixelChannels(source_image);
3567 if (p >= (pixels+channels*source_image->columns))
3569 q+=(ptrdiff_t) GetPixelChannels(image);
3571 if (SyncCacheViewAuthenticPixels(image_view,exception) == MagickFalse)
3573 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3578#if defined(MAGICKCORE_OPENMP_SUPPORT)
3582 proceed=SetImageProgress(image,CompositeImageTag,progress,image->rows);
3583 if (proceed == MagickFalse)
3587 source_view=DestroyCacheView(source_view);
3588 image_view=DestroyCacheView(image_view);
3589 if (canvas_image != (
Image * ) NULL)
3590 canvas_image=DestroyImage(canvas_image);
3592 source_image=DestroyImage(source_image);
3622MagickExport MagickBooleanType TextureImage(
Image *image,
const Image *texture,
3625#define TextureImageTag "Texture/Image"
3640 assert(image != (
Image *) NULL);
3641 assert(image->signature == MagickCoreSignature);
3642 if (IsEventLogging() != MagickFalse)
3643 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
3644 if (texture == (
const Image *) NULL)
3645 return(MagickFalse);
3646 if (SetImageStorageClass(image,DirectClass,exception) == MagickFalse)
3647 return(MagickFalse);
3648 texture_image=CloneImage(texture,0,0,MagickTrue,exception);
3649 if (texture_image == (
const Image *) NULL)
3650 return(MagickFalse);
3651 (void) TransformImageColorspace(texture_image,image->colorspace,exception);
3652 (void) SetImageVirtualPixelMethod(texture_image,TileVirtualPixelMethod,
3655 if ((image->compose != CopyCompositeOp) &&
3656 ((image->compose != OverCompositeOp) ||
3657 (image->alpha_trait != UndefinedPixelTrait) ||
3658 (texture_image->alpha_trait != UndefinedPixelTrait)))
3663 for (y=0; y < (ssize_t) image->rows; y+=(ssize_t) texture_image->rows)
3668 if (status == MagickFalse)
3670 for (x=0; x < (ssize_t) image->columns; x+=(ssize_t) texture_image->columns)
3675 thread_status=CompositeImage(image,texture_image,image->compose,
3676 MagickTrue,x+texture_image->tile_offset.x,y+
3677 texture_image->tile_offset.y,exception);
3678 if (thread_status == MagickFalse)
3680 status=thread_status;
3684 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3689 proceed=SetImageProgress(image,TextureImageTag,(MagickOffsetType) y,
3691 if (proceed == MagickFalse)
3695 (void) SetImageProgress(image,TextureImageTag,(MagickOffsetType)
3696 image->rows,image->rows);
3697 texture_image=DestroyImage(texture_image);
3704 texture_view=AcquireVirtualCacheView(texture_image,exception);
3705 image_view=AcquireAuthenticCacheView(image,exception);
3706#if defined(MAGICKCORE_OPENMP_SUPPORT)
3707 #pragma omp parallel for schedule(static) shared(status) \
3708 magick_number_threads(texture_image,image,image->rows,2)
3710 for (y=0; y < (ssize_t) image->rows; y++)
3728 if (status == MagickFalse)
3730 pixels=GetCacheViewVirtualPixels(texture_view,texture_image->tile_offset.x,
3731 (y+texture_image->tile_offset.y) % (ssize_t) texture_image->rows,
3732 texture_image->columns,1,exception);
3733 q=QueueCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
3734 if ((pixels == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
3739 for (x=0; x < (ssize_t) image->columns; x+=(ssize_t) texture_image->columns)
3745 width=texture_image->columns;
3746 if ((x+(ssize_t) width) > (ssize_t) image->columns)
3747 width=image->columns-(
size_t) x;
3748 for (j=0; j < (ssize_t) width; j++)
3753 for (i=0; i < (ssize_t) GetPixelChannels(texture_image); i++)
3755 PixelChannel channel = GetPixelChannelChannel(texture_image,i);
3756 PixelTrait traits = GetPixelChannelTraits(image,channel);
3757 PixelTrait texture_traits=GetPixelChannelTraits(texture_image,
3759 if ((traits == UndefinedPixelTrait) ||
3760 (texture_traits == UndefinedPixelTrait))
3762 SetPixelChannel(image,channel,p[i],q);
3764 p+=(ptrdiff_t) GetPixelChannels(texture_image);
3765 q+=(ptrdiff_t) GetPixelChannels(image);
3768 sync=SyncCacheViewAuthenticPixels(image_view,exception);
3769 if (sync == MagickFalse)
3771 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3776 proceed=SetImageProgress(image,TextureImageTag,(MagickOffsetType) y,
3778 if (proceed == MagickFalse)
3782 texture_view=DestroyCacheView(texture_view);
3783 image_view=DestroyCacheView(image_view);
3784 texture_image=DestroyImage(texture_image);