49#include "magick/studio.h"
50#include "magick/accelerate-private.h"
51#include "magick/annotate.h"
52#include "magick/artifact.h"
53#include "magick/attribute.h"
54#include "magick/cache.h"
55#include "magick/cache-view.h"
56#include "magick/cache-private.h"
57#include "magick/channel.h"
58#include "magick/color.h"
59#include "magick/color-private.h"
60#include "magick/colorspace.h"
61#include "magick/colorspace-private.h"
62#include "magick/composite.h"
63#include "magick/decorate.h"
64#include "magick/distort.h"
65#include "magick/draw.h"
66#include "magick/effect.h"
67#include "magick/enhance.h"
68#include "magick/exception.h"
69#include "magick/exception-private.h"
70#include "magick/gem.h"
71#include "magick/geometry.h"
72#include "magick/layer.h"
73#include "magick/list.h"
74#include "magick/log.h"
75#include "magick/image.h"
76#include "magick/image-private.h"
77#include "magick/magick.h"
78#include "magick/memory_.h"
79#include "magick/memory-private.h"
80#include "magick/monitor.h"
81#include "magick/monitor-private.h"
82#include "magick/opencl-private.h"
83#include "magick/option.h"
84#include "magick/pixel-accessor.h"
85#include "magick/pixel-private.h"
86#include "magick/property.h"
87#include "magick/quantum.h"
88#include "magick/quantum-private.h"
89#include "magick/random_.h"
90#include "magick/random-private.h"
91#include "magick/resample.h"
92#include "magick/resample-private.h"
93#include "magick/resize.h"
94#include "magick/resource_.h"
95#include "magick/splay-tree.h"
96#include "magick/statistic.h"
97#include "magick/string_.h"
98#include "magick/string-private.h"
99#include "magick/thread-private.h"
100#include "magick/threshold.h"
101#include "magick/transform.h"
102#include "magick/utility.h"
103#include "magick/visual-effects.h"
137MagickExport
Image *AddNoiseImage(
const Image *image,
const NoiseType noise_type,
143 noise_image=AddNoiseImageChannel(image,DefaultChannels,noise_type,exception);
147MagickExport
Image *AddNoiseImageChannel(
const Image *image,
148 const ChannelType channel,
const NoiseType noise_type,
ExceptionInfo *exception)
150#define AddNoiseImageTag "AddNoise/Image"
172 **magick_restrict random_info;
177#if defined(MAGICKCORE_OPENMP_SUPPORT)
185 assert(image != (
const Image *) NULL);
186 assert(image->signature == MagickCoreSignature);
188 assert(exception->signature == MagickCoreSignature);
189 if (IsEventLogging() != MagickFalse)
190 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
191#if defined(MAGICKCORE_OPENCL_SUPPORT)
192 noise_image=AccelerateAddNoiseImage(image,channel,noise_type,exception);
193 if (noise_image != (
Image *) NULL)
196 noise_image=CloneImage(image,0,0,MagickTrue,exception);
197 if (noise_image == (
Image *) NULL)
198 return((
Image *) NULL);
199 if (SetImageStorageClass(noise_image,DirectClass) == MagickFalse)
201 InheritException(exception,&noise_image->exception);
202 noise_image=DestroyImage(noise_image);
203 return((
Image *) NULL);
209 option=GetImageArtifact(image,
"attenuate");
210 if (option != (
char *) NULL)
211 attenuate=StringToDouble(option,(
char **) NULL);
214 random_info=AcquireRandomInfoTLS();
215 image_view=AcquireVirtualCacheView(image,exception);
216 noise_view=AcquireAuthenticCacheView(noise_image,exception);
217#if defined(MAGICKCORE_OPENMP_SUPPORT)
218 key=GetRandomSecretKey(random_info[0]);
219 #pragma omp parallel for schedule(static) shared(progress,status) \
220 magick_number_threads(image,noise_image,image->rows,key == ~0UL ? 0 : 2)
222 for (y=0; y < (ssize_t) image->rows; y++)
225 id = GetOpenMPThreadId();
231 *magick_restrict indexes;
237 *magick_restrict noise_indexes;
245 if (status == MagickFalse)
247 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
248 q=GetCacheViewAuthenticPixels(noise_view,0,y,noise_image->columns,1,
255 indexes=GetCacheViewVirtualIndexQueue(image_view);
256 noise_indexes=GetCacheViewAuthenticIndexQueue(noise_view);
257 for (x=0; x < (ssize_t) image->columns; x++)
259 if ((channel & RedChannel) != 0)
260 SetPixelRed(q,ClampToQuantum(GenerateDifferentialNoise(random_info[
id],
261 GetPixelRed(p),noise_type,attenuate)));
262 if (IsGrayColorspace(image->colorspace) != MagickFalse)
264 SetPixelGreen(q,GetPixelRed(q));
265 SetPixelBlue(q,GetPixelRed(q));
269 if ((channel & GreenChannel) != 0)
270 SetPixelGreen(q,ClampToQuantum(GenerateDifferentialNoise(
271 random_info[
id],GetPixelGreen(p),noise_type,attenuate)));
272 if ((channel & BlueChannel) != 0)
273 SetPixelBlue(q,ClampToQuantum(GenerateDifferentialNoise(
274 random_info[
id],GetPixelBlue(p),noise_type,attenuate)));
276 if ((channel & OpacityChannel) != 0)
277 SetPixelOpacity(q,ClampToQuantum(GenerateDifferentialNoise(
278 random_info[
id],GetPixelOpacity(p),noise_type,attenuate)));
279 if (((channel & IndexChannel) != 0) &&
280 (image->colorspace == CMYKColorspace))
281 SetPixelIndex(noise_indexes+x,ClampToQuantum(
282 GenerateDifferentialNoise(random_info[
id],GetPixelIndex(
283 indexes+x),noise_type,attenuate)));
287 sync=SyncCacheViewAuthenticPixels(noise_view,exception);
288 if (sync == MagickFalse)
290 if (image->progress_monitor != (MagickProgressMonitor) NULL)
295#if defined(MAGICKCORE_OPENMP_SUPPORT)
299 proceed=SetImageProgress(image,AddNoiseImageTag,progress,image->rows);
300 if (proceed == MagickFalse)
304 noise_view=DestroyCacheView(noise_view);
305 image_view=DestroyCacheView(image_view);
306 random_info=DestroyRandomInfoTLS(random_info);
307 if (status == MagickFalse)
308 noise_image=DestroyImage(noise_image);
340MagickExport
Image *BlueShiftImage(
const Image *image,
const double factor,
343#define BlueShiftImageTag "BlueShift/Image"
364 assert(image != (
const Image *) NULL);
365 assert(image->signature == MagickCoreSignature);
367 assert(exception->signature == MagickCoreSignature);
368 if (IsEventLogging() != MagickFalse)
369 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
370 shift_image=CloneImage(image,0,0,MagickTrue,exception);
371 if (shift_image == (
Image *) NULL)
372 return((
Image *) NULL);
373 if (SetImageStorageClass(shift_image,DirectClass) == MagickFalse)
375 InheritException(exception,&shift_image->exception);
376 shift_image=DestroyImage(shift_image);
377 return((
Image *) NULL);
384 image_view=AcquireVirtualCacheView(image,exception);
385 shift_view=AcquireAuthenticCacheView(shift_image,exception);
386#if defined(MAGICKCORE_OPENMP_SUPPORT)
387 #pragma omp parallel for schedule(static) shared(progress,status) \
388 magick_number_threads(image,shift_image,image->rows,2)
390 for (y=0; y < (ssize_t) image->rows; y++)
410 if (status == MagickFalse)
412 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
413 q=QueueCacheViewAuthenticPixels(shift_view,0,y,shift_image->columns,1,
420 for (x=0; x < (ssize_t) image->columns; x++)
422 quantum=GetPixelRed(p);
423 if (GetPixelGreen(p) < quantum)
424 quantum=GetPixelGreen(p);
425 if (GetPixelBlue(p) < quantum)
426 quantum=GetPixelBlue(p);
427 pixel.red=0.5*((MagickRealType) GetPixelRed(p)+factor*
428 (MagickRealType) quantum);
429 pixel.green=0.5*((MagickRealType) GetPixelGreen(p)+factor*
430 (MagickRealType) quantum);
431 pixel.blue=0.5*((MagickRealType) GetPixelBlue(p)+factor*
432 (MagickRealType) quantum);
433 quantum=GetPixelRed(p);
434 if (GetPixelGreen(p) > quantum)
435 quantum=GetPixelGreen(p);
436 if (GetPixelBlue(p) > quantum)
437 quantum=GetPixelBlue(p);
438 pixel.red=0.5*((MagickRealType) pixel.red+factor*
439 (MagickRealType) quantum);
440 pixel.green=0.5*((MagickRealType) pixel.green+factor*
441 (MagickRealType) quantum);
442 pixel.blue=0.5*((MagickRealType) pixel.blue+factor*
443 (MagickRealType) quantum);
444 SetPixelRed(q,ClampToQuantum(pixel.red));
445 SetPixelGreen(q,ClampToQuantum(pixel.green));
446 SetPixelBlue(q,ClampToQuantum(pixel.blue));
450 sync=SyncCacheViewAuthenticPixels(shift_view,exception);
451 if (sync == MagickFalse)
453 if (image->progress_monitor != (MagickProgressMonitor) NULL)
458#if defined(MAGICKCORE_OPENMP_SUPPORT)
462 proceed=SetImageProgress(image,BlueShiftImageTag,progress,image->rows);
463 if (proceed == MagickFalse)
467 image_view=DestroyCacheView(image_view);
468 shift_view=DestroyCacheView(shift_view);
469 if (status == MagickFalse)
470 shift_image=DestroyImage(shift_image);
505MagickExport
Image *CharcoalImage(
const Image *image,
const double radius,
515 assert(image != (
Image *) NULL);
516 assert(image->signature == MagickCoreSignature);
518 assert(exception->signature == MagickCoreSignature);
519 if (IsEventLogging() != MagickFalse)
520 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
521 edge_image=EdgeImage(image,radius,exception);
522 if (edge_image == (
Image *) NULL)
523 return((
Image *) NULL);
524 charcoal_image=(
Image *) NULL;
525 status=ClampImage(edge_image);
526 if (status != MagickFalse)
527 charcoal_image=BlurImage(edge_image,radius,sigma,exception);
528 edge_image=DestroyImage(edge_image);
529 if (charcoal_image == (
Image *) NULL)
530 return((
Image *) NULL);
531 status=NormalizeImage(charcoal_image);
532 if (status != MagickFalse)
533 status=NegateImage(charcoal_image,MagickFalse);
534 if (status != MagickFalse)
535 status=GrayscaleImage(charcoal_image,image->intensity);
536 if (status == MagickFalse)
537 charcoal_image=DestroyImage(charcoal_image);
538 return(charcoal_image);
574MagickExport
Image *ColorizeImage(
const Image *image,
const char *opacity,
577#define ColorizeImageTag "Colorize/Image"
607 assert(image != (
const Image *) NULL);
608 assert(image->signature == MagickCoreSignature);
610 assert(exception->signature == MagickCoreSignature);
611 if (IsEventLogging() != MagickFalse)
612 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
613 colorize_image=CloneImage(image,0,0,MagickTrue,exception);
614 if (colorize_image == (
Image *) NULL)
615 return((
Image *) NULL);
616 if (SetImageStorageClass(colorize_image,DirectClass) == MagickFalse)
618 InheritException(exception,&colorize_image->exception);
619 colorize_image=DestroyImage(colorize_image);
620 return((
Image *) NULL);
622 if ((IsGrayColorspace(image->colorspace) != MagickFalse) ||
623 (IsPixelGray(&colorize) != MagickFalse))
624 (void) SetImageColorspace(colorize_image,sRGBColorspace);
625 if ((colorize_image->matte == MagickFalse) &&
626 (colorize.opacity != OpaqueOpacity))
627 (void) SetImageAlphaChannel(colorize_image,OpaqueAlphaChannel);
628 if (opacity == (
const char *) NULL)
629 return(colorize_image);
633 flags=ParseGeometry(opacity,&geometry_info);
634 pixel.red=geometry_info.rho;
635 pixel.green=geometry_info.rho;
636 pixel.blue=geometry_info.rho;
637 pixel.opacity=(MagickRealType) OpaqueOpacity;
638 if ((flags & SigmaValue) != 0)
639 pixel.green=geometry_info.sigma;
640 if ((flags & XiValue) != 0)
641 pixel.blue=geometry_info.xi;
642 if ((flags & PsiValue) != 0)
643 pixel.opacity=geometry_info.psi;
649 image_view=AcquireVirtualCacheView(image,exception);
650 colorize_view=AcquireAuthenticCacheView(colorize_image,exception);
651#if defined(MAGICKCORE_OPENMP_SUPPORT)
652 #pragma omp parallel for schedule(static) shared(progress,status) \
653 magick_number_threads(image,colorize_image,image->rows,2)
655 for (y=0; y < (ssize_t) image->rows; y++)
669 if (status == MagickFalse)
671 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
672 q=QueueCacheViewAuthenticPixels(colorize_view,0,y,colorize_image->columns,1,
679 for (x=0; x < (ssize_t) image->columns; x++)
681 SetPixelRed(q,(((MagickRealType) GetPixelRed(p)*(100.0-pixel.red)+
682 (MagickRealType) colorize.red*(MagickRealType) pixel.red)/100.0));
683 SetPixelGreen(q,(((MagickRealType) GetPixelGreen(p)*(100.0-pixel.green)+
684 (MagickRealType) colorize.green*(MagickRealType) pixel.green)/100.0));
685 SetPixelBlue(q,(((MagickRealType) GetPixelBlue(p)*(100.0-pixel.blue)+
686 (MagickRealType) colorize.blue*(MagickRealType) pixel.blue)/100.0));
687 if (colorize_image->matte == MagickFalse)
688 SetPixelOpacity(q,GetPixelOpacity(p));
690 SetPixelOpacity(q,(((MagickRealType) GetPixelOpacity(p)*(100.0-
691 pixel.opacity)+(MagickRealType) colorize.opacity*pixel.opacity)/
696 sync=SyncCacheViewAuthenticPixels(colorize_view,exception);
697 if (sync == MagickFalse)
699 if (image->progress_monitor != (MagickProgressMonitor) NULL)
704#if defined(MAGICKCORE_OPENMP_SUPPORT)
708 proceed=SetImageProgress(image,ColorizeImageTag,progress,image->rows);
709 if (proceed == MagickFalse)
713 image_view=DestroyCacheView(image_view);
714 colorize_view=DestroyCacheView(colorize_view);
715 if (status == MagickFalse)
716 colorize_image=DestroyImage(colorize_image);
717 return(colorize_image);
753MagickExport
Image *ColorMatrixImage(
const Image *image,
756#define ColorMatrixImageTag "ColorMatrix/Image"
765 { 1.0, 0.0, 0.0, 0.0, 0.0, 0.0 },
766 { 0.0, 1.0, 0.0, 0.0, 0.0, 0.0 },
767 { 0.0, 0.0, 1.0, 0.0, 0.0, 0.0 },
768 { 0.0, 0.0, 0.0, 1.0, 0.0, 0.0 },
769 { 0.0, 0.0, 0.0, 0.0, 1.0, 0.0 },
770 { 0.0, 0.0, 0.0, 0.0, 0.0, 1.0 }
793 assert(image != (
Image *) NULL);
794 assert(image->signature == MagickCoreSignature);
796 assert(exception->signature == MagickCoreSignature);
797 if (IsEventLogging() != MagickFalse)
798 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
800 for (v=0; v < (ssize_t) color_matrix->height; v++)
801 for (u=0; u < (ssize_t) color_matrix->width; u++)
803 if ((v < 6) && (u < 6))
804 ColorMatrix[v][u]=color_matrix->values[i];
810 color_image=CloneImage(image,0,0,MagickTrue,exception);
811 if (color_image == (
Image *) NULL)
812 return((
Image *) NULL);
813 if (SetImageStorageClass(color_image,DirectClass) == MagickFalse)
815 InheritException(exception,&color_image->exception);
816 color_image=DestroyImage(color_image);
817 return((
Image *) NULL);
819 if (image->debug != MagickFalse)
822 format[MaxTextExtent],
825 (void) LogMagickEvent(TransformEvent,GetMagickModule(),
826 " ColorMatrix image with color matrix:");
827 message=AcquireString(
"");
828 for (v=0; v < 6; v++)
831 (void) FormatLocaleString(format,MaxTextExtent,
"%.20g: ",(
double) v);
832 (void) ConcatenateString(&message,format);
833 for (u=0; u < 6; u++)
835 (void) FormatLocaleString(format,MaxTextExtent,
"%+f ",
837 (void) ConcatenateString(&message,format);
839 (void) LogMagickEvent(TransformEvent,GetMagickModule(),
"%s",message);
841 message=DestroyString(message);
848 image_view=AcquireVirtualCacheView(image,exception);
849 color_view=AcquireAuthenticCacheView(color_image,exception);
850#if defined(MAGICKCORE_OPENMP_SUPPORT)
851 #pragma omp parallel for schedule(static) shared(progress,status) \
852 magick_number_threads(image,color_image,image->rows,1)
854 for (y=0; y < (ssize_t) image->rows; y++)
860 *magick_restrict indexes;
869 *magick_restrict color_indexes;
874 if (status == MagickFalse)
876 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
877 q=GetCacheViewAuthenticPixels(color_view,0,y,color_image->columns,1,
884 indexes=GetCacheViewVirtualIndexQueue(image_view);
885 color_indexes=GetCacheViewAuthenticIndexQueue(color_view);
886 for (x=0; x < (ssize_t) image->columns; x++)
894 height=color_matrix->height > 6 ? 6UL : color_matrix->height;
895 for (v=0; v < (ssize_t) height; v++)
897 pixel=ColorMatrix[v][0]*(MagickRealType) GetPixelRed(p)+
898 ColorMatrix[v][1]*(MagickRealType) GetPixelGreen(p)+
899 ColorMatrix[v][2]*(MagickRealType) GetPixelBlue(p);
900 if (image->matte != MagickFalse)
901 pixel+=ColorMatrix[v][3]*((MagickRealType) QuantumRange-
902 (MagickRealType) GetPixelOpacity(p));
903 if (image->colorspace == CMYKColorspace)
904 pixel+=ColorMatrix[v][4]*(MagickRealType) GetPixelIndex(indexes+x);
905 pixel+=(MagickRealType) QuantumRange*ColorMatrix[v][5];
908 case 0: SetPixelRed(q,ClampToQuantum(pixel));
break;
909 case 1: SetPixelGreen(q,ClampToQuantum(pixel));
break;
910 case 2: SetPixelBlue(q,ClampToQuantum(pixel));
break;
913 if (image->matte != MagickFalse)
914 SetPixelAlpha(q,ClampToQuantum(pixel));
919 if (image->colorspace == CMYKColorspace)
920 SetPixelIndex(color_indexes+x,ClampToQuantum(pixel));
928 if (SyncCacheViewAuthenticPixels(color_view,exception) == MagickFalse)
930 if (image->progress_monitor != (MagickProgressMonitor) NULL)
935#if defined(MAGICKCORE_OPENMP_SUPPORT)
939 proceed=SetImageProgress(image,ColorMatrixImageTag,progress,
941 if (proceed == MagickFalse)
945 color_view=DestroyCacheView(color_view);
946 image_view=DestroyCacheView(image_view);
947 if (status == MagickFalse)
948 color_image=DestroyImage(color_image);
986MagickExport
Image *ImplodeImage(
const Image *image,
const double amount,
989#define ImplodeImageTag "Implode/Image"
1020 assert(image != (
Image *) NULL);
1021 assert(image->signature == MagickCoreSignature);
1023 assert(exception->signature == MagickCoreSignature);
1024 if (IsEventLogging() != MagickFalse)
1025 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1026 implode_image=CloneImage(image,0,0,MagickTrue,exception);
1027 if (implode_image == (
Image *) NULL)
1028 return((
Image *) NULL);
1029 if (SetImageStorageClass(implode_image,DirectClass) == MagickFalse)
1031 InheritException(exception,&implode_image->exception);
1032 implode_image=DestroyImage(implode_image);
1033 return((
Image *) NULL);
1035 if (implode_image->background_color.opacity != OpaqueOpacity)
1036 implode_image->matte=MagickTrue;
1042 center.x=0.5*image->columns;
1043 center.y=0.5*image->rows;
1045 if (image->columns > image->rows)
1046 scale.y=(double) image->columns*PerceptibleReciprocal((
double)
1049 if (image->columns < image->rows)
1051 scale.x=(double) image->rows*PerceptibleReciprocal((
double)
1060 GetMagickPixelPacket(implode_image,&zero);
1061 image_view=AcquireVirtualCacheView(image,exception);
1062 implode_view=AcquireAuthenticCacheView(implode_image,exception);
1063#if defined(MAGICKCORE_OPENMP_SUPPORT)
1064 #pragma omp parallel for schedule(static) shared(progress,status) \
1065 magick_number_threads(image,implode_image,image->rows,1)
1067 for (y=0; y < (ssize_t) image->rows; y++)
1073 *magick_restrict implode_indexes;
1087 if (status == MagickFalse)
1089 q=GetCacheViewAuthenticPixels(implode_view,0,y,implode_image->columns,1,
1096 implode_indexes=GetCacheViewAuthenticIndexQueue(implode_view);
1097 delta.y=scale.y*((double) y-center.y);
1099 for (x=0; x < (ssize_t) image->columns; x++)
1104 delta.x=scale.x*((double) x-center.x);
1105 distance=delta.x*delta.x+delta.y*delta.y;
1106 if (distance < (radius*radius))
1119 factor=pow(sin((
double) (MagickPI*sqrt(distance)*
1120 PerceptibleReciprocal(radius)/2.0)),-amount);
1121 offset.x=factor*delta.x*PerceptibleReciprocal(scale.x)+center.x;
1122 offset.y=factor*delta.y*PerceptibleReciprocal(scale.y)+center.y;
1123 if ((IsValidPixelOffset(offset.x,image->columns) != MagickFalse) &&
1124 (IsValidPixelOffset(offset.y,image->rows) != MagickFalse))
1125 status=InterpolateMagickPixelPacket(image,image_view,
1126 UndefinedInterpolatePixel,offset.x,offset.y,&pixel,exception);
1127 if (status == MagickFalse)
1129 SetPixelPacket(implode_image,&pixel,q,implode_indexes+x);
1133 if (SyncCacheViewAuthenticPixels(implode_view,exception) == MagickFalse)
1135 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1140#if defined(MAGICKCORE_OPENMP_SUPPORT)
1144 proceed=SetImageProgress(image,ImplodeImageTag,progress,image->rows);
1145 if (proceed == MagickFalse)
1149 implode_view=DestroyCacheView(implode_view);
1150 image_view=DestroyCacheView(image_view);
1151 if (status == MagickFalse)
1152 implode_image=DestroyImage(implode_image);
1153 return(implode_image);
1186MagickExport
Image *MorphImages(
const Image *image,
1189#define MorphImageTag "Morph/Image"
1217 assert(image != (
Image *) NULL);
1218 assert(image->signature == MagickCoreSignature);
1220 assert(exception->signature == MagickCoreSignature);
1221 if (IsEventLogging() != MagickFalse)
1222 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1223 morph_images=CloneImage(image,0,0,MagickTrue,exception);
1224 if (morph_images == (
Image *) NULL)
1225 return((
Image *) NULL);
1226 if (GetNextImageInList(image) == (
Image *) NULL)
1231 for (i=1; i < (ssize_t) number_frames; i++)
1233 morph_image=CloneImage(image,0,0,MagickTrue,exception);
1234 if (morph_image == (
Image *) NULL)
1236 morph_images=DestroyImageList(morph_images);
1237 return((
Image *) NULL);
1239 AppendImageToList(&morph_images,morph_image);
1240 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1245 proceed=SetImageProgress(image,MorphImageTag,(MagickOffsetType) i,
1247 if (proceed == MagickFalse)
1251 return(GetFirstImageInList(morph_images));
1259 for ( ; GetNextImageInList(next) != (
Image *) NULL; next=GetNextImageInList(next))
1261 for (i=0; i < (ssize_t) number_frames; i++)
1267 beta=(double) (i+1.0)/(double) (number_frames+1.0);
1269 morph_image=ResizeImage(next,(
size_t) (alpha*next->columns+beta*
1270 GetNextImageInList(next)->columns+0.5),(
size_t) (alpha*
1271 next->rows+beta*GetNextImageInList(next)->rows+0.5),
1272 next->filter,next->blur,exception);
1273 if (morph_image == (
Image *) NULL)
1275 morph_images=DestroyImageList(morph_images);
1276 return((
Image *) NULL);
1278 if (SetImageStorageClass(morph_image,DirectClass) == MagickFalse)
1280 InheritException(exception,&morph_image->exception);
1281 morph_image=DestroyImage(morph_image);
1282 return((
Image *) NULL);
1284 AppendImageToList(&morph_images,morph_image);
1285 morph_images=GetLastImageInList(morph_images);
1286 morph_image=ResizeImage(GetNextImageInList(next),morph_images->columns,
1287 morph_images->rows,GetNextImageInList(next)->filter,
1288 GetNextImageInList(next)->blur,exception);
1289 if (morph_image == (
Image *) NULL)
1291 morph_images=DestroyImageList(morph_images);
1292 return((
Image *) NULL);
1294 image_view=AcquireVirtualCacheView(morph_image,exception);
1295 morph_view=AcquireAuthenticCacheView(morph_images,exception);
1296#if defined(MAGICKCORE_OPENMP_SUPPORT)
1297 #pragma omp parallel for schedule(static) shared(status) \
1298 magick_number_threads(morph_image,morph_image,morph_image->rows,1)
1300 for (y=0; y < (ssize_t) morph_images->rows; y++)
1314 if (status == MagickFalse)
1316 p=GetCacheViewVirtualPixels(image_view,0,y,morph_image->columns,1,
1318 q=GetCacheViewAuthenticPixels(morph_view,0,y,morph_images->columns,1,
1325 for (x=0; x < (ssize_t) morph_images->columns; x++)
1327 SetPixelRed(q,ClampToQuantum(alpha*(MagickRealType)
1328 GetPixelRed(q)+beta*(MagickRealType) GetPixelRed(p)));
1329 SetPixelGreen(q,ClampToQuantum(alpha*(MagickRealType)
1330 GetPixelGreen(q)+beta*(MagickRealType) GetPixelGreen(p)));
1331 SetPixelBlue(q,ClampToQuantum(alpha*(MagickRealType)
1332 GetPixelBlue(q)+beta*(MagickRealType) GetPixelBlue(p)));
1333 SetPixelOpacity(q,ClampToQuantum(alpha*(MagickRealType)
1334 GetPixelOpacity(q)+beta*(MagickRealType) GetPixelOpacity(p)));
1338 sync=SyncCacheViewAuthenticPixels(morph_view,exception);
1339 if (sync == MagickFalse)
1342 morph_view=DestroyCacheView(morph_view);
1343 image_view=DestroyCacheView(image_view);
1344 morph_image=DestroyImage(morph_image);
1346 if (i < (ssize_t) number_frames)
1351 morph_image=CloneImage(GetNextImageInList(next),0,0,MagickTrue,exception);
1352 if (morph_image == (
Image *) NULL)
1354 morph_images=DestroyImageList(morph_images);
1355 return((
Image *) NULL);
1357 AppendImageToList(&morph_images,morph_image);
1358 morph_images=GetLastImageInList(morph_images);
1359 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1364 proceed=SetImageProgress(image,MorphImageTag,scene,
1365 GetImageListLength(image));
1366 if (proceed == MagickFalse)
1371 if (GetNextImageInList(next) != (
Image *) NULL)
1373 morph_images=DestroyImageList(morph_images);
1374 return((
Image *) NULL);
1376 return(GetFirstImageInList(morph_images));
1411static inline Quantum PlasmaPixel(
RandomInfo *magick_restrict random_info,
1412 const MagickRealType pixel,
const double noise)
1417 plasma=pixel+noise*GetPseudoRandomValue(random_info)-noise/2.0;
1418 return(ClampToQuantum(plasma));
1421MagickExport MagickBooleanType PlasmaImageProxy(
Image *image,
1424 const SegmentInfo *magick_restrict segment,
size_t attenuate,
size_t depth)
1445 if ((fabs(segment->x2-segment->x1) < MagickEpsilon) &&
1446 (fabs(segment->y2-segment->y1) < MagickEpsilon))
1458 x_mid=CastDoubleToLong(ceil((segment->x1+segment->x2)/2-0.5));
1459 y_mid=CastDoubleToLong(ceil((segment->y1+segment->y2)/2-0.5));
1460 local_info=(*segment);
1461 local_info.x2=(double) x_mid;
1462 local_info.y2=(double) y_mid;
1463 status=PlasmaImageProxy(image,image_view,u_view,v_view,random_info,
1464 &local_info,attenuate,depth);
1465 local_info=(*segment);
1466 local_info.y1=(double) y_mid;
1467 local_info.x2=(double) x_mid;
1468 status&=PlasmaImageProxy(image,image_view,u_view,v_view,random_info,
1469 &local_info,attenuate,depth);
1470 local_info=(*segment);
1471 local_info.x1=(double) x_mid;
1472 local_info.y2=(double) y_mid;
1473 status&=PlasmaImageProxy(image,image_view,u_view,v_view,random_info,
1474 &local_info,attenuate,depth);
1475 local_info=(*segment);
1476 local_info.x1=(double) x_mid;
1477 local_info.y1=(double) y_mid;
1478 status&=PlasmaImageProxy(image,image_view,u_view,v_view,random_info,
1479 &local_info,attenuate,depth);
1480 return(status == 0 ? MagickFalse : MagickTrue);
1482 x_mid=CastDoubleToLong(ceil((segment->x1+segment->x2)/2-0.5));
1483 y_mid=CastDoubleToLong(ceil((segment->y1+segment->y2)/2-0.5));
1484 if ((fabs(segment->x1-x_mid) < MagickEpsilon) &&
1485 (fabs(segment->x2-x_mid) < MagickEpsilon) &&
1486 (fabs(segment->y1-y_mid) < MagickEpsilon) &&
1487 (fabs(segment->y2-y_mid) < MagickEpsilon))
1488 return(MagickFalse);
1493 exception=(&image->exception);
1494 plasma=(double) QuantumRange/(2.0*attenuate);
1495 if ((fabs(segment->x1-x_mid) >= MagickEpsilon) ||
1496 (fabs(segment->x2-x_mid) >= MagickEpsilon))
1504 x=CastDoubleToLong(ceil(segment->x1-0.5));
1505 (void) GetOneCacheViewVirtualPixel(u_view,x,CastDoubleToLong(ceil(
1506 segment->y1-0.5)),&u,exception);
1507 (void) GetOneCacheViewVirtualPixel(v_view,x,CastDoubleToLong(ceil(
1508 segment->y2-0.5)),&v,exception);
1509 q=QueueCacheViewAuthenticPixels(image_view,x,y_mid,1,1,exception);
1512 SetPixelRed(q,PlasmaPixel(random_info,((MagickRealType) u.red+
1513 (MagickRealType) v.red)/2.0,plasma));
1514 SetPixelGreen(q,PlasmaPixel(random_info,((MagickRealType) u.green+
1515 (MagickRealType) v.green)/2.0,plasma));
1516 SetPixelBlue(q,PlasmaPixel(random_info,((MagickRealType) u.blue+
1517 (MagickRealType) v.blue)/2.0,plasma));
1518 status=SyncCacheViewAuthenticPixels(image_view,exception);
1519 if (fabs(segment->x1-segment->x2) >= MagickEpsilon)
1524 x=CastDoubleToLong(ceil(segment->x2-0.5));
1525 (void) GetOneCacheViewVirtualPixel(u_view,x,CastDoubleToLong(ceil(
1526 segment->y1-0.5)),&u,exception);
1527 (void) GetOneCacheViewVirtualPixel(v_view,x,CastDoubleToLong(ceil(
1528 segment->y2-0.5)),&v,exception);
1529 q=QueueCacheViewAuthenticPixels(image_view,x,y_mid,1,1,exception);
1531 return(MagickFalse);
1532 SetPixelRed(q,PlasmaPixel(random_info,((MagickRealType) u.red+
1533 (MagickRealType) v.red)/2.0,plasma));
1534 SetPixelGreen(q,PlasmaPixel(random_info,((MagickRealType) u.green+
1535 (MagickRealType) v.green)/2.0,plasma));
1536 SetPixelBlue(q,PlasmaPixel(random_info,((MagickRealType) u.blue+
1537 (MagickRealType) v.blue)/2.0,plasma));
1538 status=SyncCacheViewAuthenticPixels(image_view,exception);
1541 if ((fabs(segment->y1-y_mid) >= MagickEpsilon) ||
1542 (fabs(segment->y2-y_mid) >= MagickEpsilon))
1544 if ((fabs(segment->x1-x_mid) >= MagickEpsilon) ||
1545 (fabs(segment->y2-y_mid) >= MagickEpsilon))
1553 y=CastDoubleToLong(ceil(segment->y2-0.5));
1554 (void) GetOneCacheViewVirtualPixel(u_view,CastDoubleToLong(ceil(
1555 segment->x1-0.5)),y,&u,exception);
1556 (void) GetOneCacheViewVirtualPixel(v_view,CastDoubleToLong(ceil(
1557 segment->x2-0.5)),y,&v,exception);
1558 q=QueueCacheViewAuthenticPixels(image_view,x_mid,y,1,1,exception);
1561 SetPixelRed(q,PlasmaPixel(random_info,((MagickRealType) u.red+
1562 (MagickRealType) v.red)/2.0,plasma));
1563 SetPixelGreen(q,PlasmaPixel(random_info,((MagickRealType) u.green+
1564 (MagickRealType) v.green)/2.0,plasma));
1565 SetPixelBlue(q,PlasmaPixel(random_info,((MagickRealType) u.blue+
1566 (MagickRealType) v.blue)/2.0,plasma));
1567 status=SyncCacheViewAuthenticPixels(image_view,exception);
1569 if (fabs(segment->y1-segment->y2) >= MagickEpsilon)
1577 y=CastDoubleToLong(ceil(segment->y1-0.5));
1578 (void) GetOneCacheViewVirtualPixel(u_view,CastDoubleToLong(ceil(
1579 segment->x1-0.5)),y,&u,exception);
1580 (void) GetOneCacheViewVirtualPixel(v_view,CastDoubleToLong(ceil(
1581 segment->x2-0.5)),y,&v,exception);
1582 q=QueueCacheViewAuthenticPixels(image_view,x_mid,y,1,1,exception);
1585 SetPixelRed(q,PlasmaPixel(random_info,((MagickRealType) u.red+
1586 (MagickRealType) v.red)/2.0,plasma));
1587 SetPixelGreen(q,PlasmaPixel(random_info,((MagickRealType) u.green+
1588 (MagickRealType) v.green)/2.0,plasma));
1589 SetPixelBlue(q,PlasmaPixel(random_info,((MagickRealType) u.blue+
1590 (MagickRealType) v.blue)/2.0,plasma));
1591 status=SyncCacheViewAuthenticPixels(image_view,exception);
1594 if ((fabs(segment->x1-segment->x2) >= MagickEpsilon) ||
1595 (fabs(segment->y1-segment->y2) >= MagickEpsilon))
1603 x=CastDoubleToLong(ceil(segment->x1-0.5));
1604 y=CastDoubleToLong(ceil(segment->y1-0.5));
1605 (void) GetOneCacheViewVirtualPixel(u_view,x,y,&u,exception);
1606 x=CastDoubleToLong(ceil(segment->x2-0.5));
1607 y=CastDoubleToLong(ceil(segment->y2-0.5));
1608 (void) GetOneCacheViewVirtualPixel(v_view,x,y,&v,exception);
1609 q=QueueCacheViewAuthenticPixels(image_view,x_mid,y_mid,1,1,exception);
1612 SetPixelRed(q,PlasmaPixel(random_info,((MagickRealType) u.red+
1613 (MagickRealType) v.red)/2.0,plasma));
1614 SetPixelGreen(q,PlasmaPixel(random_info,((MagickRealType) u.green+
1615 (MagickRealType) v.green)/2.0,plasma));
1616 SetPixelBlue(q,PlasmaPixel(random_info,((MagickRealType) u.blue+
1617 (MagickRealType) v.blue)/2.0,plasma));
1618 status=SyncCacheViewAuthenticPixels(image_view,exception);
1620 if ((fabs(segment->x2-segment->x1) < 3.0) &&
1621 (fabs(segment->y2-segment->y1) < 3.0))
1622 return(status == 0 ? MagickFalse : MagickTrue);
1623 return(MagickFalse);
1626MagickExport MagickBooleanType PlasmaImage(
Image *image,
1627 const SegmentInfo *segment,
size_t attenuate,
size_t depth)
1640 assert(image != (
Image *) NULL);
1641 assert(image->signature == MagickCoreSignature);
1642 if (IsEventLogging() != MagickFalse)
1643 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
1644 if (SetImageStorageClass(image,DirectClass) == MagickFalse)
1645 return(MagickFalse);
1646 image_view=AcquireAuthenticCacheView(image,&image->exception);
1647 u_view=AcquireVirtualCacheView(image,&image->exception);
1648 v_view=AcquireVirtualCacheView(image,&image->exception);
1649 random_info=AcquireRandomInfo();
1650 status=PlasmaImageProxy(image,image_view,u_view,v_view,random_info,segment,
1652 random_info=DestroyRandomInfo(random_info);
1653 v_view=DestroyCacheView(v_view);
1654 u_view=DestroyCacheView(u_view);
1655 image_view=DestroyCacheView(image_view);
1712 assert(image != (
Image *) NULL);
1713 assert(image->signature == MagickCoreSignature);
1715 assert(exception->signature == MagickCoreSignature);
1716 if (IsEventLogging() != MagickFalse)
1717 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1718 quantum=(ssize_t) MagickMax(MagickMax((
double) image->columns,(
double)
1719 image->rows)/25.0,10.0);
1720 height=image->rows+2*quantum;
1721 caption_image=(
Image *) NULL;
1722 value=GetImageProperty(image,
"Caption");
1723 if (value != (
const char *) NULL)
1731 caption_image=CloneImage(image,image->columns,1,MagickTrue,exception);
1732 if (caption_image == (
Image *) NULL)
1733 return((
Image *) NULL);
1734 caption=InterpretImageProperties((
ImageInfo *) NULL,(
Image *) image,
1736 if (caption != (
char *) NULL)
1739 geometry[MaxTextExtent];
1753 annotate_info=CloneDrawInfo((
const ImageInfo *) NULL,draw_info);
1754 (void) CloneString(&annotate_info->text,caption);
1755 count=FormatMagickCaption(caption_image,annotate_info,MagickTrue,
1757 status=SetImageExtent(caption_image,image->columns,(
size_t)
1758 ((count+1)*(metrics.ascent-metrics.descent)+0.5));
1759 if (status == MagickFalse)
1760 caption_image=DestroyImage(caption_image);
1763 caption_image->background_color=image->border_color;
1764 (void) SetImageBackgroundColor(caption_image);
1765 (void) CloneString(&annotate_info->text,caption);
1766 (void) FormatLocaleString(geometry,MaxTextExtent,
"+0+%.20g",
1768 if (annotate_info->gravity == UndefinedGravity)
1769 (void) CloneString(&annotate_info->geometry,AcquireString(
1771 (void) AnnotateImage(caption_image,annotate_info);
1772 height+=caption_image->rows;
1774 annotate_info=DestroyDrawInfo(annotate_info);
1775 caption=DestroyString(caption);
1778 picture_image=CloneImage(image,image->columns+2*quantum,height,MagickTrue,
1780 if (picture_image == (
Image *) NULL)
1782 if (caption_image != (
Image *) NULL)
1783 caption_image=DestroyImage(caption_image);
1784 return((
Image *) NULL);
1786 picture_image->background_color=image->border_color;
1787 (void) SetImageBackgroundColor(picture_image);
1788 (void) CompositeImage(picture_image,OverCompositeOp,image,quantum,quantum);
1789 if (caption_image != (
Image *) NULL)
1791 (void) CompositeImage(picture_image,OverCompositeOp,caption_image,
1792 quantum,(ssize_t) (image->rows+3*quantum/2));
1793 caption_image=DestroyImage(caption_image);
1795 (void) QueryColorDatabase(
"none",&picture_image->background_color,exception);
1796 (void) SetImageAlphaChannel(picture_image,OpaqueAlphaChannel);
1797 rotate_image=RotateImage(picture_image,90.0,exception);
1798 picture_image=DestroyImage(picture_image);
1799 if (rotate_image == (
Image *) NULL)
1800 return((
Image *) NULL);
1801 picture_image=rotate_image;
1802 bend_image=WaveImage(picture_image,0.01*picture_image->rows,2.0*
1803 picture_image->columns,exception);
1804 picture_image=DestroyImage(picture_image);
1805 if (bend_image == (
Image *) NULL)
1806 return((
Image *) NULL);
1807 InheritException(&bend_image->exception,exception);
1808 picture_image=bend_image;
1809 rotate_image=RotateImage(picture_image,-90.0,exception);
1810 picture_image=DestroyImage(picture_image);
1811 if (rotate_image == (
Image *) NULL)
1812 return((
Image *) NULL);
1813 picture_image=rotate_image;
1814 picture_image->background_color=image->background_color;
1815 polaroid_image=ShadowImage(picture_image,80.0,2.0,quantum/3,quantum/3,
1817 if (polaroid_image == (
Image *) NULL)
1819 picture_image=DestroyImage(picture_image);
1820 return(picture_image);
1822 flop_image=FlopImage(polaroid_image,exception);
1823 polaroid_image=DestroyImage(polaroid_image);
1824 if (flop_image == (
Image *) NULL)
1826 picture_image=DestroyImage(picture_image);
1827 return(picture_image);
1829 polaroid_image=flop_image;
1830 (void) CompositeImage(polaroid_image,OverCompositeOp,picture_image,
1831 (ssize_t) (-0.01*picture_image->columns/2.0),0L);
1832 picture_image=DestroyImage(picture_image);
1833 (void) QueryColorDatabase(
"none",&polaroid_image->background_color,exception);
1834 rotate_image=RotateImage(polaroid_image,angle,exception);
1835 polaroid_image=DestroyImage(polaroid_image);
1836 if (rotate_image == (
Image *) NULL)
1837 return((
Image *) NULL);
1838 polaroid_image=rotate_image;
1839 trim_image=TrimImage(polaroid_image,exception);
1840 polaroid_image=DestroyImage(polaroid_image);
1841 if (trim_image == (
Image *) NULL)
1842 return((
Image *) NULL);
1843 polaroid_image=trim_image;
1844 return(polaroid_image);
1877MagickExport
Image *SepiaToneImage(
const Image *image,
const double threshold,
1880#define SepiaToneImageTag "SepiaTone/Image"
1901 assert(image != (
const Image *) NULL);
1902 assert(image->signature == MagickCoreSignature);
1904 assert(exception->signature == MagickCoreSignature);
1905 if (IsEventLogging() != MagickFalse)
1906 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
1907 sepia_image=CloneImage(image,0,0,MagickTrue,exception);
1908 if (sepia_image == (
Image *) NULL)
1909 return((
Image *) NULL);
1910 if (SetImageStorageClass(sepia_image,DirectClass) == MagickFalse)
1912 InheritException(exception,&sepia_image->exception);
1913 sepia_image=DestroyImage(sepia_image);
1914 return((
Image *) NULL);
1921 image_view=AcquireVirtualCacheView(image,exception);
1922 sepia_view=AcquireAuthenticCacheView(sepia_image,exception);
1923#if defined(MAGICKCORE_OPENMP_SUPPORT)
1924 #pragma omp parallel for schedule(static) shared(progress,status) \
1925 magick_number_threads(image,sepia_image,image->rows,1)
1927 for (y=0; y < (ssize_t) image->rows; y++)
1938 if (status == MagickFalse)
1940 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
1941 q=QueueCacheViewAuthenticPixels(sepia_view,0,y,sepia_image->columns,1,
1948 for (x=0; x < (ssize_t) image->columns; x++)
1954 intensity=GetPixelIntensity(image,p);
1955 tone=intensity > threshold ? (double) QuantumRange : intensity+
1956 (double) QuantumRange-threshold;
1957 SetPixelRed(q,ClampToQuantum(tone));
1958 tone=intensity > (7.0*threshold/6.0) ? (
double) QuantumRange :
1959 intensity+(double) QuantumRange-7.0*threshold/6.0;
1960 SetPixelGreen(q,ClampToQuantum(tone));
1961 tone=intensity < (threshold/6.0) ? 0 : intensity-threshold/6.0;
1962 SetPixelBlue(q,ClampToQuantum(tone));
1964 if ((
double) GetPixelGreen(q) < tone)
1965 SetPixelGreen(q,ClampToQuantum(tone));
1966 if ((
double) GetPixelBlue(q) < tone)
1967 SetPixelBlue(q,ClampToQuantum(tone));
1968 SetPixelOpacity(q,GetPixelOpacity(p));
1972 if (SyncCacheViewAuthenticPixels(sepia_view,exception) == MagickFalse)
1974 if (image->progress_monitor != (MagickProgressMonitor) NULL)
1979#if defined(MAGICKCORE_OPENMP_SUPPORT)
1983 proceed=SetImageProgress(image,SepiaToneImageTag,progress,image->rows);
1984 if (proceed == MagickFalse)
1988 sepia_view=DestroyCacheView(sepia_view);
1989 image_view=DestroyCacheView(image_view);
1990 (void) NormalizeImage(sepia_image);
1991 (void) ContrastImage(sepia_image,MagickTrue);
1992 if (status == MagickFalse)
1993 sepia_image=DestroyImage(sepia_image);
1994 return(sepia_image);
2031MagickExport
Image *ShadowImage(
const Image *image,
const double opacity,
2032 const double sigma,
const ssize_t x_offset,
const ssize_t y_offset,
2035#define ShadowImageTag "Shadow/Image"
2057 assert(image != (
Image *) NULL);
2058 assert(image->signature == MagickCoreSignature);
2060 assert(exception->signature == MagickCoreSignature);
2061 if (IsEventLogging() != MagickFalse)
2062 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
2063 clone_image=CloneImage(image,0,0,MagickTrue,exception);
2064 if (clone_image == (
Image *) NULL)
2065 return((
Image *) NULL);
2066 if (IsGrayColorspace(image->colorspace) != MagickFalse)
2067 (void) SetImageColorspace(clone_image,sRGBColorspace);
2068 (void) SetImageVirtualPixelMethod(clone_image,EdgeVirtualPixelMethod);
2069 clone_image->compose=OverCompositeOp;
2070 border_info.width=CastDoubleToUnsigned(2.0*sigma+0.5);
2071 border_info.height=CastDoubleToUnsigned(2.0*sigma+0.5);
2074 (void) QueryColorDatabase(
"none",&clone_image->border_color,exception);
2075 border_image=BorderImage(clone_image,&border_info,exception);
2076 clone_image=DestroyImage(clone_image);
2077 if (border_image == (
Image *) NULL)
2078 return((
Image *) NULL);
2079 if (border_image->matte == MagickFalse)
2080 (void) SetImageAlphaChannel(border_image,OpaqueAlphaChannel);
2086 image_view=AcquireAuthenticCacheView(border_image,exception);
2087#if defined(MAGICKCORE_OPENMP_SUPPORT)
2088 #pragma omp parallel for schedule(static) shared(progress,status) \
2089 magick_number_threads(border_image,border_image,border_image->rows,1)
2091 for (y=0; y < (ssize_t) border_image->rows; y++)
2099 if (status == MagickFalse)
2101 q=GetCacheViewAuthenticPixels(image_view,0,y,border_image->columns,1,
2108 for (x=0; x < (ssize_t) border_image->columns; x++)
2110 SetPixelRed(q,border_image->background_color.red);
2111 SetPixelGreen(q,border_image->background_color.green);
2112 SetPixelBlue(q,border_image->background_color.blue);
2113 if (border_image->matte == MagickFalse)
2114 SetPixelOpacity(q,border_image->background_color.opacity);
2116 SetPixelOpacity(q,ClampToQuantum((MagickRealType) QuantumRange-
2117 GetPixelAlpha(q)*opacity/100.0));
2120 if (SyncCacheViewAuthenticPixels(image_view,exception) == MagickFalse)
2122 if (image->progress_monitor != (MagickProgressMonitor) NULL)
2127#if defined(MAGICKCORE_OPENMP_SUPPORT)
2131 proceed=SetImageProgress(image,ShadowImageTag,progress,
2132 border_image->rows);
2133 if (proceed == MagickFalse)
2137 image_view=DestroyCacheView(image_view);
2138 shadow_image=BlurImageChannel(border_image,AlphaChannel,0.0,sigma,exception);
2139 border_image=DestroyImage(border_image);
2140 if (shadow_image == (
Image *) NULL)
2141 return((
Image *) NULL);
2142 if (shadow_image->page.width == 0)
2143 shadow_image->page.width=shadow_image->columns;
2144 if (shadow_image->page.height == 0)
2145 shadow_image->page.height=shadow_image->rows;
2146 shadow_image->page.width+=x_offset-(ssize_t) border_info.width;
2147 shadow_image->page.height+=y_offset-(ssize_t) border_info.height;
2148 shadow_image->page.x+=x_offset-(ssize_t) border_info.width;
2149 shadow_image->page.y+=y_offset-(ssize_t) border_info.height;
2150 return(shadow_image);
2189MagickExport
Image *SketchImage(
const Image *image,
const double radius,
2190 const double sigma,
const double angle,
ExceptionInfo *exception)
2209 **magick_restrict random_info;
2214#if defined(MAGICKCORE_OPENMP_SUPPORT)
2222 random_image=CloneImage(image,image->columns << 1,image->rows << 1,
2223 MagickTrue,exception);
2224 if (random_image == (
Image *) NULL)
2225 return((
Image *) NULL);
2227 GetMagickPixelPacket(random_image,&zero);
2228 random_info=AcquireRandomInfoTLS();
2229 random_view=AcquireAuthenticCacheView(random_image,exception);
2230#if defined(MAGICKCORE_OPENMP_SUPPORT)
2231 key=GetRandomSecretKey(random_info[0]);
2232#pragma omp parallel for schedule(static) shared(status) \
2233magick_number_threads(random_image,random_image,random_image->rows,key == ~0UL)
2235 for (y=0; y < (ssize_t) random_image->rows; y++)
2238 id = GetOpenMPThreadId();
2244 *magick_restrict indexes;
2252 if (status == MagickFalse)
2254 q=QueueCacheViewAuthenticPixels(random_view,0,y,random_image->columns,1,
2261 indexes=GetCacheViewAuthenticIndexQueue(random_view);
2263 for (x=0; x < (ssize_t) random_image->columns; x++)
2265 pixel.red=(MagickRealType) QuantumRange*
2266 GetPseudoRandomValue(random_info[
id]);
2267 pixel.green=pixel.red;
2268 pixel.blue=pixel.red;
2269 if (image->colorspace == CMYKColorspace)
2270 pixel.index=pixel.red;
2271 SetPixelPacket(random_image,&pixel,q,indexes+x);
2274 if (SyncCacheViewAuthenticPixels(random_view,exception) == MagickFalse)
2277 random_info=DestroyRandomInfoTLS(random_info);
2278 if (status == MagickFalse)
2280 random_view=DestroyCacheView(random_view);
2281 random_image=DestroyImage(random_image);
2282 return(random_image);
2284 random_view=DestroyCacheView(random_view);
2286 blur_image=MotionBlurImage(random_image,radius,sigma,angle,exception);
2287 random_image=DestroyImage(random_image);
2288 if (blur_image == (
Image *) NULL)
2289 return((
Image *) NULL);
2290 dodge_image=EdgeImage(blur_image,radius,exception);
2291 blur_image=DestroyImage(blur_image);
2292 if (dodge_image == (
Image *) NULL)
2293 return((
Image *) NULL);
2294 status=ClampImage(dodge_image);
2295 if (status != MagickFalse)
2296 status=NormalizeImage(dodge_image);
2297 if (status != MagickFalse)
2298 status=NegateImage(dodge_image,MagickFalse);
2299 if (status != MagickFalse)
2300 status=TransformImage(&dodge_image,(
char *) NULL,
"50%");
2301 sketch_image=CloneImage(image,0,0,MagickTrue,exception);
2302 if (sketch_image == (
Image *) NULL)
2304 dodge_image=DestroyImage(dodge_image);
2305 return((
Image *) NULL);
2307 (void) CompositeImage(sketch_image,ColorDodgeCompositeOp,dodge_image,0,0);
2308 dodge_image=DestroyImage(dodge_image);
2309 blend_image=CloneImage(image,0,0,MagickTrue,exception);
2310 if (blend_image == (
Image *) NULL)
2312 sketch_image=DestroyImage(sketch_image);
2313 return((
Image *) NULL);
2315 (void) SetImageArtifact(blend_image,
"compose:args",
"20x80");
2316 (void) CompositeImage(sketch_image,BlendCompositeOp,blend_image,0,0);
2317 blend_image=DestroyImage(blend_image);
2318 return(sketch_image);
2355MagickExport MagickBooleanType SolarizeImage(
Image *image,
2356 const double threshold)
2361 status=SolarizeImageChannel(image,DefaultChannels,threshold,
2366MagickExport MagickBooleanType SolarizeImageChannel(
Image *image,
2367 const ChannelType channel,
const double threshold,
ExceptionInfo *exception)
2369#define SolarizeImageTag "Solarize/Image"
2383 assert(image != (
Image *) NULL);
2384 assert(image->signature == MagickCoreSignature);
2385 if (IsEventLogging() != MagickFalse)
2386 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
2387 if (IsGrayColorspace(image->colorspace) != MagickFalse)
2388 (void) SetImageColorspace(image,sRGBColorspace);
2389 if (image->storage_class == PseudoClass)
2397 for (i=0; i < (ssize_t) image->colors; i++)
2399 if ((channel & RedChannel) != 0)
2400 if ((
double) image->colormap[i].red > threshold)
2401 image->colormap[i].red=QuantumRange-image->colormap[i].red;
2402 if ((channel & GreenChannel) != 0)
2403 if ((
double) image->colormap[i].green > threshold)
2404 image->colormap[i].green=QuantumRange-image->colormap[i].green;
2405 if ((channel & BlueChannel) != 0)
2406 if ((
double) image->colormap[i].blue > threshold)
2407 image->colormap[i].blue=QuantumRange-image->colormap[i].blue;
2415 image_view=AcquireAuthenticCacheView(image,exception);
2416#if defined(MAGICKCORE_OPENMP_SUPPORT)
2417 #pragma omp parallel for schedule(static) shared(progress,status) \
2418 magick_number_threads(image,image,image->rows,1)
2420 for (y=0; y < (ssize_t) image->rows; y++)
2428 if (status == MagickFalse)
2430 q=GetCacheViewAuthenticPixels(image_view,0,y,image->columns,1,
2437 for (x=0; x < (ssize_t) image->columns; x++)
2439 if ((channel & RedChannel) != 0)
2440 if ((
double) GetPixelRed(q) > threshold)
2441 SetPixelRed(q,QuantumRange-GetPixelRed(q));
2442 if ((channel & GreenChannel) != 0)
2443 if ((
double) GetPixelGreen(q) > threshold)
2444 SetPixelGreen(q,QuantumRange-GetPixelGreen(q));
2445 if ((channel & BlueChannel) != 0)
2446 if ((
double) GetPixelBlue(q) > threshold)
2447 SetPixelBlue(q,QuantumRange-GetPixelBlue(q));
2450 if (SyncCacheViewAuthenticPixels(image_view,exception) == MagickFalse)
2452 if (image->progress_monitor != (MagickProgressMonitor) NULL)
2457#if defined(MAGICKCORE_OPENMP_SUPPORT)
2461 proceed=SetImageProgress(image,SolarizeImageTag,progress,image->rows);
2462 if (proceed == MagickFalse)
2466 image_view=DestroyCacheView(image_view);
2499MagickExport
Image *SteganoImage(
const Image *image,
const Image *watermark,
2502#define GetBit(alpha,i) ((((size_t) (alpha) >> (size_t) (i)) & 0x01) != 0)
2503#define SetBit(alpha,i,set) (alpha)=(Quantum) ((set) != 0 ? (size_t) (alpha) \
2504 | (one << (size_t) (i)) : (size_t) (alpha) & ~(one << (size_t) (i)))
2505#define SteganoImageTag "Stegano/Image"
2542 assert(image != (
const Image *) NULL);
2543 assert(image->signature == MagickCoreSignature);
2544 assert(watermark != (
const Image *) NULL);
2545 assert(watermark->signature == MagickCoreSignature);
2547 assert(exception->signature == MagickCoreSignature);
2548 if (IsEventLogging() != MagickFalse)
2549 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
2551 stegano_image=CloneImage(image,0,0,MagickTrue,exception);
2552 if (stegano_image == (
Image *) NULL)
2553 return((
Image *) NULL);
2554 if (SetImageStorageClass(stegano_image,DirectClass) == MagickFalse)
2556 InheritException(exception,&stegano_image->exception);
2557 stegano_image=DestroyImage(stegano_image);
2558 return((
Image *) NULL);
2560 stegano_image->depth=MAGICKCORE_QUANTUM_DEPTH;
2567 depth=stegano_image->depth;
2570 watermark_view=AcquireVirtualCacheView(watermark,exception);
2571 stegano_view=AcquireAuthenticCacheView(stegano_image,exception);
2572 for (i=(ssize_t) depth-1; (i >= 0) && (j < (ssize_t) depth); i--)
2574 for (y=0; (y < (ssize_t) watermark->rows) && (j < (ssize_t) depth); y++)
2576 for (x=0; (x < (ssize_t) watermark->columns) && (j < (ssize_t) depth); x++)
2578 (void) GetOneCacheViewVirtualPixel(watermark_view,x,y,&pixel,exception);
2579 if ((k/(ssize_t) stegano_image->columns) >= (ssize_t) stegano_image->rows)
2581 q=GetCacheViewAuthenticPixels(stegano_view,k % (ssize_t)
2582 stegano_image->columns,k/(ssize_t) stegano_image->columns,1,1,
2590 SetBit(GetPixelRed(q),j,GetBit(ClampToQuantum(GetPixelIntensity(
2596 SetBit(GetPixelGreen(q),j,GetBit(ClampToQuantum(GetPixelIntensity(
2602 SetBit(GetPixelBlue(q),j,GetBit(ClampToQuantum(GetPixelIntensity(
2607 if (SyncCacheViewAuthenticPixels(stegano_view,exception) == MagickFalse)
2613 if (k == (ssize_t) (stegano_image->columns*stegano_image->columns))
2615 if (k == image->offset)
2619 if (image->progress_monitor != (MagickProgressMonitor) NULL)
2624 proceed=SetImageProgress(image,SteganoImageTag,(MagickOffsetType)
2626 if (proceed == MagickFalse)
2630 stegano_view=DestroyCacheView(stegano_view);
2631 watermark_view=DestroyCacheView(watermark_view);
2632 if (stegano_image->storage_class == PseudoClass)
2633 (void) SyncImage(stegano_image);
2634 if (status == MagickFalse)
2635 stegano_image=DestroyImage(stegano_image);
2636 return(stegano_image);
2678MagickExport
Image *StereoImage(
const Image *left_image,
2681 return(StereoAnaglyphImage(left_image,right_image,0,0,exception));
2684MagickExport
Image *StereoAnaglyphImage(
const Image *left_image,
2685 const Image *right_image,
const ssize_t x_offset,
const ssize_t y_offset,
2688#define StereoImageTag "Stereo/Image"
2702 assert(left_image != (
const Image *) NULL);
2703 assert(left_image->signature == MagickCoreSignature);
2704 assert(right_image != (
const Image *) NULL);
2705 assert(right_image->signature == MagickCoreSignature);
2707 assert(exception->signature == MagickCoreSignature);
2708 if (IsEventLogging() != MagickFalse)
2709 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",
2710 left_image->filename);
2712 if ((left_image->columns != right_image->columns) ||
2713 (left_image->rows != right_image->rows))
2714 ThrowImageException(ImageError,
"LeftAndRightImageSizesDiffer");
2718 stereo_image=CloneImage(left_image,left_image->columns,left_image->rows,
2719 MagickTrue,exception);
2720 if (stereo_image == (
Image *) NULL)
2721 return((
Image *) NULL);
2722 if (SetImageStorageClass(stereo_image,DirectClass) == MagickFalse)
2724 InheritException(exception,&stereo_image->exception);
2725 stereo_image=DestroyImage(stereo_image);
2726 return((
Image *) NULL);
2728 (void) SetImageColorspace(stereo_image,sRGBColorspace);
2733 for (y=0; y < (ssize_t) stereo_image->rows; y++)
2745 p=GetVirtualPixels(left_image,-x_offset,y-y_offset,image->columns,1,
2747 q=GetVirtualPixels(right_image,0,y,right_image->columns,1,exception);
2748 r=QueueAuthenticPixels(stereo_image,0,y,stereo_image->columns,1,exception);
2752 for (x=0; x < (ssize_t) stereo_image->columns; x++)
2754 SetPixelRed(r,GetPixelRed(p));
2755 SetPixelGreen(r,GetPixelGreen(q));
2756 SetPixelBlue(r,GetPixelBlue(q));
2757 SetPixelOpacity(r,(GetPixelOpacity(p)+q->opacity)/2);
2762 if (SyncAuthenticPixels(stereo_image,exception) == MagickFalse)
2764 if (image->progress_monitor != (MagickProgressMonitor) NULL)
2769 proceed=SetImageProgress(image,StereoImageTag,(MagickOffsetType) y,
2770 stereo_image->rows);
2771 if (proceed == MagickFalse)
2775 if (status == MagickFalse)
2776 stereo_image=DestroyImage(stereo_image);
2777 return(stereo_image);
2809MagickExport
Image *SwirlImage(
const Image *image,
double degrees,
2812#define SwirlImageTag "Swirl/Image"
2843 assert(image != (
const Image *) NULL);
2844 assert(image->signature == MagickCoreSignature);
2846 assert(exception->signature == MagickCoreSignature);
2847 if (IsEventLogging() != MagickFalse)
2848 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
2849 swirl_image=CloneImage(image,0,0,MagickTrue,exception);
2850 if (swirl_image == (
Image *) NULL)
2851 return((
Image *) NULL);
2852 if (SetImageStorageClass(swirl_image,DirectClass) == MagickFalse)
2854 InheritException(exception,&swirl_image->exception);
2855 swirl_image=DestroyImage(swirl_image);
2856 return((
Image *) NULL);
2858 if (swirl_image->background_color.opacity != OpaqueOpacity)
2859 swirl_image->matte=MagickTrue;
2863 center.x=(double) image->columns/2.0;
2864 center.y=(double) image->rows/2.0;
2865 radius=MagickMax(center.x,center.y);
2868 if (image->columns > image->rows)
2869 scale.y=(double) image->columns/(
double) image->rows;
2871 if (image->columns < image->rows)
2872 scale.x=(double) image->rows/(
double) image->columns;
2873 degrees=(double) DegreesToRadians(degrees);
2879 GetMagickPixelPacket(swirl_image,&zero);
2880 image_view=AcquireVirtualCacheView(image,exception);
2881 swirl_view=AcquireAuthenticCacheView(swirl_image,exception);
2882#if defined(MAGICKCORE_OPENMP_SUPPORT)
2883 #pragma omp parallel for schedule(static) shared(progress,status) \
2884 magick_number_threads(image,swirl_image,image->rows,1)
2886 for (y=0; y < (ssize_t) image->rows; y++)
2898 *magick_restrict swirl_indexes;
2906 if (status == MagickFalse)
2908 q=GetCacheViewAuthenticPixels(swirl_view,0,y,swirl_image->columns,1,
2915 swirl_indexes=GetCacheViewAuthenticIndexQueue(swirl_view);
2916 delta.y=scale.y*(double) (y-center.y);
2918 for (x=0; x < (ssize_t) image->columns; x++)
2923 delta.x=scale.x*(double) (x-center.x);
2924 distance=delta.x*delta.x+delta.y*delta.y;
2925 if (distance < (radius*radius))
2935 factor=1.0-sqrt(distance)/radius;
2936 sine=sin((
double) (degrees*factor*factor));
2937 cosine=cos((
double) (degrees*factor*factor));
2938 status=InterpolateMagickPixelPacket(image,image_view,
2939 UndefinedInterpolatePixel,(
double) ((cosine*delta.x-sine*delta.y)/
2940 scale.x+center.x),(
double) ((sine*delta.x+cosine*delta.y)/scale.y+
2941 center.y),&pixel,exception);
2942 if (status == MagickFalse)
2944 SetPixelPacket(swirl_image,&pixel,q,swirl_indexes+x);
2948 if (SyncCacheViewAuthenticPixels(swirl_view,exception) == MagickFalse)
2950 if (image->progress_monitor != (MagickProgressMonitor) NULL)
2955#if defined(MAGICKCORE_OPENMP_SUPPORT)
2959 proceed=SetImageProgress(image,SwirlImageTag,progress,image->rows);
2960 if (proceed == MagickFalse)
2964 swirl_view=DestroyCacheView(swirl_view);
2965 image_view=DestroyCacheView(image_view);
2966 if (status == MagickFalse)
2967 swirl_image=DestroyImage(swirl_image);
2968 return(swirl_image);
3002MagickExport
Image *TintImage(
const Image *image,
const char *opacity,
3005#define TintImageTag "Tint/Image"
3036 assert(image != (
const Image *) NULL);
3037 assert(image->signature == MagickCoreSignature);
3039 assert(exception->signature == MagickCoreSignature);
3040 if (IsEventLogging() != MagickFalse)
3041 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
3042 tint_image=CloneImage(image,0,0,MagickTrue,exception);
3043 if (tint_image == (
Image *) NULL)
3044 return((
Image *) NULL);
3045 if (SetImageStorageClass(tint_image,DirectClass) == MagickFalse)
3047 InheritException(exception,&tint_image->exception);
3048 tint_image=DestroyImage(tint_image);
3049 return((
Image *) NULL);
3051 if ((IsGrayColorspace(image->colorspace) != MagickFalse) &&
3052 (IsPixelGray(&tint) == MagickFalse))
3053 (void) SetImageColorspace(tint_image,sRGBColorspace);
3054 if (opacity == (
const char *) NULL)
3059 flags=ParseGeometry(opacity,&geometry_info);
3060 pixel.red=geometry_info.rho;
3061 pixel.green=geometry_info.rho;
3062 pixel.blue=geometry_info.rho;
3063 pixel.opacity=(MagickRealType) OpaqueOpacity;
3064 if ((flags & SigmaValue) != 0)
3065 pixel.green=geometry_info.sigma;
3066 if ((flags & XiValue) != 0)
3067 pixel.blue=geometry_info.xi;
3068 if ((flags & PsiValue) != 0)
3069 pixel.opacity=geometry_info.psi;
3070 color_vector.red=(MagickRealType) pixel.red*(MagickRealType) tint.red/
3071 100.0-(MagickRealType) PixelPacketIntensity(&tint);
3072 color_vector.green=(MagickRealType) pixel.green*(MagickRealType) tint.green/
3073 100.0-(MagickRealType) PixelPacketIntensity(&tint);
3074 color_vector.blue=(MagickRealType) pixel.blue*(MagickRealType) tint.blue/
3075 100.0-(MagickRealType) PixelPacketIntensity(&tint);
3081 image_view=AcquireVirtualCacheView(image,exception);
3082 tint_view=AcquireAuthenticCacheView(tint_image,exception);
3083#if defined(MAGICKCORE_OPENMP_SUPPORT)
3084 #pragma omp parallel for schedule(static) shared(progress,status) \
3085 magick_number_threads(image,tint_image,image->rows,1)
3087 for (y=0; y < (ssize_t) image->rows; y++)
3098 if (status == MagickFalse)
3100 p=GetCacheViewVirtualPixels(image_view,0,y,image->columns,1,exception);
3101 q=QueueCacheViewAuthenticPixels(tint_view,0,y,tint_image->columns,1,
3108 for (x=0; x < (ssize_t) image->columns; x++)
3116 weight=QuantumScale*(MagickRealType) GetPixelRed(p)-0.5;
3117 pixel.red=(MagickRealType) GetPixelRed(p)+color_vector.red*(1.0-(4.0*
3119 SetPixelRed(q,ClampToQuantum(pixel.red));
3120 weight=QuantumScale*(MagickRealType) GetPixelGreen(p)-0.5;
3121 pixel.green=(MagickRealType) GetPixelGreen(p)+color_vector.green*(1.0-
3122 (4.0*(weight*weight)));
3123 SetPixelGreen(q,ClampToQuantum(pixel.green));
3124 weight=QuantumScale*(MagickRealType) GetPixelBlue(p)-0.5;
3125 pixel.blue=(MagickRealType) GetPixelBlue(p)+color_vector.blue*(1.0-(4.0*
3127 SetPixelBlue(q,ClampToQuantum(pixel.blue));
3128 SetPixelOpacity(q,GetPixelOpacity(p));
3132 if (SyncCacheViewAuthenticPixels(tint_view,exception) == MagickFalse)
3134 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3139#if defined(MAGICKCORE_OPENMP_SUPPORT)
3143 proceed=SetImageProgress(image,TintImageTag,progress,image->rows);
3144 if (proceed == MagickFalse)
3148 tint_view=DestroyCacheView(tint_view);
3149 image_view=DestroyCacheView(image_view);
3150 if (status == MagickFalse)
3151 tint_image=DestroyImage(tint_image);
3187MagickExport
Image *VignetteImage(
const Image *image,
const double radius,
3188 const double sigma,
const ssize_t x,
const ssize_t y,
ExceptionInfo *exception)
3191 ellipse[MaxTextExtent];
3202 assert(image != (
Image *) NULL);
3203 assert(image->signature == MagickCoreSignature);
3205 assert(exception->signature == MagickCoreSignature);
3206 if (IsEventLogging() != MagickFalse)
3207 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
3208 canvas_image=CloneImage(image,0,0,MagickTrue,exception);
3209 if (canvas_image == (
Image *) NULL)
3210 return((
Image *) NULL);
3211 if (SetImageStorageClass(canvas_image,DirectClass) == MagickFalse)
3213 InheritException(exception,&canvas_image->exception);
3214 canvas_image=DestroyImage(canvas_image);
3215 return((
Image *) NULL);
3217 canvas_image->matte=MagickTrue;
3218 oval_image=CloneImage(canvas_image,canvas_image->columns,canvas_image->rows,
3219 MagickTrue,exception);
3220 if (oval_image == (
Image *) NULL)
3222 canvas_image=DestroyImage(canvas_image);
3223 return((
Image *) NULL);
3225 (void) QueryColorDatabase(
"#000000",&oval_image->background_color,exception);
3226 (void) SetImageBackgroundColor(oval_image);
3228 (void) QueryColorDatabase(
"#ffffff",&draw_info->fill,exception);
3229 (void) QueryColorDatabase(
"#ffffff",&draw_info->stroke,exception);
3230 (void) FormatLocaleString(ellipse,MaxTextExtent,
3231 "ellipse %g,%g,%g,%g,0.0,360.0",image->columns/2.0,
3232 image->rows/2.0,image->columns/2.0-x,image->rows/2.0-y);
3233 draw_info->primitive=AcquireString(ellipse);
3234 (void) DrawImage(oval_image,draw_info);
3235 draw_info=DestroyDrawInfo(draw_info);
3236 blur_image=BlurImage(oval_image,radius,sigma,exception);
3237 oval_image=DestroyImage(oval_image);
3238 if (blur_image == (
Image *) NULL)
3240 canvas_image=DestroyImage(canvas_image);
3241 return((
Image *) NULL);
3243 blur_image->matte=MagickFalse;
3244 (void) CompositeImage(canvas_image,CopyOpacityCompositeOp,blur_image,0,0);
3245 blur_image=DestroyImage(blur_image);
3246 vignette_image=MergeImageLayers(canvas_image,FlattenLayer,exception);
3247 canvas_image=DestroyImage(canvas_image);
3248 if (vignette_image != (
Image *) NULL)
3249 (void) TransformImageColorspace(vignette_image,image->colorspace);
3250 return(vignette_image);
3283MagickExport
Image *WaveImage(
const Image *image,
const double amplitude,
3286#define WaveImageTag "Wave/Image"
3316 assert(image != (
Image *) NULL);
3317 assert(image->signature == MagickCoreSignature);
3319 assert(exception->signature == MagickCoreSignature);
3320 if (IsEventLogging() != MagickFalse)
3321 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
3322 wave_image=CloneImage(image,image->columns,(
size_t) (image->rows+2.0*
3323 fabs(amplitude)),MagickTrue,exception);
3324 if (wave_image == (
Image *) NULL)
3325 return((
Image *) NULL);
3326 if (SetImageStorageClass(wave_image,DirectClass) == MagickFalse)
3328 InheritException(exception,&wave_image->exception);
3329 wave_image=DestroyImage(wave_image);
3330 return((
Image *) NULL);
3332 if (wave_image->background_color.opacity != OpaqueOpacity)
3333 wave_image->matte=MagickTrue;
3337 sine_map=(
float *) AcquireQuantumMemory((
size_t) wave_image->columns,
3339 if (sine_map == (
float *) NULL)
3341 wave_image=DestroyImage(wave_image);
3342 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
3344 for (i=0; i < (ssize_t) wave_image->columns; i++)
3345 sine_map[i]=(
float) (fabs(amplitude)+amplitude*sin((
double)
3346 ((2.0*MagickPI*i)*PerceptibleReciprocal(wave_length))));
3352 GetMagickPixelPacket(wave_image,&zero);
3353 image_view=AcquireVirtualCacheView(image,exception);
3354 wave_view=AcquireAuthenticCacheView(wave_image,exception);
3355 (void) SetCacheViewVirtualPixelMethod(image_view,
3356 BackgroundVirtualPixelMethod);
3357#if defined(MAGICKCORE_OPENMP_SUPPORT)
3358 #pragma omp parallel for schedule(static) shared(progress,status) \
3359 magick_number_threads(image,wave_image,wave_image->rows,1)
3361 for (y=0; y < (ssize_t) wave_image->rows; y++)
3367 *magick_restrict indexes;
3375 if (status == MagickFalse)
3377 q=QueueCacheViewAuthenticPixels(wave_view,0,y,wave_image->columns,1,
3384 indexes=GetCacheViewAuthenticIndexQueue(wave_view);
3386 for (x=0; x < (ssize_t) wave_image->columns; x++)
3388 status=InterpolateMagickPixelPacket(image,image_view,
3389 UndefinedInterpolatePixel,(
double) x,(
double) (y-sine_map[x]),&pixel,
3391 if (status == MagickFalse)
3393 SetPixelPacket(wave_image,&pixel,q,indexes+x);
3396 if (SyncCacheViewAuthenticPixels(wave_view,exception) == MagickFalse)
3398 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3403#if defined(MAGICKCORE_OPENMP_SUPPORT)
3407 proceed=SetImageProgress(image,WaveImageTag,progress,image->rows);
3408 if (proceed == MagickFalse)
3412 wave_view=DestroyCacheView(wave_view);
3413 image_view=DestroyCacheView(image_view);
3414 sine_map=(
float *) RelinquishMagickMemory(sine_map);
3415 if (status == MagickFalse)
3416 wave_image=DestroyImage(wave_image);
3455static inline void HatTransform(
const float *magick_restrict pixels,
3456 const size_t stride,
const size_t extent,
const size_t scale,
float *kernel)
3467 q=pixels+scale*stride,
3468 r=pixels+scale*stride;
3469 for (i=0; i < (ssize_t) scale; i++)
3471 kernel[i]=0.25f*(*p+(*p)+(*q)+(*r));
3472 p+=(ptrdiff_t) stride;
3474 r+=(ptrdiff_t) stride;
3476 for ( ; i < (ssize_t) (extent-scale); i++)
3478 kernel[i]=0.25f*(2.0f*(*p)+*(p-scale*stride)+*(p+scale*stride));
3479 p+=(ptrdiff_t) stride;
3482 r=pixels+stride*(extent-2);
3483 for ( ; i < (ssize_t) extent; i++)
3485 kernel[i]=0.25f*(*p+(*p)+(*q)+(*r));
3486 p+=(ptrdiff_t) stride;
3487 q+=(ptrdiff_t) stride;
3492MagickExport
Image *WaveletDenoiseImage(
const Image *image,
3493 const double threshold,
const double softness,
ExceptionInfo *exception)
3523 0.8002, 0.2735, 0.1202, 0.0585, 0.0291, 0.0152, 0.0080, 0.0044 };
3528 assert(image != (
const Image *) NULL);
3529 assert(image->signature == MagickCoreSignature);
3531 assert(exception->signature == MagickCoreSignature);
3532 if (IsEventLogging() != MagickFalse)
3533 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
3534 noise_image=(
Image *) NULL;
3535#if defined(MAGICKCORE_OPENCL_SUPPORT)
3536 noise_image=AccelerateWaveletDenoiseImage(image,threshold,exception);
3537 if (noise_image != (
Image *) NULL)
3538 return(noise_image);
3540 noise_image=CloneImage(image,0,0,MagickTrue,exception);
3541 if (noise_image == (
Image *) NULL)
3542 return((
Image *) NULL);
3543 if (SetImageStorageClass(noise_image,DirectClass) == MagickFalse)
3545 noise_image=DestroyImage(noise_image);
3546 return((
Image *) NULL);
3548 if (AcquireMagickResource(WidthResource,3*image->columns) == MagickFalse)
3549 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
3550 pixels_info=AcquireVirtualMemory(3*image->columns,image->rows*
3552 kernel=(
float *) AcquireQuantumMemory(MagickMax(image->rows,image->columns)+1,
3553 GetOpenMPMaximumThreads()*
sizeof(*kernel));
3554 if ((pixels_info == (
MemoryInfo *) NULL) || (kernel == (
float *) NULL))
3556 if (kernel != (
float *) NULL)
3557 kernel=(
float *) RelinquishMagickMemory(kernel);
3559 pixels_info=RelinquishVirtualMemory(pixels_info);
3560 ThrowImageException(ResourceLimitError,
"MemoryAllocationFailed");
3562 pixels=(
float *) GetVirtualMemoryBlob(pixels_info);
3564 number_pixels=image->columns*image->rows;
3565 max_channels=(size_t) (image->colorspace == CMYKColorspace ? 4 : 3);
3566 image_view=AcquireAuthenticCacheView(image,exception);
3567 noise_view=AcquireAuthenticCacheView(noise_image,exception);
3568 for (channel=0; channel < (ssize_t) max_channels; channel++)
3581 if (status == MagickFalse)
3587 for (y=0; y < (ssize_t) image->rows; y++)
3590 *magick_restrict indexes;
3598 p=GetCacheViewAuthenticPixels(image_view,0,y,image->columns,1,exception);
3604 indexes=GetCacheViewVirtualIndexQueue(image_view);
3605 for (x=0; x < (ssize_t) image->columns; x++)
3609 case 0: pixels[i]=(float) GetPixelRed(p);
break;
3610 case 1: pixels[i]=(float) GetPixelGreen(p);
break;
3611 case 2: pixels[i]=(float) GetPixelBlue(p);
break;
3612 case 3: pixels[i]=(float) indexes[x];
break;
3625 for (level=0; level < 5; level++)
3634 low_pass=(size_t) (number_pixels*((level & 0x01)+1));
3635#if defined(MAGICKCORE_OPENMP_SUPPORT)
3636 #pragma omp parallel for schedule(static,1) \
3637 magick_number_threads(image,image,image->rows,1)
3639 for (y=0; y < (ssize_t) image->rows; y++)
3642 id = GetOpenMPThreadId();
3651 p=kernel+
id*image->columns;
3652 q=pixels+y*image->columns;
3653 HatTransform(q+high_pass,1,image->columns,(
size_t) (1UL << level),p);
3654 q+=(ptrdiff_t) low_pass;
3655 for (x=0; x < (ssize_t) image->columns; x++)
3658#if defined(MAGICKCORE_OPENMP_SUPPORT)
3659 #pragma omp parallel for schedule(static,1) \
3660 magick_number_threads(image,image,image->columns,1)
3662 for (x=0; x < (ssize_t) image->columns; x++)
3665 id = GetOpenMPThreadId();
3674 p=kernel+
id*image->rows;
3675 q=pixels+x+low_pass;
3676 HatTransform(q,image->columns,image->rows,(
size_t) (1UL << level),p);
3677 for (y=0; y < (ssize_t) image->rows; y++)
3680 q+=(ptrdiff_t) image->columns;
3687 magnitude=threshold*noise_levels[level];
3688 for (i=0; i < (ssize_t) number_pixels; ++i)
3690 pixels[high_pass+i]-=pixels[low_pass+i];
3691 if ((MagickRealType) pixels[high_pass+i] < -magnitude)
3692 pixels[high_pass+i]+=(float) (magnitude-softness*magnitude);
3694 if ((MagickRealType) pixels[high_pass+i] > magnitude)
3695 pixels[high_pass+i]-=(float) (magnitude-softness*magnitude);
3697 pixels[high_pass+i]*=(float) softness;
3699 pixels[i]+=pixels[high_pass+i];
3707 for (y=0; y < (ssize_t) image->rows; y++)
3713 *magick_restrict noise_indexes;
3721 q=GetCacheViewAuthenticPixels(noise_view,0,y,noise_image->columns,1,
3728 noise_indexes=GetCacheViewAuthenticIndexQueue(noise_view);
3729 for (x=0; x < (ssize_t) image->columns; x++)
3734 pixel=pixels[i]+pixels[low_pass+i];
3737 case 0: SetPixelRed(q,ClampToQuantum(pixel));
break;
3738 case 1: SetPixelGreen(q,ClampToQuantum(pixel));
break;
3739 case 2: SetPixelBlue(q,ClampToQuantum(pixel));
break;
3740 case 3: SetPixelIndex(noise_indexes+x,ClampToQuantum(pixel));
break;
3746 sync=SyncCacheViewAuthenticPixels(noise_view,exception);
3747 if (sync == MagickFalse)
3750 if (image->progress_monitor != (MagickProgressMonitor) NULL)
3755 proceed=SetImageProgress(image,AddNoiseImageTag,(MagickOffsetType)
3756 channel,max_channels);
3757 if (proceed == MagickFalse)
3761 noise_view=DestroyCacheView(noise_view);
3762 image_view=DestroyCacheView(image_view);
3763 kernel=(
float *) RelinquishMagickMemory(kernel);
3764 pixels_info=RelinquishVirtualMemory(pixels_info);
3765 return(noise_image);