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,
186 ExceptionInfo *exception)
198 kernel_info=AcquireKernelInfo(kernel,exception);
199 if (kernel_info == (KernelInfo *) NULL)
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,
215 ExceptionInfo *exception)
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,
301 const Image *beta_image,
const Image *dx_image,
const Image *dy_image,
302 ExceptionInfo *exception)
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,
491 const Image *beta_image,ExceptionInfo *exception)
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,
617 const Image *mask_image,ExceptionInfo *exception)
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);
681static Image *BlendMeanImage(Image *image,
const Image *mask_image,
682 ExceptionInfo *exception)
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,
809 const Image *beta_image,
double *residual,ExceptionInfo *exception)
817 channel_residual = 0.0;
823 columns = MagickMax(alpha_image->columns,beta_image->columns),
824 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 reduction(+:area) reduction(+:channel_residual) \
834 magick_number_threads(alpha_image,alpha_image,rows,1)
836 for (y=0; y < (ssize_t) rows; y++)
845 if (status == MagickFalse)
847 p=GetCacheViewVirtualPixels(alpha_view,0,y,columns,1,exception);
848 q=GetCacheViewVirtualPixels(beta_view,0,y,columns,1,exception);
849 if ((p == (
const Quantum *) NULL) || (q == (
const Quantum *) NULL))
854 channel_residual=0.0;
855 for (x=0; x < (ssize_t) columns; x++)
864 if ((GetPixelReadMask(alpha_image,p) <= (QuantumRange/2)) ||
865 (GetPixelReadMask(beta_image,q) <= (QuantumRange/2)))
867 p+=(ptrdiff_t) GetPixelChannels(alpha_image);
868 q+=(ptrdiff_t) GetPixelChannels(beta_image);
871 Sa=QuantumScale*(double) GetPixelAlpha(alpha_image,p);
872 Da=QuantumScale*(double) GetPixelAlpha(beta_image,q);
873 for (i=0; i < (ssize_t) GetPixelChannels(alpha_image); i++)
878 PixelChannel channel = GetPixelChannelChannel(alpha_image,i);
879 PixelTrait traits = GetPixelChannelTraits(alpha_image,channel);
880 PixelTrait beta_traits = GetPixelChannelTraits(beta_image,channel);
881 if ((traits == UndefinedPixelTrait) ||
882 (beta_traits == UndefinedPixelTrait) ||
883 ((beta_traits & UpdatePixelTrait) == 0))
885 if (channel == AlphaPixelChannel)
886 distance=QuantumScale*((double) p[i]-(double) GetPixelChannel(
887 beta_image,channel,q));
889 distance=QuantumScale*(Sa*(double) p[i]-Da*(double) GetPixelChannel(
890 beta_image,channel,q));
891 channel_residual+=distance*distance;
894 p+=(ptrdiff_t) GetPixelChannels(alpha_image);
895 q+=(ptrdiff_t) GetPixelChannels(beta_image);
898 beta_view=DestroyCacheView(beta_view);
899 alpha_view=DestroyCacheView(alpha_view);
900 area=MagickSafeReciprocal(area);
901 *residual=sqrt(area*channel_residual/(
double) GetImageChannels(alpha_image));
905static MagickBooleanType CompositeOverImage(Image *image,
906 const Image *source_image,
const MagickBooleanType clip_to_self,
907 const ssize_t x_offset,
const ssize_t y_offset,ExceptionInfo *exception)
909#define CompositeImageTag "Composite/Image"
934 value=GetImageArtifact(image,
"compose:clamp");
935 if (value != (
const char *) NULL)
936 clamp=IsStringTrue(value);
939 source_view=AcquireVirtualCacheView(source_image,exception);
940 image_view=AcquireAuthenticCacheView(image,exception);
941#if defined(MAGICKCORE_OPENMP_SUPPORT)
942 #pragma omp parallel for schedule(static) shared(progress,status) \
943 magick_number_threads(source_image,image,image->rows,1)
945 for (y=0; y < (ssize_t) image->rows; y++)
963 if (status == MagickFalse)
965 if (clip_to_self != MagickFalse)
969 if ((y-y_offset) >= (ssize_t) source_image->rows)
975 pixels=(Quantum *) NULL;
977 if ((y >= y_offset) &&
978 ((y-y_offset) < (ssize_t) source_image->rows))
980 p=GetCacheViewVirtualPixels(source_view,0,CastDoubleToSsizeT((
double) y-
981 y_offset),source_image->columns,1,exception);
982 if (p == (
const Quantum *) NULL)
989 p-=(ptrdiff_t) CastDoubleToSsizeT((
double) x_offset*
990 GetPixelChannels(source_image));
992 q=GetCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
993 if (q == (Quantum *) NULL)
998 GetPixelInfo(image,&canvas_pixel);
999 GetPixelInfo(source_image,&source_pixel);
1000 for (x=0; x < (ssize_t) image->columns; x++)
1020 if (clip_to_self != MagickFalse)
1024 q+=(ptrdiff_t) GetPixelChannels(image);
1027 if ((x-x_offset) >= (ssize_t) source_image->columns)
1030 if ((pixels == (Quantum *) NULL) || (x < x_offset) ||
1031 ((x-x_offset) >= (ssize_t) source_image->columns))
1034 source[MaxPixelChannels];
1041 (void) GetOneVirtualPixel(source_image,
1042 CastDoubleToSsizeT((
double) x-x_offset),
1043 CastDoubleToSsizeT((
double) y-y_offset),source,exception);
1044 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
1049 PixelChannel channel = GetPixelChannelChannel(image,i);
1050 PixelTrait traits = GetPixelChannelTraits(image,channel);
1051 PixelTrait source_traits=GetPixelChannelTraits(source_image,
1053 if ((traits == UndefinedPixelTrait) ||
1054 (source_traits == UndefinedPixelTrait))
1056 if (channel == AlphaPixelChannel)
1057 pixel=(MagickRealType) TransparentAlpha;
1059 pixel=(MagickRealType) q[i];
1060 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
1061 ClampToQuantum(pixel);
1063 q+=(ptrdiff_t) GetPixelChannels(image);
1071 Sa=QuantumScale*(double) GetPixelAlpha(source_image,p);
1072 Da=QuantumScale*(double) GetPixelAlpha(image,q);
1074 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
1079 PixelChannel channel = GetPixelChannelChannel(image,i);
1080 PixelTrait traits = GetPixelChannelTraits(image,channel);
1081 PixelTrait source_traits=GetPixelChannelTraits(source_image,channel);
1082 if (traits == UndefinedPixelTrait)
1084 if ((source_traits == UndefinedPixelTrait) &&
1085 (channel != AlphaPixelChannel))
1087 if (channel == AlphaPixelChannel)
1092 pixel=(double) QuantumRange*alpha;
1093 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
1094 ClampToQuantum(pixel);
1101 Sc=(MagickRealType) GetPixelChannel(source_image,channel,p);
1102 Dc=(MagickRealType) q[i];
1103 if ((traits & CopyPixelTrait) != 0)
1108 q[i]=ClampToQuantum(Sc);
1116 Sca=QuantumScale*Sa*Sc;
1117 Dca=QuantumScale*Da*Dc;
1118 gamma=MagickSafeReciprocal(alpha);
1119 pixel=(double) QuantumRange*gamma*(Sca+Dca*(1.0-Sa));
1120 q[i]=clamp != MagickFalse ? ClampPixel(pixel) : ClampToQuantum(pixel);
1122 p+=(ptrdiff_t) GetPixelChannels(source_image);
1123 channels=GetPixelChannels(source_image);
1124 if (p >= (pixels+channels*source_image->columns))
1126 q+=(ptrdiff_t) GetPixelChannels(image);
1128 if (SyncCacheViewAuthenticPixels(image_view,exception) == MagickFalse)
1130 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1135#if defined(MAGICKCORE_OPENMP_SUPPORT)
1139 proceed=SetImageProgress(image,CompositeImageTag,progress,image->rows);
1140 if (proceed == MagickFalse)
1144 source_view=DestroyCacheView(source_view);
1145 image_view=DestroyCacheView(image_view);
1149static MagickBooleanType SaliencyBlendImage(Image *image,
1150 const Image *source_image,
const ssize_t x_offset,
const ssize_t y_offset,
1151 const double iterations,
const double residual_threshold,
const size_t tick,
1152 ExceptionInfo *exception)
1158 *residual_image = (Image *) NULL;
1164 status = MagickTrue,
1165 verbose = MagickFalse;
1169 source_image->columns,
1181 crop_image=CropImage(image,&crop_info,exception);
1182 if (crop_image == (Image *) NULL)
1183 return(MagickFalse);
1184 DisableCompositeClampUnlessSpecified(crop_image);
1185 divergent_image=BlendDivergentImage(crop_image,source_image,exception);
1186 if (divergent_image == (Image *) NULL)
1188 crop_image=DestroyImage(crop_image);
1189 return(MagickFalse);
1191 (void) ResetImagePage(crop_image,
"0x0+0+0");
1192 relax_image=BlendMeanImage(crop_image,source_image,exception);
1193 if (relax_image == (Image *) NULL)
1195 crop_image=DestroyImage(crop_image);
1196 divergent_image=DestroyImage(divergent_image);
1197 return(MagickFalse);
1199 status=BlendMaskAlphaChannel(crop_image,source_image,exception);
1200 if (status == MagickFalse)
1202 crop_image=DestroyImage(crop_image);
1203 divergent_image=DestroyImage(divergent_image);
1204 return(MagickFalse);
1206 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1207 if (residual_image == (Image *) NULL)
1209 crop_image=DestroyImage(crop_image);
1210 relax_image=DestroyImage(relax_image);
1211 return(MagickFalse);
1216 kernel_info=AcquireKernelInfo(
"3x3:0,0.25,0,0.25,0,0.25,0,0.25,0",exception);
1217 if (kernel_info == (KernelInfo *) NULL)
1219 crop_image=DestroyImage(crop_image);
1220 residual_image=DestroyImage(residual_image);
1221 relax_image=DestroyImage(relax_image);
1222 return(MagickFalse);
1224 verbose=IsStringTrue(GetImageArtifact(image,
"verbose"));
1225 if (verbose != MagickFalse)
1226 (void) FormatLocaleFile(stderr,
"saliency blending:\n");
1227 for (i=0; i < (ssize_t) iterations; i++)
1236 convolve_image=ConvolveImage(relax_image,kernel_info,exception);
1237 if (convolve_image == (Image *) NULL)
1239 relax_image=DestroyImage(relax_image);
1240 relax_image=convolve_image;
1241 sum_image=BlendSumImage(relax_image,divergent_image,1.0,-1.0,exception);
1242 if (sum_image == (Image *) NULL)
1244 relax_image=DestroyImage(relax_image);
1245 relax_image=sum_image;
1246 status=CompositeOverImage(relax_image,crop_image,MagickTrue,0,0,exception);
1247 if (status == MagickFalse)
1249 status=BlendRMSEResidual(relax_image,residual_image,&residual,exception);
1250 if (status == MagickFalse)
1252 if ((verbose != MagickFalse) && ((i % MagickMax(tick,1)) == 0))
1253 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(
double) residual);
1254 if (residual < residual_threshold)
1256 if (verbose != MagickFalse)
1257 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(
double)
1261 residual_image=DestroyImage(residual_image);
1262 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1263 if (residual_image == (Image *) NULL)
1266 kernel_info=DestroyKernelInfo(kernel_info);
1267 crop_image=DestroyImage(crop_image);
1268 divergent_image=DestroyImage(divergent_image);
1269 residual_image=DestroyImage(residual_image);
1273 status=CompositeOverImage(image,relax_image,MagickTrue,x_offset,y_offset,
1275 relax_image=DestroyImage(relax_image);
1279static MagickBooleanType SeamlessBlendImage(Image *image,
1280 const Image *source_image,
const ssize_t x_offset,
const ssize_t y_offset,
1281 const double iterations,
const double residual_threshold,
const size_t tick,
1282 ExceptionInfo *exception)
1296 status = MagickTrue,
1297 verbose = MagickFalse;
1301 source_image->columns,
1313 crop_image=CropImage(image,&crop_info,exception);
1314 if (crop_image == (Image *) NULL)
1315 return(MagickFalse);
1316 DisableCompositeClampUnlessSpecified(crop_image);
1317 (void) ResetImagePage(crop_image,
"0x0+0+0");
1318 sum_image=BlendSumImage(crop_image,source_image,1.0,-1.0,exception);
1319 crop_image=DestroyImage(crop_image);
1320 if (sum_image == (Image *) NULL)
1321 return(MagickFalse);
1322 mean_image=BlendMeanImage(sum_image,source_image,exception);
1323 sum_image=DestroyImage(sum_image);
1324 if (mean_image == (Image *) NULL)
1325 return(MagickFalse);
1326 relax_image=CloneImage(mean_image,0,0,MagickTrue,exception);
1327 if (relax_image == (Image *) NULL)
1329 mean_image=DestroyImage(mean_image);
1330 return(MagickFalse);
1332 status=BlendMaskAlphaChannel(mean_image,source_image,exception);
1333 if (status == MagickFalse)
1335 relax_image=DestroyImage(relax_image);
1336 mean_image=DestroyImage(mean_image);
1337 return(MagickFalse);
1339 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1340 if (residual_image == (Image *) NULL)
1342 relax_image=DestroyImage(relax_image);
1343 mean_image=DestroyImage(mean_image);
1344 return(MagickFalse);
1349 kernel_info=AcquireKernelInfo(
"3x3:0,0.25,0,0.25,0,0.25,0,0.25,0",exception);
1350 if (kernel_info == (KernelInfo *) NULL)
1352 residual_image=DestroyImage(residual_image);
1353 relax_image=DestroyImage(relax_image);
1354 mean_image=DestroyImage(mean_image);
1355 return(MagickFalse);
1357 verbose=IsStringTrue(GetImageArtifact(image,
"verbose"));
1358 if (verbose != MagickFalse)
1359 (void) FormatLocaleFile(stderr,
"seamless blending:\n");
1360 for (i=0; i < (ssize_t) iterations; i++)
1368 convolve_image=ConvolveImage(relax_image,kernel_info,exception);
1369 if (convolve_image == (Image *) NULL)
1371 relax_image=DestroyImage(relax_image);
1372 relax_image=convolve_image;
1373 status=CompositeOverImage(relax_image,mean_image,MagickTrue,0,0,exception);
1374 if (status == MagickFalse)
1376 status=BlendRMSEResidual(relax_image,residual_image,&residual,exception);
1377 if (status == MagickFalse)
1379 if ((verbose != MagickFalse) && ((i % MagickMax(tick,1)) == 0))
1380 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(
double) residual);
1381 if (residual < residual_threshold)
1383 if (verbose != MagickFalse)
1384 (void) FormatLocaleFile(stderr,
" %g: %g\n",(
double) i,(
double)
1388 if (residual_image != (Image *) NULL)
1389 residual_image=DestroyImage(residual_image);
1390 residual_image=CloneImage(relax_image,0,0,MagickTrue,exception);
1391 if (residual_image == (Image *) NULL)
1394 kernel_info=DestroyKernelInfo(kernel_info);
1395 mean_image=DestroyImage(mean_image);
1396 residual_image=DestroyImage(residual_image);
1400 foreground_image=BlendSumImage(source_image,relax_image,1.0,1.0,exception);
1401 relax_image=DestroyImage(relax_image);
1402 if (foreground_image == (Image *) NULL)
1403 return(MagickFalse);
1404 (void) SetImageMask(foreground_image,ReadPixelMask,(
const Image *) NULL,
1406 status=CompositeOverImage(image,foreground_image,MagickTrue,x_offset,y_offset,
1408 foreground_image=DestroyImage(foreground_image);
1412MagickExport MagickBooleanType CompositeImage(Image *image,
1413 const Image *composite,
const CompositeOperator compose,
1414 const MagickBooleanType clip_to_self,
const ssize_t x_offset,
1415 const ssize_t y_offset,ExceptionInfo *exception)
1417#define CompositeImageTag "Composite/Image"
1424 colorspace = HCLColorspace;
1430 white_luminance = 10000.0;
1436 illuminant = D65Illuminant;
1465 assert(image != (Image *) NULL);
1466 assert(image->signature == MagickCoreSignature);
1467 assert(composite != (Image *) NULL);
1468 assert(composite->signature == MagickCoreSignature);
1469 if (IsEventLogging() != MagickFalse)
1470 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1471 if (SetImageStorageClass(image,DirectClass,exception) == MagickFalse)
1472 return(MagickFalse);
1473 source_image=CloneImage(composite,0,0,MagickTrue,exception);
1474 if (source_image == (
const Image *) NULL)
1475 return(MagickFalse);
1476 (void) SetImageColorspace(source_image,image->colorspace,exception);
1477 if ((compose == OverCompositeOp) || (compose == SrcOverCompositeOp))
1479 status=CompositeOverImage(image,source_image,clip_to_self,x_offset,
1480 y_offset,exception);
1481 source_image=DestroyImage(source_image);
1485 canvas_image=(Image *) NULL;
1486 canvas_dissolve=1.0;
1487 white_luminance=10000.0;
1488 artifact=GetImageArtifact(image,
"compose:white-luminance");
1489 if (artifact != (
const char *) NULL)
1490 white_luminance=StringToDouble(artifact,(
char **) NULL);
1491 artifact=GetImageArtifact(image,
"compose:illuminant");
1492 if (artifact != (
const char *) NULL)
1497 illuminant_type=ParseCommandOption(MagickIlluminantOptions,MagickFalse,
1499 if (illuminant_type < 0)
1500 illuminant=UndefinedIlluminant;
1502 illuminant=(IlluminantType) illuminant_type;
1504 artifact=GetImageArtifact(image,
"compose:colorspace");
1505 if (artifact != (
const char *) NULL)
1510 colorspace_type=ParseCommandOption(MagickColorspaceOptions,MagickFalse,
1512 if (colorspace_type < 0)
1513 colorspace=UndefinedColorspace;
1515 colorspace=(ColorspaceType) colorspace_type;
1518 artifact=GetImageArtifact(image,
"compose:clamp");
1519 if (artifact != (
const char *) NULL)
1520 clamp=IsStringTrue(artifact);
1521 compose_sync=MagickTrue;
1522 artifact=GetImageArtifact(image,
"compose:sync");
1523 if (artifact != (
const char *) NULL)
1524 compose_sync=IsStringTrue(artifact);
1525 SetGeometryInfo(&geometry_info);
1527 percent_chroma=100.0;
1528 source_dissolve=1.0;
1532 case CopyCompositeOp:
1534 if ((x_offset < 0) || (y_offset < 0))
1536 if ((x_offset+(ssize_t) source_image->columns) > (ssize_t) image->columns)
1538 if ((y_offset+(ssize_t) source_image->rows) > (ssize_t) image->rows)
1540 if ((source_image->alpha_trait == UndefinedPixelTrait) &&
1541 (image->alpha_trait != UndefinedPixelTrait))
1542 (void) SetImageAlphaChannel(source_image,OpaqueAlphaChannel,exception);
1544 source_view=AcquireVirtualCacheView(source_image,exception);
1545 image_view=AcquireAuthenticCacheView(image,exception);
1546#if defined(MAGICKCORE_OPENMP_SUPPORT)
1547 #pragma omp parallel for schedule(static) shared(status) \
1548 magick_number_threads(source_image,image,source_image->rows,4)
1550 for (y=0; y < (ssize_t) source_image->rows; y++)
1564 if (status == MagickFalse)
1566 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
1568 q=GetCacheViewAuthenticPixels(image_view,x_offset,y+y_offset,
1569 source_image->columns,1,exception);
1570 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
1575 for (x=0; x < (ssize_t) source_image->columns; x++)
1580 if (GetPixelReadMask(source_image,p) <= (QuantumRange/2))
1582 p+=(ptrdiff_t) GetPixelChannels(source_image);
1583 q+=(ptrdiff_t) GetPixelChannels(image);
1586 for (i=0; i < (ssize_t) GetPixelChannels(source_image); i++)
1588 PixelChannel channel = GetPixelChannelChannel(source_image,i);
1589 PixelTrait source_traits = GetPixelChannelTraits(source_image,
1591 PixelTrait traits = GetPixelChannelTraits(image,channel);
1592 if ((source_traits == UndefinedPixelTrait) ||
1593 (traits == UndefinedPixelTrait))
1595 SetPixelChannel(image,channel,p[i],q);
1597 p+=(ptrdiff_t) GetPixelChannels(source_image);
1598 q+=(ptrdiff_t) GetPixelChannels(image);
1600 sync=SyncCacheViewAuthenticPixels(image_view,exception);
1601 if (sync == MagickFalse)
1603 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1608 proceed=SetImageProgress(image,CompositeImageTag,(MagickOffsetType)
1610 if (proceed == MagickFalse)
1614 source_view=DestroyCacheView(source_view);
1615 image_view=DestroyCacheView(image_view);
1616 source_image=DestroyImage(source_image);
1619 case IntensityCompositeOp:
1621 if ((x_offset < 0) || (y_offset < 0))
1623 if ((x_offset+(ssize_t) source_image->columns) > (ssize_t) image->columns)
1625 if ((y_offset+(ssize_t) source_image->rows) > (ssize_t) image->rows)
1628 source_view=AcquireVirtualCacheView(source_image,exception);
1629 image_view=AcquireAuthenticCacheView(image,exception);
1630#if defined(MAGICKCORE_OPENMP_SUPPORT)
1631 #pragma omp parallel for schedule(static) shared(status) \
1632 magick_number_threads(source_image,image,source_image->rows,4)
1634 for (y=0; y < (ssize_t) source_image->rows; y++)
1648 if (status == MagickFalse)
1650 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
1652 q=GetCacheViewAuthenticPixels(image_view,x_offset,y+y_offset,
1653 source_image->columns,1,exception);
1654 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
1659 for (x=0; x < (ssize_t) source_image->columns; x++)
1661 if (GetPixelReadMask(source_image,p) <= (QuantumRange/2))
1663 p+=(ptrdiff_t) GetPixelChannels(source_image);
1664 q+=(ptrdiff_t) GetPixelChannels(image);
1667 SetPixelAlpha(image,clamp != MagickFalse ?
1668 ClampPixel(GetPixelIntensity(source_image,p)) :
1669 ClampToQuantum(GetPixelIntensity(source_image,p)),q);
1670 p+=(ptrdiff_t) GetPixelChannels(source_image);
1671 q+=(ptrdiff_t) GetPixelChannels(image);
1673 sync=SyncCacheViewAuthenticPixels(image_view,exception);
1674 if (sync == MagickFalse)
1676 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1681 proceed=SetImageProgress(image,CompositeImageTag,(MagickOffsetType)
1683 if (proceed == MagickFalse)
1687 source_view=DestroyCacheView(source_view);
1688 image_view=DestroyCacheView(image_view);
1689 source_image=DestroyImage(source_image);
1692 case CopyAlphaCompositeOp:
1693 case ChangeMaskCompositeOp:
1699 if ((image->alpha_trait & BlendPixelTrait) == 0)
1700 (void) SetImageAlphaChannel(image,OpaqueAlphaChannel,exception);
1703 case BlurCompositeOp:
1728 canvas_image=CloneImage(image,0,0,MagickTrue,exception);
1729 if (canvas_image == (Image *) NULL)
1731 source_image=DestroyImage(source_image);
1732 return(MagickFalse);
1738 artifact=GetImageArtifact(image,
"compose:args");
1739 if (artifact != (
const char *) NULL)
1740 flags=ParseGeometry(artifact,&geometry_info);
1741 if ((flags & WidthValue) == 0)
1743 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
1744 "InvalidSetting",
"'%s' '%s'",
"compose:args",artifact);
1745 source_image=DestroyImage(source_image);
1746 canvas_image=DestroyImage(canvas_image);
1747 return(MagickFalse);
1754 width=2.0*geometry_info.rho;
1756 if ((flags & HeightValue) != 0)
1757 height=2.0*geometry_info.sigma;
1765 if ((flags & XValue) != 0 )
1773 angle=DegreesToRadians(geometry_info.xi);
1774 blur.x1=width*cos(angle);
1775 blur.x2=width*sin(angle);
1776 blur.y1=(-height*sin(angle));
1777 blur.y2=height*cos(angle);
1781 if ((flags & YValue) != 0 )
1786 angle_start=DegreesToRadians(geometry_info.xi);
1787 angle_range=DegreesToRadians(geometry_info.psi)-angle_start;
1798 resample_filter=AcquireResampleFilter(image,exception);
1799 SetResampleFilter(resample_filter,GaussianFilter);
1803 GetPixelInfo(image,&pixel);
1804 source_view=AcquireVirtualCacheView(source_image,exception);
1805 canvas_view=AcquireAuthenticCacheView(canvas_image,exception);
1806 for (y=0; y < (ssize_t) source_image->rows; y++)
1820 if (((y+y_offset) < 0) || ((y+y_offset) >= (ssize_t) image->rows))
1822 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
1824 q=QueueCacheViewAuthenticPixels(canvas_view,0,y,canvas_image->columns,1,
1826 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
1828 for (x=0; x < (ssize_t) source_image->columns; x++)
1830 if (((x_offset+x) < 0) || ((x_offset+x) >= (ssize_t) image->columns))
1832 p+=(ptrdiff_t) GetPixelChannels(source_image);
1835 if (fabs(angle_range) > MagickEpsilon)
1840 angle=angle_start+angle_range*QuantumScale*(double)
1841 GetPixelBlue(source_image,p);
1842 blur.x1=width*cos(angle);
1843 blur.x2=width*sin(angle);
1844 blur.y1=(-height*sin(angle));
1845 blur.y2=height*cos(angle);
1847 ScaleResampleFilter(resample_filter,
1848 blur.x1*QuantumScale*(
double) GetPixelRed(source_image,p),
1849 blur.y1*QuantumScale*(
double) GetPixelGreen(source_image,p),
1850 blur.x2*QuantumScale*(
double) GetPixelRed(source_image,p),
1851 blur.y2*QuantumScale*(
double) GetPixelGreen(source_image,p) );
1852 (void) ResamplePixelColor(resample_filter,(
double) x_offset+x,
1853 (
double) y_offset+y,&pixel,exception);
1854 SetPixelViaPixelInfo(canvas_image,&pixel,q);
1855 p+=(ptrdiff_t) GetPixelChannels(source_image);
1856 q+=(ptrdiff_t) GetPixelChannels(canvas_image);
1858 sync=SyncCacheViewAuthenticPixels(canvas_view,exception);
1859 if (sync == MagickFalse)
1862 resample_filter=DestroyResampleFilter(resample_filter);
1863 source_view=DestroyCacheView(source_view);
1864 canvas_view=DestroyCacheView(canvas_view);
1865 source_image=DestroyImage(source_image);
1866 source_image=canvas_image;
1869 case DisplaceCompositeOp:
1870 case DistortCompositeOp:
1891 canvas_image=CloneImage(image,0,0,MagickTrue,exception);
1892 if (canvas_image == (Image *) NULL)
1894 source_image=DestroyImage(source_image);
1895 return(MagickFalse);
1897 SetGeometryInfo(&geometry_info);
1899 artifact=GetImageArtifact(image,
"compose:args");
1900 if (artifact != (
char *) NULL)
1901 flags=ParseGeometry(artifact,&geometry_info);
1902 if ((flags & (WidthValue | HeightValue)) == 0 )
1904 if ((flags & AspectValue) == 0)
1906 horizontal_scale=(MagickRealType) (source_image->columns-1)/2.0;
1907 vertical_scale=(MagickRealType) (source_image->rows-1)/2.0;
1911 horizontal_scale=(MagickRealType) (image->columns-1)/2.0;
1912 vertical_scale=(MagickRealType) (image->rows-1)/2.0;
1917 horizontal_scale=geometry_info.rho;
1918 vertical_scale=geometry_info.sigma;
1919 if ((flags & PercentValue) != 0)
1921 if ((flags & AspectValue) == 0)
1923 horizontal_scale*=(source_image->columns-1)/200.0;
1924 vertical_scale*=(source_image->rows-1)/200.0;
1928 horizontal_scale*=(image->columns-1)/200.0;
1929 vertical_scale*=(image->rows-1)/200.0;
1932 if ((flags & HeightValue) == 0)
1933 vertical_scale=horizontal_scale;
1943 center.x=(MagickRealType) x_offset;
1944 center.y=(MagickRealType) y_offset;
1945 if (compose == DistortCompositeOp)
1947 if ((flags & XValue) == 0)
1948 if ((flags & AspectValue) != 0)
1949 center.x=(MagickRealType) ((image->columns-1)/2.0);
1951 center.x=(MagickRealType) (x_offset+(source_image->columns-1)/
1954 if ((flags & AspectValue) != 0)
1955 center.x=geometry_info.xi;
1957 center.x=(MagickRealType) (x_offset+geometry_info.xi);
1958 if ((flags & YValue) == 0)
1959 if ((flags & AspectValue) != 0)
1960 center.y=(MagickRealType) ((image->rows-1)/2.0);
1962 center.y=(MagickRealType) (y_offset+(source_image->rows-1)/2.0);
1964 if ((flags & AspectValue) != 0)
1965 center.y=geometry_info.psi;
1967 center.y=(MagickRealType) (y_offset+geometry_info.psi);
1973 GetPixelInfo(image,&pixel);
1974 image_view=AcquireVirtualCacheView(image,exception);
1975 source_view=AcquireVirtualCacheView(source_image,exception);
1976 canvas_view=AcquireAuthenticCacheView(canvas_image,exception);
1977 for (y=0; y < (ssize_t) source_image->rows; y++)
1991 if (((y+y_offset) < 0) || ((y+y_offset) >= (ssize_t) image->rows))
1993 p=GetCacheViewVirtualPixels(source_view,0,y,source_image->columns,1,
1995 q=QueueCacheViewAuthenticPixels(canvas_view,0,y,canvas_image->columns,1,
1997 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
1999 for (x=0; x < (ssize_t) source_image->columns; x++)
2001 if (((x_offset+x) < 0) || ((x_offset+x) >= (ssize_t) image->columns))
2003 p+=(ptrdiff_t) GetPixelChannels(source_image);
2009 offset.x=(double) (horizontal_scale*((
double) GetPixelRed(
2010 source_image,p)-(((MagickRealType) QuantumRange+1.0)/2.0)))/
2011 (((MagickRealType) QuantumRange+1.0)/2.0)+center.x+
2012 ((compose == DisplaceCompositeOp) ? x : 0);
2013 offset.y=(double) (vertical_scale*((
double) GetPixelGreen(
2014 source_image,p)-(((MagickRealType) QuantumRange+1.0)/2.0)))/
2015 (((MagickRealType) QuantumRange+1.0)/2.0)+center.y+
2016 ((compose == DisplaceCompositeOp) ? y : 0);
2017 status=InterpolatePixelInfo(image,image_view,
2018 UndefinedInterpolatePixel,(
double) offset.x,(
double) offset.y,
2020 if (status == MagickFalse)
2025 pixel.alpha=(MagickRealType) QuantumRange*(QuantumScale*pixel.alpha)*
2026 (QuantumScale*(double) GetPixelAlpha(source_image,p));
2027 SetPixelViaPixelInfo(canvas_image,&pixel,q);
2028 p+=(ptrdiff_t) GetPixelChannels(source_image);
2029 q+=(ptrdiff_t) GetPixelChannels(canvas_image);
2031 if (x < (ssize_t) source_image->columns)
2033 sync=SyncCacheViewAuthenticPixels(canvas_view,exception);
2034 if (sync == MagickFalse)
2037 canvas_view=DestroyCacheView(canvas_view);
2038 source_view=DestroyCacheView(source_view);
2039 image_view=DestroyCacheView(image_view);
2040 source_image=DestroyImage(source_image);
2041 source_image=canvas_image;
2044 case DissolveCompositeOp:
2049 artifact=GetImageArtifact(image,
"compose:args");
2050 if (artifact != (
char *) NULL)
2052 flags=ParseGeometry(artifact,&geometry_info);
2053 source_dissolve=geometry_info.rho/100.0;
2054 canvas_dissolve=1.0;
2055 if ((source_dissolve-MagickEpsilon) < 0.0)
2056 source_dissolve=0.0;
2057 if ((source_dissolve+MagickEpsilon) > 1.0)
2059 canvas_dissolve=2.0-source_dissolve;
2060 source_dissolve=1.0;
2062 if ((flags & SigmaValue) != 0)
2063 canvas_dissolve=geometry_info.sigma/100.0;
2064 if ((canvas_dissolve-MagickEpsilon) < 0.0)
2065 canvas_dissolve=0.0;
2066 if ((canvas_dissolve+MagickEpsilon) > 1.0)
2067 canvas_dissolve=1.0;
2071 case BlendCompositeOp:
2073 artifact=GetImageArtifact(image,
"compose:args");
2074 if (artifact != (
char *) NULL)
2076 flags=ParseGeometry(artifact,&geometry_info);
2077 source_dissolve=geometry_info.rho/100.0;
2078 canvas_dissolve=1.0-source_dissolve;
2079 if ((flags & SigmaValue) != 0)
2080 canvas_dissolve=geometry_info.sigma/100.0;
2084 case SaliencyBlendCompositeOp:
2087 residual_threshold = 0.0002,
2093 artifact=GetImageArtifact(image,
"compose:args");
2094 if (artifact != (
char *) NULL)
2096 flags=ParseGeometry(artifact,&geometry_info);
2097 iterations=geometry_info.rho;
2098 if ((flags & SigmaValue) != 0)
2099 residual_threshold=geometry_info.sigma;
2100 if ((flags & XiValue) != 0)
2101 tick=(size_t) geometry_info.xi;
2103 status=SaliencyBlendImage(image,composite,x_offset,y_offset,iterations,
2104 residual_threshold,tick,exception);
2105 source_image=DestroyImage(source_image);
2108 case SeamlessBlendCompositeOp:
2111 residual_threshold = 0.0002,
2117 artifact=GetImageArtifact(image,
"compose:args");
2118 if (artifact != (
char *) NULL)
2120 flags=ParseGeometry(artifact,&geometry_info);
2121 iterations=geometry_info.rho;
2122 if ((flags & SigmaValue) != 0)
2123 residual_threshold=geometry_info.sigma;
2124 if ((flags & XiValue) != 0)
2125 tick=(size_t) geometry_info.xi;
2127 status=SeamlessBlendImage(image,composite,x_offset,y_offset,iterations,
2128 residual_threshold,tick,exception);
2129 source_image=DestroyImage(source_image);
2132 case MathematicsCompositeOp:
2142 SetGeometryInfo(&geometry_info);
2143 artifact=GetImageArtifact(image,
"compose:args");
2144 if (artifact != (
char *) NULL)
2146 flags=ParseGeometry(artifact,&geometry_info);
2147 if (flags == NoValue)
2148 (void) ThrowMagickException(exception,GetMagickModule(),OptionError,
2149 "InvalidGeometry",
"`%s'",artifact);
2153 case ModulateCompositeOp:
2158 artifact=GetImageArtifact(image,
"compose:args");
2159 if (artifact != (
char *) NULL)
2161 flags=ParseGeometry(artifact,&geometry_info);
2162 percent_luma=geometry_info.rho;
2163 if ((flags & SigmaValue) != 0)
2164 percent_chroma=geometry_info.sigma;
2168 case ThresholdCompositeOp:
2173 artifact=GetImageArtifact(image,
"compose:args");
2174 if (artifact != (
char *) NULL)
2176 flags=ParseGeometry(artifact,&geometry_info);
2177 amount=geometry_info.rho;
2178 threshold=geometry_info.sigma;
2179 if ((flags & SigmaValue) == 0)
2182 threshold*=(double) QuantumRange;
2193 midpoint=((MagickRealType) QuantumRange+1.0)/2;
2194 source_view=AcquireVirtualCacheView(source_image,exception);
2195 image_view=AcquireAuthenticCacheView(image,exception);
2196#if defined(MAGICKCORE_OPENMP_SUPPORT)
2197 #pragma omp parallel for schedule(static) shared(progress,status) \
2198 magick_number_threads(source_image,image,image->rows,1)
2200 for (y=0; y < (ssize_t) image->rows; y++)
2224 if (status == MagickFalse)
2226 if (clip_to_self != MagickFalse)
2230 if ((y-y_offset) >= (ssize_t) source_image->rows)
2236 pixels=(Quantum *) NULL;
2238 if ((y >= y_offset) &&
2239 ((y-y_offset) < (ssize_t) source_image->rows))
2241 p=GetCacheViewVirtualPixels(source_view,0,
2242 CastDoubleToSsizeT((
double) y-y_offset),source_image->columns,1,
2244 if (p == (
const Quantum *) NULL)
2251 p-=(ptrdiff_t) CastDoubleToSsizeT((
double) x_offset*
2252 GetPixelChannels(source_image));
2254 q=GetCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
2255 if (q == (Quantum *) NULL)
2260 GetPixelInfo(image,&canvas_pixel);
2261 GetPixelInfo(source_image,&source_pixel);
2262 for (x=0; x < (ssize_t) image->columns; x++)
2287 if (clip_to_self != MagickFalse)
2291 q+=(ptrdiff_t) GetPixelChannels(image);
2294 if ((x-x_offset) >= (ssize_t) source_image->columns)
2297 if ((pixels == (Quantum *) NULL) || (x < x_offset) ||
2298 ((x-x_offset) >= (ssize_t) source_image->columns))
2301 source[MaxPixelChannels];
2308 (void) GetOneVirtualPixel(source_image,
2309 CastDoubleToSsizeT((
double) x-x_offset),
2310 CastDoubleToSsizeT((
double) y-y_offset),source,exception);
2311 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
2316 PixelChannel channel = GetPixelChannelChannel(image,i);
2317 PixelTrait traits = GetPixelChannelTraits(image,channel);
2318 PixelTrait source_traits = GetPixelChannelTraits(source_image,
2320 if ((traits == UndefinedPixelTrait) ||
2321 (source_traits == UndefinedPixelTrait))
2325 case AlphaCompositeOp:
2326 case ChangeMaskCompositeOp:
2327 case CopyAlphaCompositeOp:
2328 case DstAtopCompositeOp:
2329 case DstInCompositeOp:
2331 case OutCompositeOp:
2332 case SrcInCompositeOp:
2333 case SrcOutCompositeOp:
2335 if (channel == AlphaPixelChannel)
2336 pixel=(MagickRealType) TransparentAlpha;
2338 pixel=(MagickRealType) q[i];
2341 case ClearCompositeOp:
2342 case CopyCompositeOp:
2343 case ReplaceCompositeOp:
2345 if (channel == AlphaPixelChannel)
2346 pixel=(MagickRealType) TransparentAlpha;
2351 case BlendCompositeOp:
2352 case DissolveCompositeOp:
2354 if (channel == AlphaPixelChannel)
2355 pixel=canvas_dissolve*(double) GetPixelAlpha(source_image,
2358 pixel=(MagickRealType) source[channel];
2363 pixel=(MagickRealType) source[channel];
2367 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
2368 ClampToQuantum(pixel);
2370 q+=(ptrdiff_t) GetPixelChannels(image);
2378 Sa=QuantumScale*(double) GetPixelAlpha(source_image,p);
2379 Da=QuantumScale*(double) GetPixelAlpha(image,q);
2382 case BumpmapCompositeOp:
2383 case ColorBurnCompositeOp:
2384 case ColorDodgeCompositeOp:
2385 case DarkenCompositeOp:
2386 case DifferenceCompositeOp:
2387 case DivideDstCompositeOp:
2388 case DivideSrcCompositeOp:
2389 case ExclusionCompositeOp:
2390 case FreezeCompositeOp:
2391 case HardLightCompositeOp:
2392 case HardMixCompositeOp:
2393 case InterpolateCompositeOp:
2394 case LightenCompositeOp:
2395 case LinearBurnCompositeOp:
2396 case LinearDodgeCompositeOp:
2397 case LinearLightCompositeOp:
2398 case MathematicsCompositeOp:
2399 case MinusDstCompositeOp:
2400 case MinusSrcCompositeOp:
2401 case MultiplyCompositeOp:
2402 case NegateCompositeOp:
2403 case OverlayCompositeOp:
2404 case PegtopLightCompositeOp:
2405 case PinLightCompositeOp:
2406 case ReflectCompositeOp:
2407 case ScreenCompositeOp:
2408 case SoftBurnCompositeOp:
2409 case SoftDodgeCompositeOp:
2410 case SoftLightCompositeOp:
2411 case StampCompositeOp:
2412 case VividLightCompositeOp:
2414 alpha=RoundToUnity(Sa+Da-Sa*Da);
2417 case DstAtopCompositeOp:
2418 case DstInCompositeOp:
2420 case SrcInCompositeOp:
2425 case DissolveCompositeOp:
2427 alpha=source_dissolve*Sa*(-canvas_dissolve*Da)+source_dissolve*Sa+
2431 case DstOverCompositeOp:
2432 case OverCompositeOp:
2433 case SrcOverCompositeOp:
2438 case DstOutCompositeOp:
2443 case OutCompositeOp:
2444 case SrcOutCompositeOp:
2449 case BlendCompositeOp:
2450 case PlusCompositeOp:
2452 alpha=RoundToUnity(source_dissolve*Sa+canvas_dissolve*Da);
2455 case XorCompositeOp:
2457 alpha=Sa+Da-2.0*Sa*Da;
2460 case ModulusAddCompositeOp:
2467 alpha=((Sa+Da)-1.0);
2470 case ModulusSubtractCompositeOp:
2477 alpha=((Sa-Da)+1.0);
2488 case ColorizeCompositeOp:
2489 case HueCompositeOp:
2490 case LuminizeCompositeOp:
2491 case ModulateCompositeOp:
2492 case RMSECompositeOp:
2493 case SaturateCompositeOp:
2495 Si=GetPixelIntensity(source_image,p);
2496 GetPixelInfoPixel(source_image,p,&source_pixel);
2497 GetPixelInfoPixel(image,q,&canvas_pixel);
2500 case BumpmapCompositeOp:
2501 case CopyAlphaCompositeOp:
2502 case DarkenIntensityCompositeOp:
2503 case LightenIntensityCompositeOp:
2505 Si=GetPixelIntensity(source_image,p);
2506 Di=GetPixelIntensity(image,q);
2512 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
2518 PixelChannel channel = GetPixelChannelChannel(image,i);
2519 PixelTrait traits = GetPixelChannelTraits(image,channel);
2520 PixelTrait source_traits = GetPixelChannelTraits(source_image,channel);
2521 if (traits == UndefinedPixelTrait)
2523 if ((channel == AlphaPixelChannel) &&
2524 ((traits & UpdatePixelTrait) != 0))
2531 case AlphaCompositeOp:
2533 pixel=(double) QuantumRange*Sa;
2536 case AtopCompositeOp:
2537 case CopyBlackCompositeOp:
2538 case CopyBlueCompositeOp:
2539 case CopyCyanCompositeOp:
2540 case CopyGreenCompositeOp:
2541 case CopyMagentaCompositeOp:
2542 case CopyRedCompositeOp:
2543 case CopyYellowCompositeOp:
2544 case SrcAtopCompositeOp:
2545 case DstCompositeOp:
2548 pixel=(double) QuantumRange*Da;
2551 case BumpmapCompositeOp:
2556 case ChangeMaskCompositeOp:
2558 if (IsFuzzyEquivalencePixel(source_image,p,image,q) != MagickFalse)
2559 pixel=(MagickRealType) TransparentAlpha;
2561 pixel=(double) QuantumRange*Da;
2564 case ClearCompositeOp:
2566 pixel=(MagickRealType) TransparentAlpha;
2569 case ColorizeCompositeOp:
2570 case HueCompositeOp:
2571 case LuminizeCompositeOp:
2572 case RMSECompositeOp:
2573 case SaturateCompositeOp:
2575 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
2577 pixel=(double) QuantumRange*Da;
2580 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
2582 pixel=(double) QuantumRange*Sa;
2587 pixel=(double) QuantumRange*Da;
2590 pixel=(double) QuantumRange*Sa;
2593 case CopyAlphaCompositeOp:
2598 case BlurCompositeOp:
2599 case CopyCompositeOp:
2600 case DisplaceCompositeOp:
2601 case DistortCompositeOp:
2602 case DstAtopCompositeOp:
2603 case ReplaceCompositeOp:
2604 case SrcCompositeOp:
2606 pixel=(double) QuantumRange*Sa;
2609 case DarkenIntensityCompositeOp:
2611 if (compose_sync == MagickFalse)
2613 pixel=Si < Di ? Sa : Da;
2616 pixel=Sa*Si < Da*Di ? Sa : Da;
2619 case DifferenceCompositeOp:
2621 pixel=(double) QuantumRange*fabs((
double) (Sa-Da));
2624 case FreezeCompositeOp:
2626 pixel=(double) QuantumRange*(1.0-(1.0-Sa)*(1.0-Sa)*
2627 MagickSafeReciprocal(Da));
2632 case InterpolateCompositeOp:
2634 pixel=(double) QuantumRange*(0.5-0.25*cos(MagickPI*Sa)-0.25*
2638 case LightenIntensityCompositeOp:
2640 if (compose_sync == MagickFalse)
2642 pixel=Si > Di ? Sa : Da;
2645 pixel=Sa*Si > Da*Di ? Sa : Da;
2648 case ModulateCompositeOp:
2650 pixel=(double) QuantumRange*Da;
2653 case MultiplyCompositeOp:
2655 if (compose_sync == MagickFalse)
2657 pixel=(double) QuantumRange*Sa*Da;
2660 pixel=(double) QuantumRange*alpha;
2663 case NegateCompositeOp:
2665 pixel=(double) QuantumRange*((1.0-Sa-Da));
2668 case ReflectCompositeOp:
2670 pixel=(double) QuantumRange*(Sa*Sa*
2671 MagickSafeReciprocal(1.0-Da));
2672 if (pixel > (
double) QuantumRange)
2673 pixel=(double) QuantumRange;
2676 case StampCompositeOp:
2678 pixel=(double) QuantumRange*(Sa+Da*Da-1.0);
2681 case StereoCompositeOp:
2683 pixel=(double) QuantumRange*(Sa+Da)/2;
2688 pixel=(double) QuantumRange*alpha;
2692 q[i]=clamp != MagickFalse ? ClampPixel(pixel) :
2693 ClampToQuantum(pixel);
2696 if (source_traits == UndefinedPixelTrait)
2702 Sc=(MagickRealType) GetPixelChannel(source_image,channel,p);
2703 Dc=(MagickRealType) q[i];
2704 if ((traits & CopyPixelTrait) != 0)
2709 q[i]=ClampToQuantum(Dc);
2717 Sca=QuantumScale*Sa*Sc;
2718 Dca=QuantumScale*Da*Dc;
2721 case DarkenCompositeOp:
2722 case LightenCompositeOp:
2723 case ModulusSubtractCompositeOp:
2725 gamma=MagickSafeReciprocal(1.0-alpha);
2730 gamma=MagickSafeReciprocal(alpha);
2737 case AlphaCompositeOp:
2739 pixel=(double) QuantumRange*Sa;
2742 case AtopCompositeOp:
2743 case SrcAtopCompositeOp:
2745 pixel=(double) QuantumRange*(Sca*Da+Dca*(1.0-Sa));
2748 case BlendCompositeOp:
2750 pixel=gamma*(source_dissolve*Sa*Sc+canvas_dissolve*Da*Dc);
2753 case CopyCompositeOp:
2754 case ReplaceCompositeOp:
2756 pixel=(double) QuantumRange*Sca;
2759 case BlurCompositeOp:
2760 case DisplaceCompositeOp:
2761 case DistortCompositeOp:
2762 case SrcCompositeOp:
2767 case BumpmapCompositeOp:
2769 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
2774 pixel=(QuantumScale*Si)*Dc;
2777 case ChangeMaskCompositeOp:
2782 case ClearCompositeOp:
2787 case ColorBurnCompositeOp:
2789 D=(Da > 0.0) ? RoundToUnity(Dca/Da) : 0.0;
2790 S=(Sa > 0.0) ? RoundToUnity(Sca/Sa) : 0.0;
2794 blend=1.0-(1.0-D)/S;
2795 pixel=(double) QuantumRange*gamma*RoundToUnity(Sa*Da*
2796 RoundToUnity(blend)+Sa*(1.0-Da)*S+Da*(1.0-Sa)*D);
2799 case ColorDodgeCompositeOp:
2802 S=RoundToUnity(Sca/Sa);
2806 D=RoundToUnity(Dca/Da);
2819 blend=MagickMin(1.0,D/(1.0-S));
2821 pixel=(double) QuantumRange*gamma*(Sa*Da*blend+Sca*(1.0-Da)+Dca*
2825 case ColorizeCompositeOp:
2827 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
2832 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
2837 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
2838 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
2839 white_luminance,illuminant,&sans,&sans,&luma);
2840 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
2841 (
double) source_pixel.green,(
double) source_pixel.blue,
2842 white_luminance,illuminant,&hue,&chroma,&sans);
2843 ConvertGenericToRGB(colorspace,hue,chroma,luma,
2844 white_luminance,illuminant,&red,&green,&blue);
2847 case RedPixelChannel: pixel=red;
break;
2848 case GreenPixelChannel: pixel=green;
break;
2849 case BluePixelChannel: pixel=blue;
break;
2850 default: pixel=Dc;
break;
2854 case CopyAlphaCompositeOp:
2855 case DstCompositeOp:
2860 case CopyBlackCompositeOp:
2862 if (channel == BlackPixelChannel)
2863 pixel=(MagickRealType) GetPixelBlack(source_image,p);
2866 case CopyBlueCompositeOp:
2867 case CopyYellowCompositeOp:
2869 if (channel == BluePixelChannel)
2870 pixel=(MagickRealType) GetPixelBlue(source_image,p);
2873 case CopyGreenCompositeOp:
2874 case CopyMagentaCompositeOp:
2876 if (channel == GreenPixelChannel)
2877 pixel=(MagickRealType) GetPixelGreen(source_image,p);
2880 case CopyRedCompositeOp:
2881 case CopyCyanCompositeOp:
2883 if (channel == RedPixelChannel)
2884 pixel=(MagickRealType) GetPixelRed(source_image,p);
2887 case DarkenCompositeOp:
2889 if (compose_sync == MagickFalse)
2891 pixel=RoundToUnity(MagickMin(Sca,Dca)+Sca*(1.0-Da)+Dca*
2895 pixel=(double) QuantumRange*RoundToUnity(MagickMin(Sca,Dca)+Sca*
2896 (1.0-Da)+Dca*(1.0-Sa));
2899 case DarkenIntensityCompositeOp:
2901 if (compose_sync == MagickFalse)
2903 pixel=RoundToUnity(MagickMin(Sca,Dca)+Sca*(1.0-Di)+Dca*
2907 pixel=(double) QuantumRange*RoundToUnity(MagickMin(Sca,Dca)+Sca*
2908 (1.0-Di)+Dca*(1.0-Si));
2911 case DifferenceCompositeOp:
2913 if (compose_sync == MagickFalse)
2915 pixel=(double) QuantumRange*RoundToUnity(fabs((
double) Sc-
2919 S=(Sa > 0.0) ? (Sca/Sa) : 0.0;
2920 D=(Da > 0.0) ? (Dca/Da) : 0.0;
2921 pixel=(double) QuantumRange*RoundToUnity(fabs(S-D));
2924 case DissolveCompositeOp:
2926 pixel=gamma*(source_dissolve*Sa*Sc-source_dissolve*Sa*
2927 canvas_dissolve*Da*Dc+canvas_dissolve*Da*Dc);
2930 case DivideDstCompositeOp:
2932 S=(Sa > 0.0) ? (Sca/Sa) : 0.0;
2933 D=(Da > 0.0) ? (Dca/Da) : 0.0;
2937 blend=MagickMin(1.0,D/S);
2938 pixel=(double) QuantumRange*RoundToUnity(Sca*(1.0-Da)+Dca*(1.0-Sa)+
2942 case DivideSrcCompositeOp:
2944 S=(Sa > 0.0) ? (Sca/Sa) : 0.0;
2945 D=(Da > 0.0) ? (Dca/Da) : 0.0;
2949 blend=MagickMin(1.0,S/D);
2950 pixel=(double) QuantumRange*RoundToUnity(Sca*(1.0-Da)+Dca*(1.0-Sa)+
2954 case DstAtopCompositeOp:
2956 pixel=(double) QuantumRange*(Dca*Sa+Sca*(1.0-Da));
2959 case DstInCompositeOp:
2961 pixel=(double) QuantumRange*gamma*(Dca*Sa);
2964 case DstOutCompositeOp:
2966 pixel=(double) QuantumRange*gamma*(Dca*(1.0-Sa));
2969 case DstOverCompositeOp:
2971 pixel=(double) QuantumRange*gamma*(Dca+Sca*(1.0-Da));
2974 case ExclusionCompositeOp:
2976 S=(Sa > 0.0) ? Sca/Sa : 0.0;
2977 D=(Da > 0.0) ? Dca/Da : 0.0;
2978 blend=RoundToUnity(S+D-2.0*S*D);
2979 pixel=(double) QuantumRange*RoundToUnity((blend*Sa+D*(1.0-Sa))*
2983 case FreezeCompositeOp:
2986 blend=1.0-(1.0-Sca)*(1.0-Sca)/Dca;
2989 pixel=(double) QuantumRange*gamma*RoundToUnity(blend);
2992 case HardLightCompositeOp:
2994 D=(Da > 0.0) ? (Dca/Da) : 0.0;
2995 S=(Sa > 0.0) ? (Sca/Sa) : 0.0;
2999 blend=1.0-2.0*(1.0-S)*(1.0-D);
3000 pixel=(double) QuantumRange*gamma*RoundToUnity((1.0-Da)*Sca+
3001 (1.0-Sa)*Dca+blend*Sa*Da);
3004 case HardMixCompositeOp:
3008 if ((2.0*Sca) == 0.0)
3011 blend=1.0-(1.0-Dca)/(2.0*Sca);
3015 if ((1.0-((2.0*Sca)-1.0)) == 0.0)
3018 blend=Dca/(1.0-(2.0*Sca-1.0));
3020 pixel=gamma*((blend < 0.5) ? 0.0 : (double) QuantumRange);
3023 case HueCompositeOp:
3025 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
3030 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
3035 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3036 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3037 white_luminance,illuminant,&hue,&chroma,&luma);
3038 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
3039 (
double) source_pixel.green,(
double) source_pixel.blue,
3040 white_luminance,illuminant,&hue,&sans,&sans);
3041 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3042 white_luminance,illuminant,&red,&green,&blue);
3045 case RedPixelChannel: pixel=red;
break;
3046 case GreenPixelChannel: pixel=green;
break;
3047 case BluePixelChannel: pixel=blue;
break;
3048 default: pixel=Dc;
break;
3053 case SrcInCompositeOp:
3055 pixel=(double) QuantumRange*(Sca*Da);
3058 case InterpolateCompositeOp:
3060 pixel=(double) QuantumRange*(0.5-0.25*cos(MagickPI*Sca)-0.25*
3064 case LinearBurnCompositeOp:
3072 pixel=(double) QuantumRange*(Sca+Dca-Sa*Da);
3075 case LinearDodgeCompositeOp:
3077 pixel=gamma*(Sa*Sc+Da*Dc);
3080 case LinearLightCompositeOp:
3087 D=(Da > 0.0) ? Dca/Da : 0.0,
3088 S=(Sa > 0.0) ? Sca/Sa : 0.0;
3089 pixel=(double) QuantumRange*gamma*(RoundToUnity(D+2.0*S-1.0)*Da);
3092 case LightenCompositeOp:
3094 if (compose_sync == MagickFalse)
3096 pixel=MagickMax(Sc,Dc);
3099 S=(Sa > 0.0) ? (Sca/Sa) : 0.0;
3100 D=(Da > 0.0) ? (Dca/Da) : 0.0;
3101 pixel=(double) QuantumRange*RoundToUnity(Sca*(1.0-Da)+Dca*(1.0-Sa)+
3102 MagickMax(S,D)*Sa*Da);
3105 case LightenIntensityCompositeOp:
3107 pixel=Si > Di ? Sc : Dc;
3110 case LuminizeCompositeOp:
3112 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
3117 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
3122 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3123 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3124 white_luminance,illuminant,&hue,&chroma,&luma);
3125 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
3126 (
double) source_pixel.green,(
double) source_pixel.blue,
3127 white_luminance,illuminant,&sans,&sans,&luma);
3128 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3129 white_luminance,illuminant,&red,&green,&blue);
3132 case RedPixelChannel: pixel=red;
break;
3133 case GreenPixelChannel: pixel=green;
break;
3134 case BluePixelChannel: pixel=blue;
break;
3135 default: pixel=Dc;
break;
3139 case MathematicsCompositeOp:
3160 if (compose_sync == MagickFalse)
3162 pixel=geometry_info.rho*Sc*Dc+geometry_info.sigma*Sc+
3163 geometry_info.xi*Dc+geometry_info.psi;
3166 pixel=(double) QuantumRange*gamma*(geometry_info.rho*Sca*Dca+
3167 geometry_info.sigma*Sca*Da+geometry_info.xi*Dca*Sa+
3168 geometry_info.psi*Sa*Da+Sca*(1.0-Da)+Dca*(1.0-Sa));
3171 case MinusDstCompositeOp:
3173 if (compose_sync == MagickFalse)
3178 pixel=gamma*(Sa*Sc+Da*Dc-2.0*Da*Dc*Sa);
3181 case MinusSrcCompositeOp:
3183 if (compose_sync == MagickFalse)
3188 pixel=gamma*(Da*Dc+Sa*Sc-2.0*Sa*Sc*Da);
3191 case ModulateCompositeOp:
3196 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
3201 offset=(ssize_t) (Si-midpoint);
3207 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3208 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3209 white_luminance,illuminant,&hue,&chroma,&luma);
3210 luma+=(0.01*percent_luma*offset)/midpoint;
3211 chroma*=0.01*percent_chroma;
3212 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3213 white_luminance,illuminant,&red,&green,&blue);
3216 case RedPixelChannel: pixel=red;
break;
3217 case GreenPixelChannel: pixel=green;
break;
3218 case BluePixelChannel: pixel=blue;
break;
3219 default: pixel=Dc;
break;
3223 case ModulusAddCompositeOp:
3225 if (compose_sync == MagickFalse)
3227 pixel=(Quantum) QuantumRange*((Sc+Dc)-floor(Sc+Dc));
3230 pixel=(Quantum) QuantumRange*((Sca+Dca)-floor(Sca+Dca));
3233 case ModulusSubtractCompositeOp:
3235 if (compose_sync == MagickFalse)
3237 pixel=(Quantum) QuantumRange*((Sc-Dc)-floor(Sc-Dc));
3240 pixel=(Quantum) QuantumRange*((Sca-Dca)-floor(Sca-Dca));
3243 case MultiplyCompositeOp:
3245 if (compose_sync == MagickFalse)
3247 pixel=(double) QuantumRange*Sc*Dc;
3250 pixel=(double) QuantumRange*(Sca*Dca+Sca*(1.0-Da)+Dca*(1.0-Sa));
3253 case NegateCompositeOp:
3255 D=(Da > 0.0) ? Dca/Da : 0.0;
3256 S=(Sa > 0.0) ? Sca/Sa : 0.0;
3257 pixel=(double) QuantumRange*((1.0-fabs(1.0-S-D))*Da);
3262 pixel=(double) QuantumRange*Dca;
3265 case OutCompositeOp:
3266 case SrcOutCompositeOp:
3268 pixel=(double) QuantumRange*(Sca*(1.0-Da));
3271 case OverCompositeOp:
3272 case SrcOverCompositeOp:
3274 if ((Sa+Da*(1.0-Sa)) <= MagickEpsilon)
3277 pixel=(double) QuantumRange*gamma*((Sca+Dca*(1.0-Sa))/
3281 case OverlayCompositeOp:
3285 pixel=(double) QuantumRange*gamma*(2.0*Dca*Sca+Dca*(1.0-Sa)+
3289 pixel=(double) QuantumRange*gamma*(Da*Sa-2.0*(Sa-Sca)*(Da-Dca)+Dca*
3290 (1.0-Sa)+Sca*(1.0-Da));
3293 case PegtopLightCompositeOp:
3295 if (fabs((
double) Da) < MagickEpsilon)
3297 pixel=(double) QuantumRange*gamma*Sca;
3300 if (RoundToUnity(Sca) <= 0.5)
3301 blend=RoundToUnity(Dca/Da)-(1.0-2.0*RoundToUnity(Sca))*
3302 RoundToUnity(Dca/Da)*(1.0-RoundToUnity(Dca/Da));
3305 if (RoundToUnity(Dca/Da) <= 0.25)
3306 blend=((16.0*RoundToUnity(Dca/Da)-12.0)*RoundToUnity(Dca/Da)+
3307 4.0)*RoundToUnity(Dca/Da);
3309 blend=sqrt(RoundToUnity(Dca/Da));
3310 blend=RoundToUnity(Dca/Da)+(2.0*RoundToUnity(Sca)-1.0)*
3311 (blend-RoundToUnity(Dca/Da));
3313 pixel=(double) QuantumRange*gamma*(RoundToUnity(blend)*Da*Sa+Dca*
3317 case PinLightCompositeOp:
3327 D=(Da > 0.0) ? RoundToUnity(Dca/Da) : 0.0;
3328 S=(Sa > 0.0) ? RoundToUnity(Sca/Sa) : 0.0;
3330 blend=MagickMin(D,2.0*S);
3332 blend=MagickMax(D,2.0*S-1.0);
3333 pixel=(double) QuantumRange*gamma*(blend*RoundToUnity(Sa+Da-Sa*Da));
3336 case PlusCompositeOp:
3338 D=(Da > 0.0) ? Dca/Da : 0.0;
3339 S=(Sa > 0.0) ? Sca/Sa : 0.0;
3340 pixel=(double) QuantumRange*(RoundToUnity(Sa+Da-Sa*Da)*
3344 case ReflectCompositeOp:
3346 if (compose_sync == MagickFalse)
3349 blend=(Sc*Sc)/(1.0-Dc);
3352 pixel=(double) QuantumRange*RoundToUnity(blend);
3364 blend=(S*S)/(1.0-D);
3367 pixel=(double) QuantumRange*RoundToUnity((Sa+Da-Sa*Da)*blend);
3370 case RMSECompositeOp:
3375 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
3380 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
3386 (canvas_pixel.red-source_pixel.red)*
3387 (canvas_pixel.red-source_pixel.red)+
3388 (canvas_pixel.green-source_pixel.green)*
3389 (canvas_pixel.green-source_pixel.green)+
3390 (canvas_pixel.blue-source_pixel.blue)*
3391 (canvas_pixel.blue-source_pixel.blue)/3.0);
3394 case RedPixelChannel: pixel=gray;
break;
3395 case GreenPixelChannel: pixel=gray;
break;
3396 case BluePixelChannel: pixel=gray;
break;
3397 default: pixel=Dc;
break;
3401 case SaturateCompositeOp:
3403 if (fabs((
double) QuantumRange*Sa-(
double) TransparentAlpha) < MagickEpsilon)
3408 if (fabs((
double) QuantumRange*Da-(
double) TransparentAlpha) < MagickEpsilon)
3413 ConvertRGBToGeneric(colorspace,(
double) canvas_pixel.red,
3414 (
double) canvas_pixel.green,(
double) canvas_pixel.blue,
3415 white_luminance,illuminant,&hue,&chroma,&luma);
3416 ConvertRGBToGeneric(colorspace,(
double) source_pixel.red,
3417 (
double) source_pixel.green,(
double) source_pixel.blue,
3418 white_luminance,illuminant,&sans,&chroma,&sans);
3419 ConvertGenericToRGB(colorspace,hue,chroma,luma,
3420 white_luminance,illuminant,&red,&green,&blue);
3423 case RedPixelChannel: pixel=red;
break;
3424 case GreenPixelChannel: pixel=green;
break;
3425 case BluePixelChannel: pixel=blue;
break;
3426 default: pixel=Dc;
break;
3430 case ScreenCompositeOp:
3432 if (compose_sync == MagickFalse)
3434 pixel=(double) QuantumRange*RoundToUnity(Sc+Dc-Sc*Dc);
3445 if ((Sa+Da-Sa*Da) > 0.0)
3446 pixel=(double) QuantumRange*RoundToUnity((Sa+Da-Sa*Da)*(S+D-S*D));
3451 case SoftBurnCompositeOp:
3453 if (RoundToUnity(Dca) <= 0.0)
3456 if (RoundToUnity(Sca) >= 1.0)
3459 blend=1.0-MagickMin(1.0,(1.0-RoundToUnity(Dca))/
3461 pixel=(double) QuantumRange*gamma*RoundToUnity(blend);
3464 case SoftDodgeCompositeOp:
3466 if (RoundToUnity(Sca) <= 0.0)
3467 blend=RoundToUnity(Dca);
3469 if (RoundToUnity(Dca) >= 1.0)
3472 blend=MagickMin(1.0,RoundToUnity(Dca)/(1.0-RoundToUnity(Sca)));
3473 pixel=(double) QuantumRange*gamma*RoundToUnity(blend);
3476 case SoftLightCompositeOp:
3478 if (RoundToUnity(Sca) <= 0.5)
3480 pixel=(double) QuantumRange*gamma*(RoundToUnity(Dca)*Sa+Da*
3481 (RoundToUnity(Dca)-(1.0-2.0*RoundToUnity(Sca))*
3482 RoundToUnity(Dca)*(1.0-RoundToUnity(Dca)))+RoundToUnity(Sca)*
3483 (1.0-Da)+RoundToUnity(Dca)*(1.0-Sa));
3486 if (RoundToUnity(Dca) > 0.25)
3487 blend=sqrt(RoundToUnity(Dca));
3489 blend=((16.0*RoundToUnity(Dca)-12.0)*RoundToUnity(Dca)+4.0)*
3491 pixel=(double) QuantumRange*gamma*(RoundToUnity(Dca)*Sa+Da*
3492 (RoundToUnity(Dca)+(2.0*RoundToUnity(Sca)-1.0)*
3493 (RoundToUnity(blend)-RoundToUnity(Dca)))+RoundToUnity(Sca)*
3494 (1.0-Da)+RoundToUnity(Dca)*(1.0-Sa));
3497 case StampCompositeOp:
3499 pixel=(double) QuantumRange*RoundToUnity(Sca+Dca-1.0);
3502 case StereoCompositeOp:
3504 if (channel == RedPixelChannel)
3505 pixel=(MagickRealType) GetPixelRed(source_image,p);
3508 case ThresholdCompositeOp:
3510 S=(Sa > 0.0) ? (Sca/Sa) : 0.0;
3511 D=(Da > 0.0) ? (Dca/Da) : 0.0;
3512 if (fabs(2.0*(S-D)) < threshold)
3515 blend=D+(S-D)*amount;
3516 pixel=(double) QuantumRange*(Sa*Da*RoundToUnity(blend)+Sca*
3517 (1.0-Da)+Dca*(1.0-Sa));
3520 case VividLightCompositeOp:
3522 if ((fabs((
double) Sa) < MagickEpsilon) ||
3523 (fabs((
double) Da) < MagickEpsilon))
3525 pixel=(double) QuantumRange*gamma*(Sa*Da+Sca*(1.0-Da)+Dca*
3529 if (RoundToUnity(Sca/Sa) <= 0.0)
3532 if (RoundToUnity(Sca/Sa) < 0.5)
3533 blend=1.0-(1.0-RoundToUnity(Dca/Da))/(2.0*RoundToUnity(Sca/Sa));
3535 if (RoundToUnity(Sca/Sa) < 1.0)
3536 blend=RoundToUnity(Dca/Da)/(2.0*(1.0-RoundToUnity(Sca/Sa)));
3539 pixel=(double) QuantumRange*gamma*(Sa*Da*RoundToUnity(blend)+Sca*
3540 (1.0-Da)+Dca*(1.0-Sa));
3543 case XorCompositeOp:
3545 pixel=(double) QuantumRange*(Sca*(1.0-Da)+Dca*(1.0-Sa));
3554 q[i]=clamp != MagickFalse ? ClampPixel(pixel) : ClampToQuantum(pixel);
3556 p+=(ptrdiff_t) GetPixelChannels(source_image);
3557 channels=GetPixelChannels(source_image);
3558 if (p >= (pixels+channels*source_image->columns))
3560 q+=(ptrdiff_t) GetPixelChannels(image);
3562 if (SyncCacheViewAuthenticPixels(image_view,exception) == MagickFalse)
3564 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3569#if defined(MAGICKCORE_OPENMP_SUPPORT)
3573 proceed=SetImageProgress(image,CompositeImageTag,progress,image->rows);
3574 if (proceed == MagickFalse)
3578 source_view=DestroyCacheView(source_view);
3579 image_view=DestroyCacheView(image_view);
3580 if (canvas_image != (Image * ) NULL)
3581 canvas_image=DestroyImage(canvas_image);
3583 source_image=DestroyImage(source_image);
3613MagickExport MagickBooleanType TextureImage(Image *image,
const Image *texture,
3614 ExceptionInfo *exception)
3616#define TextureImageTag "Texture/Image"
3631 assert(image != (Image *) NULL);
3632 assert(image->signature == MagickCoreSignature);
3633 if (IsEventLogging() != MagickFalse)
3634 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
3635 if (texture == (
const Image *) NULL)
3636 return(MagickFalse);
3637 if (SetImageStorageClass(image,DirectClass,exception) == MagickFalse)
3638 return(MagickFalse);
3639 texture_image=CloneImage(texture,0,0,MagickTrue,exception);
3640 if (texture_image == (
const Image *) NULL)
3641 return(MagickFalse);
3642 (void) TransformImageColorspace(texture_image,image->colorspace,exception);
3643 (void) SetImageVirtualPixelMethod(texture_image,TileVirtualPixelMethod,
3646 if ((image->compose != CopyCompositeOp) &&
3647 ((image->compose != OverCompositeOp) ||
3648 (image->alpha_trait != UndefinedPixelTrait) ||
3649 (texture_image->alpha_trait != UndefinedPixelTrait)))
3654 for (y=0; y < (ssize_t) image->rows; y+=(ssize_t) texture_image->rows)
3659 if (status == MagickFalse)
3661 for (x=0; x < (ssize_t) image->columns; x+=(ssize_t) texture_image->columns)
3666 thread_status=CompositeImage(image,texture_image,image->compose,
3667 MagickTrue,x+texture_image->tile_offset.x,y+
3668 texture_image->tile_offset.y,exception);
3669 if (thread_status == MagickFalse)
3671 status=thread_status;
3675 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3680 proceed=SetImageProgress(image,TextureImageTag,(MagickOffsetType) y,
3682 if (proceed == MagickFalse)
3686 (void) SetImageProgress(image,TextureImageTag,(MagickOffsetType)
3687 image->rows,image->rows);
3688 texture_image=DestroyImage(texture_image);
3695 texture_view=AcquireVirtualCacheView(texture_image,exception);
3696 image_view=AcquireAuthenticCacheView(image,exception);
3697#if defined(MAGICKCORE_OPENMP_SUPPORT)
3698 #pragma omp parallel for schedule(static) shared(status) \
3699 magick_number_threads(texture_image,image,image->rows,2)
3701 for (y=0; y < (ssize_t) image->rows; y++)
3719 if (status == MagickFalse)
3721 pixels=GetCacheViewVirtualPixels(texture_view,texture_image->tile_offset.x,
3722 (y+texture_image->tile_offset.y) % (ssize_t) texture_image->rows,
3723 texture_image->columns,1,exception);
3724 q=QueueCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
3725 if ((pixels == (
const Quantum *) NULL) || (q == (Quantum *) NULL))
3730 for (x=0; x < (ssize_t) image->columns; x+=(ssize_t) texture_image->columns)
3736 width=texture_image->columns;
3737 if ((x+(ssize_t) width) > (ssize_t) image->columns)
3738 width=image->columns-(size_t) x;
3739 for (j=0; j < (ssize_t) width; j++)
3744 for (i=0; i < (ssize_t) GetPixelChannels(texture_image); i++)
3746 PixelChannel channel = GetPixelChannelChannel(texture_image,i);
3747 PixelTrait traits = GetPixelChannelTraits(image,channel);
3748 PixelTrait texture_traits=GetPixelChannelTraits(texture_image,
3750 if ((traits == UndefinedPixelTrait) ||
3751 (texture_traits == UndefinedPixelTrait))
3753 SetPixelChannel(image,channel,p[i],q);
3755 p+=(ptrdiff_t) GetPixelChannels(texture_image);
3756 q+=(ptrdiff_t) GetPixelChannels(image);
3759 sync=SyncCacheViewAuthenticPixels(image_view,exception);
3760 if (sync == MagickFalse)
3762 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3767 proceed=SetImageProgress(image,TextureImageTag,(MagickOffsetType) y,
3769 if (proceed == MagickFalse)
3773 texture_view=DestroyCacheView(texture_view);
3774 image_view=DestroyCacheView(image_view);
3775 texture_image=DestroyImage(texture_image);