Repository: Intervention/image
Branch: develop
Commit: c073d1c93a11
Files: 551
Total size: 933.6 KB
Directory structure:
gitextract_zeq9zksh/
├── .editorconfig
├── .gitattributes
├── .github/
│ ├── FUNDING.yml
│ ├── ISSUE_TEMPLATE/
│ │ ├── bug_report.md
│ │ └── feature_request.md
│ └── workflows/
│ └── run-tests.yml
├── .gitignore
├── .well-known/
│ └── funding-manifest-urls
├── CONTRIBUTING.md
├── Dockerfile
├── LICENSE
├── composer.json
├── docker-compose.yml
├── phpcs.xml.dist
├── phpstan.dist.neon
├── phpunit.dist.xml
├── readme.md
├── src/
│ ├── Analyzers/
│ │ ├── ColorspaceAnalyzer.php
│ │ ├── HeightAnalyzer.php
│ │ ├── PixelColorAnalyzer.php
│ │ ├── PixelColorsAnalyzer.php
│ │ ├── ProfileAnalyzer.php
│ │ ├── ResolutionAnalyzer.php
│ │ └── WidthAnalyzer.php
│ ├── Collection.php
│ ├── Colors/
│ │ ├── AbstractColor.php
│ │ ├── AbstractColorChannel.php
│ │ ├── Cmyk/
│ │ │ ├── Channels/
│ │ │ │ ├── Cyan.php
│ │ │ │ ├── Key.php
│ │ │ │ ├── Magenta.php
│ │ │ │ └── Yellow.php
│ │ │ ├── Color.php
│ │ │ ├── Colorspace.php
│ │ │ └── Decoders/
│ │ │ └── StringColorDecoder.php
│ │ ├── Hsl/
│ │ │ ├── Channels/
│ │ │ │ ├── Hue.php
│ │ │ │ ├── Luminance.php
│ │ │ │ └── Saturation.php
│ │ │ ├── Color.php
│ │ │ ├── Colorspace.php
│ │ │ └── Decoders/
│ │ │ └── StringColorDecoder.php
│ │ ├── Hsv/
│ │ │ ├── Channels/
│ │ │ │ ├── Hue.php
│ │ │ │ ├── Saturation.php
│ │ │ │ └── Value.php
│ │ │ ├── Color.php
│ │ │ ├── Colorspace.php
│ │ │ └── Decoders/
│ │ │ └── StringColorDecoder.php
│ │ ├── Profile.php
│ │ └── Rgb/
│ │ ├── Channels/
│ │ │ ├── Alpha.php
│ │ │ ├── Blue.php
│ │ │ ├── Green.php
│ │ │ └── Red.php
│ │ ├── Color.php
│ │ ├── Colorspace.php
│ │ └── Decoders/
│ │ ├── HexColorDecoder.php
│ │ ├── HtmlColornameDecoder.php
│ │ ├── StringColorDecoder.php
│ │ └── TransparentColorDecoder.php
│ ├── Config.php
│ ├── Decoders/
│ │ ├── Base64ImageDecoder.php
│ │ ├── BinaryImageDecoder.php
│ │ ├── ColorObjectDecoder.php
│ │ ├── DataUriImageDecoder.php
│ │ ├── EncodedImageObjectDecoder.php
│ │ ├── FilePathImageDecoder.php
│ │ ├── FilePointerImageDecoder.php
│ │ ├── ImageObjectDecoder.php
│ │ ├── NativeObjectDecoder.php
│ │ └── SplFileInfoImageDecoder.php
│ ├── Drivers/
│ │ ├── AbstractDecoder.php
│ │ ├── AbstractDriver.php
│ │ ├── AbstractEncoder.php
│ │ ├── AbstractFontProcessor.php
│ │ ├── AbstractFrame.php
│ │ ├── Gd/
│ │ │ ├── Analyzers/
│ │ │ │ ├── ColorspaceAnalyzer.php
│ │ │ │ ├── HeightAnalyzer.php
│ │ │ │ ├── PixelColorAnalyzer.php
│ │ │ │ ├── PixelColorsAnalyzer.php
│ │ │ │ ├── ResolutionAnalyzer.php
│ │ │ │ └── WidthAnalyzer.php
│ │ │ ├── Cloner.php
│ │ │ ├── ColorProcessor.php
│ │ │ ├── Core.php
│ │ │ ├── Decoders/
│ │ │ │ ├── AbstractDecoder.php
│ │ │ │ ├── Base64ImageDecoder.php
│ │ │ │ ├── BinaryImageDecoder.php
│ │ │ │ ├── DataUriImageDecoder.php
│ │ │ │ ├── EncodedImageObjectDecoder.php
│ │ │ │ ├── FilePathImageDecoder.php
│ │ │ │ ├── FilePointerImageDecoder.php
│ │ │ │ ├── NativeObjectDecoder.php
│ │ │ │ └── SplFileInfoImageDecoder.php
│ │ │ ├── Driver.php
│ │ │ ├── Encoders/
│ │ │ │ ├── AvifEncoder.php
│ │ │ │ ├── BmpEncoder.php
│ │ │ │ ├── GifEncoder.php
│ │ │ │ ├── JpegEncoder.php
│ │ │ │ ├── PngEncoder.php
│ │ │ │ └── WebpEncoder.php
│ │ │ ├── FontProcessor.php
│ │ │ ├── Frame.php
│ │ │ └── Modifiers/
│ │ │ ├── AlignRotationModifier.php
│ │ │ ├── BlendTransparencyModifier.php
│ │ │ ├── BlurModifier.php
│ │ │ ├── BrightnessModifier.php
│ │ │ ├── ColorizeModifier.php
│ │ │ ├── ColorspaceModifier.php
│ │ │ ├── ContainModifier.php
│ │ │ ├── ContrastModifier.php
│ │ │ ├── CoverDownModifier.php
│ │ │ ├── CoverModifier.php
│ │ │ ├── CropModifier.php
│ │ │ ├── DrawBezierModifier.php
│ │ │ ├── DrawEllipseModifier.php
│ │ │ ├── DrawLineModifier.php
│ │ │ ├── DrawPixelModifier.php
│ │ │ ├── DrawPolygonModifier.php
│ │ │ ├── DrawRectangleModifier.php
│ │ │ ├── FillModifier.php
│ │ │ ├── FlipModifier.php
│ │ │ ├── FlopModifier.php
│ │ │ ├── GammaModifier.php
│ │ │ ├── GreyscaleModifier.php
│ │ │ ├── InvertModifier.php
│ │ │ ├── PadModifier.php
│ │ │ ├── PixelateModifier.php
│ │ │ ├── PlaceModifier.php
│ │ │ ├── ProfileModifier.php
│ │ │ ├── ProfileRemovalModifier.php
│ │ │ ├── QuantizeColorsModifier.php
│ │ │ ├── RemoveAnimationModifier.php
│ │ │ ├── ResizeCanvasModifier.php
│ │ │ ├── ResizeCanvasRelativeModifier.php
│ │ │ ├── ResizeDownModifier.php
│ │ │ ├── ResizeModifier.php
│ │ │ ├── ResolutionModifier.php
│ │ │ ├── RotateModifier.php
│ │ │ ├── ScaleDownModifier.php
│ │ │ ├── ScaleModifier.php
│ │ │ ├── SharpenModifier.php
│ │ │ ├── SliceAnimationModifier.php
│ │ │ ├── TextModifier.php
│ │ │ └── TrimModifier.php
│ │ ├── Imagick/
│ │ │ ├── Analyzers/
│ │ │ │ ├── ColorspaceAnalyzer.php
│ │ │ │ ├── HeightAnalyzer.php
│ │ │ │ ├── PixelColorAnalyzer.php
│ │ │ │ ├── PixelColorsAnalyzer.php
│ │ │ │ ├── ProfileAnalyzer.php
│ │ │ │ ├── ResolutionAnalyzer.php
│ │ │ │ └── WidthAnalyzer.php
│ │ │ ├── ColorProcessor.php
│ │ │ ├── Core.php
│ │ │ ├── Decoders/
│ │ │ │ ├── Base64ImageDecoder.php
│ │ │ │ ├── BinaryImageDecoder.php
│ │ │ │ ├── DataUriImageDecoder.php
│ │ │ │ ├── EncodedImageObjectDecoder.php
│ │ │ │ ├── FilePathImageDecoder.php
│ │ │ │ ├── FilePointerImageDecoder.php
│ │ │ │ ├── NativeObjectDecoder.php
│ │ │ │ └── SplFileInfoImageDecoder.php
│ │ │ ├── Driver.php
│ │ │ ├── Encoders/
│ │ │ │ ├── AvifEncoder.php
│ │ │ │ ├── BmpEncoder.php
│ │ │ │ ├── GifEncoder.php
│ │ │ │ ├── HeicEncoder.php
│ │ │ │ ├── Jpeg2000Encoder.php
│ │ │ │ ├── JpegEncoder.php
│ │ │ │ ├── PngEncoder.php
│ │ │ │ ├── TiffEncoder.php
│ │ │ │ └── WebpEncoder.php
│ │ │ ├── FontProcessor.php
│ │ │ ├── Frame.php
│ │ │ └── Modifiers/
│ │ │ ├── AlignRotationModifier.php
│ │ │ ├── BlendTransparencyModifier.php
│ │ │ ├── BlurModifier.php
│ │ │ ├── BrightnessModifier.php
│ │ │ ├── ColorizeModifier.php
│ │ │ ├── ColorspaceModifier.php
│ │ │ ├── ContainModifier.php
│ │ │ ├── ContrastModifier.php
│ │ │ ├── CoverDownModifier.php
│ │ │ ├── CoverModifier.php
│ │ │ ├── CropModifier.php
│ │ │ ├── DrawBezierModifier.php
│ │ │ ├── DrawEllipseModifier.php
│ │ │ ├── DrawLineModifier.php
│ │ │ ├── DrawPixelModifier.php
│ │ │ ├── DrawPolygonModifier.php
│ │ │ ├── DrawRectangleModifier.php
│ │ │ ├── FillModifier.php
│ │ │ ├── FlipModifier.php
│ │ │ ├── FlopModifier.php
│ │ │ ├── GammaModifier.php
│ │ │ ├── GreyscaleModifier.php
│ │ │ ├── InvertModifier.php
│ │ │ ├── PadModifier.php
│ │ │ ├── PixelateModifier.php
│ │ │ ├── PlaceModifier.php
│ │ │ ├── ProfileModifier.php
│ │ │ ├── ProfileRemovalModifier.php
│ │ │ ├── QuantizeColorsModifier.php
│ │ │ ├── RemoveAnimationModifier.php
│ │ │ ├── ResizeCanvasModifier.php
│ │ │ ├── ResizeCanvasRelativeModifier.php
│ │ │ ├── ResizeDownModifier.php
│ │ │ ├── ResizeModifier.php
│ │ │ ├── ResolutionModifier.php
│ │ │ ├── RotateModifier.php
│ │ │ ├── ScaleDownModifier.php
│ │ │ ├── ScaleModifier.php
│ │ │ ├── SharpenModifier.php
│ │ │ ├── SliceAnimationModifier.php
│ │ │ ├── StripMetaModifier.php
│ │ │ ├── TextModifier.php
│ │ │ └── TrimModifier.php
│ │ ├── Specializable.php
│ │ ├── SpecializableAnalyzer.php
│ │ ├── SpecializableDecoder.php
│ │ ├── SpecializableEncoder.php
│ │ └── SpecializableModifier.php
│ ├── EncodedImage.php
│ ├── Encoders/
│ │ ├── AutoEncoder.php
│ │ ├── AvifEncoder.php
│ │ ├── BmpEncoder.php
│ │ ├── FileExtensionEncoder.php
│ │ ├── FilePathEncoder.php
│ │ ├── GifEncoder.php
│ │ ├── HeicEncoder.php
│ │ ├── Jpeg2000Encoder.php
│ │ ├── JpegEncoder.php
│ │ ├── MediaTypeEncoder.php
│ │ ├── PngEncoder.php
│ │ ├── TiffEncoder.php
│ │ └── WebpEncoder.php
│ ├── Exceptions/
│ │ ├── AnimationException.php
│ │ ├── ColorException.php
│ │ ├── DecoderException.php
│ │ ├── DriverException.php
│ │ ├── EncoderException.php
│ │ ├── FontException.php
│ │ ├── GeometryException.php
│ │ ├── InputException.php
│ │ ├── NotSupportedException.php
│ │ ├── NotWritableException.php
│ │ └── RuntimeException.php
│ ├── File.php
│ ├── FileExtension.php
│ ├── Format.php
│ ├── Geometry/
│ │ ├── Bezier.php
│ │ ├── Circle.php
│ │ ├── Ellipse.php
│ │ ├── Factories/
│ │ │ ├── BezierFactory.php
│ │ │ ├── CircleFactory.php
│ │ │ ├── Drawable.php
│ │ │ ├── EllipseFactory.php
│ │ │ ├── LineFactory.php
│ │ │ ├── PolygonFactory.php
│ │ │ └── RectangleFactory.php
│ │ ├── Line.php
│ │ ├── Pixel.php
│ │ ├── Point.php
│ │ ├── Polygon.php
│ │ ├── Rectangle.php
│ │ ├── Tools/
│ │ │ └── RectangleResizer.php
│ │ └── Traits/
│ │ ├── HasBackgroundColor.php
│ │ └── HasBorder.php
│ ├── Image.php
│ ├── ImageManager.php
│ ├── InputHandler.php
│ ├── Interfaces/
│ │ ├── AnalyzerInterface.php
│ │ ├── CollectionInterface.php
│ │ ├── ColorChannelInterface.php
│ │ ├── ColorInterface.php
│ │ ├── ColorProcessorInterface.php
│ │ ├── ColorspaceInterface.php
│ │ ├── CoreInterface.php
│ │ ├── DecoderInterface.php
│ │ ├── DrawableFactoryInterface.php
│ │ ├── DrawableInterface.php
│ │ ├── DriverInterface.php
│ │ ├── EncodedImageInterface.php
│ │ ├── EncoderInterface.php
│ │ ├── FileInterface.php
│ │ ├── FontInterface.php
│ │ ├── FontProcessorInterface.php
│ │ ├── FrameInterface.php
│ │ ├── ImageInterface.php
│ │ ├── ImageManagerInterface.php
│ │ ├── InputHandlerInterface.php
│ │ ├── ModifierInterface.php
│ │ ├── PointInterface.php
│ │ ├── ProfileInterface.php
│ │ ├── ResolutionInterface.php
│ │ ├── SizeInterface.php
│ │ ├── SpecializableInterface.php
│ │ └── SpecializedInterface.php
│ ├── MediaType.php
│ ├── ModifierStack.php
│ ├── Modifiers/
│ │ ├── AbstractDrawModifier.php
│ │ ├── AlignRotationModifier.php
│ │ ├── BlendTransparencyModifier.php
│ │ ├── BlurModifier.php
│ │ ├── BrightnessModifier.php
│ │ ├── ColorizeModifier.php
│ │ ├── ColorspaceModifier.php
│ │ ├── ContainModifier.php
│ │ ├── ContrastModifier.php
│ │ ├── CoverDownModifier.php
│ │ ├── CoverModifier.php
│ │ ├── CropModifier.php
│ │ ├── DrawBezierModifier.php
│ │ ├── DrawEllipseModifier.php
│ │ ├── DrawLineModifier.php
│ │ ├── DrawPixelModifier.php
│ │ ├── DrawPolygonModifier.php
│ │ ├── DrawRectangleModifier.php
│ │ ├── FillModifier.php
│ │ ├── FlipModifier.php
│ │ ├── FlopModifier.php
│ │ ├── GammaModifier.php
│ │ ├── GreyscaleModifier.php
│ │ ├── InvertModifier.php
│ │ ├── PadModifier.php
│ │ ├── PixelateModifier.php
│ │ ├── PlaceModifier.php
│ │ ├── ProfileModifier.php
│ │ ├── ProfileRemovalModifier.php
│ │ ├── QuantizeColorsModifier.php
│ │ ├── RemoveAnimationModifier.php
│ │ ├── ResizeCanvasModifier.php
│ │ ├── ResizeCanvasRelativeModifier.php
│ │ ├── ResizeDownModifier.php
│ │ ├── ResizeModifier.php
│ │ ├── ResolutionModifier.php
│ │ ├── RotateModifier.php
│ │ ├── ScaleDownModifier.php
│ │ ├── ScaleModifier.php
│ │ ├── SharpenModifier.php
│ │ ├── SliceAnimationModifier.php
│ │ ├── TextModifier.php
│ │ └── TrimModifier.php
│ ├── Origin.php
│ ├── Resolution.php
│ ├── Traits/
│ │ ├── CanBeDriverSpecialized.php
│ │ └── CanBuildFilePointer.php
│ └── Typography/
│ ├── Font.php
│ ├── FontFactory.php
│ ├── Line.php
│ └── TextBlock.php
└── tests/
├── BaseTestCase.php
├── Feature/
│ ├── Gd/
│ │ └── ConvertPngGif.php
│ └── Imagick/
│ ├── ConvertPngGif.php
│ └── CropResizePngTest.php
├── GdTestCase.php
├── ImagickTestCase.php
├── Traits/
│ ├── CanDetectProgressiveJpeg.php
│ └── CanInspectPngFormat.php
└── Unit/
├── CollectionTest.php
├── Colors/
│ ├── Cmyk/
│ │ ├── ChannelTest.php
│ │ ├── ColorTest.php
│ │ ├── ColorspaceTest.php
│ │ └── Decoders/
│ │ └── StringColorDecoderTest.php
│ ├── Hsl/
│ │ ├── ChannelTest.php
│ │ ├── Channels/
│ │ │ └── SaturationTest.php
│ │ ├── ColorTest.php
│ │ ├── ColorspaceTest.php
│ │ └── Decoders/
│ │ └── StringColorDecoderTest.php
│ ├── Hsv/
│ │ ├── ChannelTest.php
│ │ ├── Channels/
│ │ │ ├── SaturationTest.php
│ │ │ └── ValueTest.php
│ │ ├── ColorTest.php
│ │ ├── ColorspaceTest.php
│ │ └── Decoders/
│ │ └── StringColorDecoderTest.php
│ ├── ProfileTest.php
│ └── Rgb/
│ ├── ChannelTest.php
│ ├── Channels/
│ │ └── AlphaTest.php
│ ├── ColorTest.php
│ ├── ColorspaceTest.php
│ └── Decoders/
│ ├── HexColorDecoderTest.php
│ ├── HtmlColornameDecoderTest.php
│ └── StringColorDecoderTest.php
├── ConfigTest.php
├── Drivers/
│ ├── AbstractDecoderTest.php
│ ├── AbstractEncoderTest.php
│ ├── AbstractFontProcessorTest.php
│ ├── Gd/
│ │ ├── Analyzers/
│ │ │ ├── ColorspaceAnalyzerTest.php
│ │ │ ├── HeightAnalyzerTest.php
│ │ │ ├── PixelColorAnalyzerTest.php
│ │ │ ├── PixelColorsAnalyzerTest.php
│ │ │ ├── ResolutionAnalyzerTest.php
│ │ │ └── WidthAnalyzerTest.php
│ │ ├── ClonerTest.php
│ │ ├── ColorProcessorTest.php
│ │ ├── CoreTest.php
│ │ ├── Decoders/
│ │ │ ├── AbstractDecoderTest.php
│ │ │ ├── Base64ImageDecoderTest.php
│ │ │ ├── BinaryImageDecoderTest.php
│ │ │ ├── DataUriImageDecoderTest.php
│ │ │ ├── EncodedImageObjectDecoderTest.php
│ │ │ ├── FilePathImageDecoderTest.php
│ │ │ ├── FilePointerImageDecoderTest.php
│ │ │ ├── ImageObjectDecoderTest.php
│ │ │ ├── NativeObjectDecoderTest.php
│ │ │ └── SplFileInfoImageDecoderTest.php
│ │ ├── DriverTest.php
│ │ ├── Encoders/
│ │ │ ├── AvifEncoderTest.php
│ │ │ ├── BmpEncoderTest.php
│ │ │ ├── GifEncoderTest.php
│ │ │ ├── JpegEncoderTest.php
│ │ │ ├── PngEncoderTest.php
│ │ │ └── WebpEncoderTest.php
│ │ ├── FontProcessorTest.php
│ │ ├── FrameTest.php
│ │ ├── ImageTest.php
│ │ └── Modifiers/
│ │ ├── BlurModifierTest.php
│ │ ├── BrightnessModifierTest.php
│ │ ├── ColorizeModifierTest.php
│ │ ├── ContainModifierTest.php
│ │ ├── ContrastModifierTest.php
│ │ ├── CoverDownModifierTest.php
│ │ ├── CoverModifierTest.php
│ │ ├── CropModifierTest.php
│ │ ├── DrawBezierModifierTest.php
│ │ ├── DrawEllipseModifierTest.php
│ │ ├── DrawLineModifierTest.php
│ │ ├── DrawPixelModifierTest.php
│ │ ├── DrawPolygonModifierTest.php
│ │ ├── DrawRectangleModifierTest.php
│ │ ├── FillModifierTest.php
│ │ ├── FlipFlopModifierTest.php
│ │ ├── GammaModifierTest.php
│ │ ├── GreyscaleModifierTest.php
│ │ ├── InvertModifierTest.php
│ │ ├── PadModifierTest.php
│ │ ├── PixelateModifierTest.php
│ │ ├── PlaceModifierTest.php
│ │ ├── QuantizeColorsModifierTest.php
│ │ ├── RemoveAnimationModifierTest.php
│ │ ├── ResizeCanvasModifierTest.php
│ │ ├── ResizeCanvasRelativeModifierTest.php
│ │ ├── ResizeModifierTest.php
│ │ ├── ResolutionModifierTest.php
│ │ ├── RotateModifierTest.php
│ │ ├── SharpenModifierTest.php
│ │ ├── TextModifierTest.php
│ │ └── TrimModifierTest.php
│ ├── Imagick/
│ │ ├── Analyzers/
│ │ │ ├── ColorspaceAnalyzerTest.php
│ │ │ ├── HeightAnalyzerTest.php
│ │ │ ├── PixelColorAnalyzerTest.php
│ │ │ ├── PixelColorsAnalyzerTest.php
│ │ │ ├── ProfileAnalyzerTest.php
│ │ │ ├── ResolutionAnalyzerTest.php
│ │ │ └── WidthAnalyzerTest.php
│ │ ├── ColorProcessorTest.php
│ │ ├── CoreTest.php
│ │ ├── Decoders/
│ │ │ ├── Base64ImageDecoderTest.php
│ │ │ ├── BinaryImageDecoderTest.php
│ │ │ ├── DataUriImageDecoderTest.php
│ │ │ ├── EncodedImageObjectDecoderTest.php
│ │ │ ├── FilePathImageDecoderTest.php
│ │ │ ├── FilePointerImageDecoderTest.php
│ │ │ ├── ImageObjectDecoderTest.php
│ │ │ ├── NativeObjectDecoderTest.php
│ │ │ └── SplFileInfoImageDecoderTest.php
│ │ ├── DriverTest.php
│ │ ├── Encoders/
│ │ │ ├── AvifEncoderTest.php
│ │ │ ├── BmpEncoderTest.php
│ │ │ ├── GifEncoderTest.php
│ │ │ ├── HeicEncoderTest.php
│ │ │ ├── Jpeg2000EncoderTest.php
│ │ │ ├── JpegEncoderTest.php
│ │ │ ├── PngEncoderTest.php
│ │ │ ├── TiffEncoderTest.php
│ │ │ └── WebpEncoderTest.php
│ │ ├── FontProcessorTest.php
│ │ ├── FrameTest.php
│ │ ├── ImageTest.php
│ │ └── Modifiers/
│ │ ├── BlurModifierTest.php
│ │ ├── BrightnessModifierTest.php
│ │ ├── ColorizeModifierTest.php
│ │ ├── ContainModifierTest.php
│ │ ├── ContrastModifierTest.php
│ │ ├── CoverDownModifierTest.php
│ │ ├── CoverModifierTest.php
│ │ ├── CropModifierTest.php
│ │ ├── DrawBezierModifierTest.php
│ │ ├── DrawEllipseModifierTest.php
│ │ ├── DrawLineModifierTest.php
│ │ ├── DrawPixelModifierTest.php
│ │ ├── DrawPolygonModifierTest.php
│ │ ├── DrawRectangleModifierTest.php
│ │ ├── FillModifierTest.php
│ │ ├── FlipFlopModifierTest.php
│ │ ├── GammaModifierTest.php
│ │ ├── GreyscaleModifierTest.php
│ │ ├── InvertModifierTest.php
│ │ ├── PadModifierTest.php
│ │ ├── PixelateModifierTest.php
│ │ ├── PlaceModifierTest.php
│ │ ├── QuantizeColorsModifierTest.php
│ │ ├── RemoveAnimationModifierTest.php
│ │ ├── ResizeCanvasModifierTest.php
│ │ ├── ResizeCanvasRelativeModifierTest.php
│ │ ├── ResizeModifierTest.php
│ │ ├── ResolutionModifierTest.php
│ │ ├── RotateModifierTest.php
│ │ ├── SharpenModifierTest.php
│ │ ├── StripMetaModifierTest.php
│ │ ├── TextModifierTest.php
│ │ └── TrimModifierTest.php
│ ├── SpecializableAnalyzerTest.php
│ ├── SpecializableDecoderTest.php
│ └── SpecializableModifierTest.php
├── EncodedImageTest.php
├── Encoders/
│ ├── FileExtensionEncoderTest.php
│ └── MediaTypeEncoderTest.php
├── FileExtensionTest.php
├── FileTest.php
├── FormatTest.php
├── Geometry/
│ ├── BezierTest.php
│ ├── CircleTest.php
│ ├── EllipseTest.php
│ ├── Factories/
│ │ ├── BezierFactoryTest.php
│ │ ├── CircleFactoryTest.php
│ │ ├── DrawableTest.php
│ │ ├── EllipseFactoryTest.php
│ │ ├── LineFactoryTest.php
│ │ ├── PolygonFactoryTest.php
│ │ └── RectangleFactoryTest.php
│ ├── LineTest.php
│ ├── PixelTest.php
│ ├── PointTest.php
│ ├── PolygonTest.php
│ ├── RectangleResizerTest.php
│ ├── RectangleTest.php
│ └── Traits/
│ ├── HasBackgroundColorTest.php
│ └── HasBorderTest.php
├── ImageManagerTestGd.php
├── ImageManagerTestImagick.php
├── InputHandlerTest.php
├── MediaTypeTest.php
├── ModifierStackTest.php
├── Modifiers/
│ ├── ColorspaceModifierTest.php
│ ├── PadModifierTest.php
│ ├── RemoveAnimationModifierTest.php
│ └── TextModifierTest.php
├── OriginTest.php
├── ResolutionTest.php
└── Typography/
├── FontFactoryTest.php
├── FontTest.php
├── LineTest.php
└── TextBlockTest.php
================================================
FILE CONTENTS
================================================
================================================
FILE: .editorconfig
================================================
# editorconfig.org
root = true
[*.php]
indent_size = 4
indent_style = space
indent_size = 4
end_of_line = lf
charset = utf-8
trim_trailing_whitespace = true
insert_final_newline = true
================================================
FILE: .gitattributes
================================================
* text=auto
/.editorconfig export-ignore
/.github export-ignore
/.well-known export-ignore
/CONTRIBUTING.md export-ignore
/tests export-ignore
/.gitattributes export-ignore
/.gitignore export-ignore
/phpcs.xml.dist export-ignore
/phpunit.dist.xml export-ignore
/phpstan.dist.neon export-ignore
/docker-compose.yml export-ignore
/Dockerfile export-ignore
================================================
FILE: .github/FUNDING.yml
================================================
github: [Intervention]
ko_fi: interventionphp
custom: https://paypal.me/interventionio
================================================
FILE: .github/ISSUE_TEMPLATE/bug_report.md
================================================
---
name: Bug report
about: Create a report to help us improve
title: ''
labels: ''
assignees: ''
---
## Describe the bug
A clear and concise description of what the bug is.
## Code Example
Code example to reproduce the behavior.
## Expected behavior
A clear and concise description of what you expected to happen.
## Images
If applicable, add problematic images or screenshots to help explain your problem.
## Environment (please complete the following information):
- PHP Version:
- OS:
- Intervention Image Version:
- GD, Imagick or libvips:
================================================
FILE: .github/ISSUE_TEMPLATE/feature_request.md
================================================
---
name: Feature request
about: Suggest an idea for this project
title: ''
labels: ''
assignees: ''
---
## Describe the feature you'd like
A clear and concise description of what you want to happen.
## Is your feature request related to a problem? Please describe.
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
================================================
FILE: .github/workflows/run-tests.yml
================================================
name: Tests
on: [ push, pull_request ]
jobs:
run:
runs-on: ubuntu-22.04
strategy:
fail-fast: false
matrix:
php: [ '8.1', '8.2', '8.3', '8.4', '8.5' ]
imagemagick: [ '6.9.13-25', '7.1.1-47' ]
name: PHP ${{ matrix.php }} - ImageMagick ${{ matrix.imagemagick }}
steps:
- name: Checkout project
uses: actions/checkout@v4
- name: Setup PHP
uses: shivammathur/setup-php@v2
with:
php-version: ${{ matrix.php }}
extensions: mbstring, gd
coverage: none
- name: Prepare environment for Imagemagick
run: |
sudo apt remove -y imagemagick imagemagick-6-common libmagic-dev
sudo apt update --allow-releaseinfo-change
sudo apt update
sudo apt install -y libjpeg-dev libgif-dev libtiff-dev libpng-dev libwebp-dev libavif-dev libheif-dev libraqm-dev libmagickwand-dev
- name: Cache ImageMagick
uses: actions/cache@v4
id: cache-imagemagick
with:
path: /home/runner/im/imagemagick-${{ matrix.imagemagick }}
key: ${{ runner.os }}-ImageMagick-${{ matrix.imagemagick }}-${{ hashFiles('**/composer.json') }}
restore-keys: ${{ runner.os }}-ImageMagick-${{ matrix.imagemagick }}-
- name: Check ImageMagick cache exists
uses: andstor/file-existence-action@v3
id: cache-imagemagick-exists
with:
files: /home/runner/im/imagemagick-${{ matrix.imagemagick }}
- name: Install ImageMagick
if: ( steps.cache-imagemagick.outputs.cache-hit != 'true' || steps.cache-imagemagick-exists.outputs.files_exists != 'true' )
run: |
curl -o /tmp/ImageMagick.tar.xz -sL https://imagemagick.org/archive/releases/ImageMagick-${{ matrix.imagemagick }}.tar.xz
(
cd /tmp || exit 1
tar xf ImageMagick.tar.xz
cd ImageMagick-${{ matrix.imagemagick }}
sudo ./configure --prefix=/home/runner/im/imagemagick-${{ matrix.imagemagick }}
sudo make -j$(nproc)
sudo make install
)
- name: Install Imagick PHP extension
run: |
git clone --depth=1 --single-branch --branch master https://github.com/Imagick/imagick /tmp/imagick
(
cd /tmp/imagick || exit 1
phpize
sudo ./configure --with-imagick=/home/runner/im/imagemagick-${{ matrix.imagemagick }}
sudo make -j$(nproc)
sudo make install
)
sudo bash -c 'echo "extension=imagick.so" >> /etc/php/${{ matrix.php }}/cli/php.ini'
php --ri imagick;
- name: Get composer cache directory
id: composer-cache
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
- name: Cache composer dependencies
uses: actions/cache@v4
with:
path: ${{ steps.composer-cache.outputs.dir }}
key: ${{ runner.os }}-php-${{ matrix.php }}-${{ hashFiles('**/composer.json') }}
restore-keys: ${{ runner.os }}-php-${{ matrix.php }}-
- name: Install dependencies
run: composer update --prefer-stable --prefer-dist --no-interaction
- name: GD Version
run: php -r 'var_dump(gd_info());'
- name: Imagick Version
run: php -r 'var_dump(Imagick::getVersion());'
- name: Supported Imagick Formats
run: php -r 'var_dump(Imagick::queryFormats());'
- name: Execute tests
run: vendor/bin/phpunit --no-coverage
- name: Run analyzer
run: vendor/bin/phpstan
- name: Validate coding standards
run: vendor/bin/phpcs
================================================
FILE: .gitignore
================================================
build/
vendor/
/.phpunit.cache
composer.lock
phpunit.xml
phpcs.xml
phpstan.neon
================================================
FILE: .well-known/funding-manifest-urls
================================================
https://intervention.io/funding.json
https://image.intervention.io/funding.json
================================================
FILE: CONTRIBUTING.md
================================================
# Contributing to Intervention Image
Thank you for your interest in contributing to the project.
You can read the full contribution guide on https://intervention.io/contributing/.
================================================
FILE: Dockerfile
================================================
FROM php:8.1-cli
# install dependencies
RUN apt update \
&& apt install -y \
libmagickwand-dev \
libwebp-dev \
libpng-dev \
libavif-dev \
git \
zip \
&& pecl install imagick \
&& pecl install xdebug \
&& docker-php-ext-configure gd --with-freetype --with-jpeg --with-webp --with-avif \
&& docker-php-ext-enable \
imagick \
xdebug \
&& docker-php-ext-install \
gd \
exif \
&& apt-get clean
# install composer
COPY --from=composer /usr/bin/composer /usr/bin/composer
# install composer dependencies
COPY composer.json composer.lock ./
RUN composer install
================================================
FILE: LICENSE
================================================
The MIT License (MIT)
Copyright (c) 2013-present Oliver Vogel
Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
of the Software, and to permit persons to whom the Software is furnished to do
so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
================================================
FILE: composer.json
================================================
{
"name": "intervention/image",
"description": "PHP Image Processing",
"homepage": "https://image.intervention.io",
"keywords": [
"image",
"gd",
"imagick",
"watermark",
"thumbnail",
"resize"
],
"license": "MIT",
"authors": [
{
"name": "Oliver Vogel",
"email": "oliver@intervention.io",
"homepage": "https://intervention.io"
}
],
"require": {
"php": "^8.1",
"ext-mbstring": "*",
"intervention/gif": "^4.2"
},
"require-dev": {
"phpunit/phpunit": "^10.0 || ^11.0 || ^12.0",
"mockery/mockery": "^1.6",
"phpstan/phpstan": "^2.1",
"squizlabs/php_codesniffer": "^4",
"slevomat/coding-standard": "~8.0"
},
"suggest": {
"ext-exif": "Recommended to be able to read EXIF data properly."
},
"autoload": {
"psr-4": {
"Intervention\\Image\\": "src"
}
},
"autoload-dev": {
"psr-4": {
"Intervention\\Image\\Tests\\": "tests"
}
},
"config": {
"allow-plugins": {
"dealerdirect/phpcodesniffer-composer-installer": true
}
}
}
================================================
FILE: docker-compose.yml
================================================
services:
tests:
build: ./
working_dir: /project
entrypoint: ["./vendor/bin/phpunit"]
volumes:
- ./:/project
coverage:
build: ./
working_dir: /project
entrypoint: ["./vendor/bin/phpunit", "--coverage-text"]
volumes:
- ./:/project
environment:
- XDEBUG_MODE=coverage
analysis:
build: ./
working_dir: /project
entrypoint: ["./vendor/bin/phpstan", "analyze", "--memory-limit=512M"]
volumes:
- ./:/project
standards:
build: ./
working_dir: /project
entrypoint: ["./vendor/bin/phpcs"]
volumes:
- ./:/project
================================================
FILE: phpcs.xml.dist
================================================
src/
tests/
================================================
FILE: phpstan.dist.neon
================================================
parameters:
level: 6
paths:
- src
reportUnmatchedIgnoredErrors: false
ignoreErrors:
-
identifier: unset.possiblyHookedProperty
exceptions:
check:
missingCheckedExceptionInThrows: true
uncheckedExceptionClasses:
- ImagickException
- ImagickDrawException
- ImagickPixelException
- Error
================================================
FILE: phpunit.dist.xml
================================================
./tests/Unit
./tests/Feature
src
================================================
FILE: readme.md
================================================
# Intervention Image
## PHP Image Processing
[](https://packagist.org/packages/intervention/image)
[](https://github.com/Intervention/image/actions)
[](https://packagist.org/packages/intervention/image/stats)
[](https://ko-fi.com/interventionphp)
Intervention Image is a **PHP image processing library** that provides a simple
and expressive way to create, edit, and compose images. It comes with a universal
interface for the two most popular PHP image manipulation extensions. You can
choose between the GD library or Imagick as the base layer for all operations.
- Simple interface for common image editing tasks
- Interchangeable driver architecture
- Support for animated images
- Framework-agnostic
- PSR-12 compliant
## Installation
You can easily install this library using [Composer](https://getcomposer.org).
Simply request the package with the following command:
```bash
composer require intervention/image
```
## Getting Started
Learn the [basics](https://image.intervention.io/v3/basics/instantiation/) on
how to use Intervention Image and more with the [official
documentation](https://image.intervention.io/v3/).
## Code Examples
```php
use Intervention\Image\ImageManager;
// create image manager with desired driver
$manager = new ImageManager(
new Intervention\Image\Drivers\Gd\Driver()
);
// open an image file
$image = $manager->read('images/example.gif');
// resize image instance
$image->resize(height: 300);
// insert a watermark
$image->place('images/watermark.png');
// encode edited image
$encoded = $image->toJpg();
// save encoded image
$encoded->save('images/example.jpg');
```
## Requirements
Before you begin with the installation make sure that your server environment
supports the following requirements.
- PHP >= 8.1
- Mbstring PHP Extension
- Image Processing PHP Extension
## Supported Image Libraries
Depending on your environment Intervention Image lets you choose between
different image processing extensions.
- GD Library
- Imagick PHP extension
- [libvips](https://github.com/Intervention/image-driver-vips)
## Security
If you discover any security related issues, please email oliver@intervention.io directly.
## Authors
This library is developed and maintained by [Oliver Vogel](https://intervention.io)
Thanks to the community of [contributors](https://github.com/Intervention/image/graphs/contributors) who have helped to improve this project.
## License
Intervention Image is licensed under the [MIT License](LICENSE).
================================================
FILE: src/Analyzers/ColorspaceAnalyzer.php
================================================
*/
class Collection implements CollectionInterface, IteratorAggregate, Countable
{
/**
* Create new collection object
*
* @param array $items
* @return void
*/
public function __construct(protected array $items = [])
{
//
}
/**
* Static constructor
*
* @param array $items
* @return self
*/
public static function create(array $items = []): self
{
return new self($items);
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::has()
*/
public function has(int|string $key): bool
{
return array_key_exists($key, $this->items);
}
/**
* Returns Iterator
*
* @return Traversable
*/
public function getIterator(): Traversable
{
return new ArrayIterator($this->items);
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::toArray()
*/
public function toArray(): array
{
return $this->items;
}
/**
* Count items in collection
*/
public function count(): int
{
return count($this->items);
}
/**
* Append new item to collection
*
* @return CollectionInterface
*/
public function push(mixed $item): CollectionInterface
{
$this->items[] = $item;
return $this;
}
/**
* Return first item in collection
*/
public function first(): mixed
{
if ($item = reset($this->items)) {
return $item;
}
return null;
}
/**
* Returns last item in collection
*/
public function last(): mixed
{
if ($item = end($this->items)) {
return $item;
}
return null;
}
/**
* Return item at given position starting at 0
*/
public function getAtPosition(int $key = 0, mixed $default = null): mixed
{
if ($this->count() == 0) {
return $default;
}
$positions = array_values($this->items);
if (!array_key_exists($key, $positions)) {
return $default;
}
return $positions[$key];
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::get()
*/
public function get(int|string $query, mixed $default = null): mixed
{
if ($this->count() == 0) {
return $default;
}
if (is_int($query) && array_key_exists($query, $this->items)) {
return $this->items[$query];
}
if (is_string($query) && !str_contains($query, '.')) {
return array_key_exists($query, $this->items) ? $this->items[$query] : $default;
}
$query = explode('.', (string) $query);
$result = $default;
$items = $this->items;
foreach ($query as $key) {
if (!is_array($items) || !array_key_exists($key, $items)) {
$result = $default;
break;
}
$result = $items[$key];
$items = $result;
}
return $result;
}
/**
* Map each item of collection by given callback
*/
public function map(callable $callback): self
{
return new self(
array_map(
fn(mixed $item) => $callback($item),
$this->items,
)
);
}
/**
* Run callback on each item of the collection an remove it if it does not return true
*/
public function filter(callable $callback): self
{
return new self(
array_filter(
$this->items,
fn(mixed $item) => $callback($item),
)
);
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::empty()
*/
public function empty(): CollectionInterface
{
$this->items = [];
return $this;
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::slice()
*/
public function slice(int $offset, ?int $length = null): CollectionInterface
{
$this->items = array_slice($this->items, $offset, $length);
return $this;
}
}
================================================
FILE: src/Colors/AbstractColor.php
================================================
*/
protected array $channels;
/**
* {@inheritdoc}
*
* @see ColorInterface::channels()
*/
public function channels(): array
{
return $this->channels;
}
/**
* {@inheritdoc}
*
* @see ColorInterface::channel()
*/
public function channel(string $classname): ColorChannelInterface
{
$channels = array_filter(
$this->channels(),
fn(ColorChannelInterface $channel): bool => $channel::class === $classname,
);
if (count($channels) == 0) {
throw new ColorException('Color channel ' . $classname . ' could not be found.');
}
return reset($channels);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::normalize()
*/
public function normalize(): array
{
return array_map(
fn(ColorChannelInterface $channel): float => $channel->normalize(),
$this->channels(),
);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::toArray()
*/
public function toArray(): array
{
return array_map(
fn(ColorChannelInterface $channel): int => $channel->value(),
$this->channels()
);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::convertTo()
*/
public function convertTo(string|ColorspaceInterface $colorspace): ColorInterface
{
$colorspace = match (true) {
is_object($colorspace) => $colorspace,
default => new $colorspace(),
};
return $colorspace->importColor($this);
}
/**
* Show debug info for the current color
*
* @return array
*/
public function __debugInfo(): array
{
return array_reduce($this->channels(), function (array $result, ColorChannelInterface $item) {
$key = strtolower((new ReflectionClass($item))->getShortName());
$result[$key] = $item->value();
return $result;
}, []);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::__toString()
*/
public function __toString(): string
{
return $this->toString();
}
}
================================================
FILE: src/Colors/AbstractColorChannel.php
================================================
value = $this->validate(
match (true) {
is_null($value) && is_numeric($normalized) => intval(round($normalized * $this->max())),
is_numeric($value) && is_null($normalized) => $value,
default => throw new ColorException('Color channels must either have a value or a normalized value')
}
);
}
/**
* Alias of value()
*/
public function toInt(): int
{
return $this->value;
}
/**
* {@inheritdoc}
*
* @see ColorChannelInterface::value()
*/
public function value(): int
{
return $this->value;
}
/**
* {@inheritdoc}
*
* @see ColorChannelInterface::normalize()
*/
public function normalize(int $precision = 32): float
{
return round(($this->value() - $this->min()) / ($this->max() - $this->min()), $precision);
}
/**
* {@inheritdoc}
*
* @see ColorChannelInterface::validate()
*/
public function validate(mixed $value): mixed
{
if ($value < $this->min() || $value > $this->max()) {
throw new ColorException('Color channel value must be in range ' . $this->min() . ' to ' . $this->max());
}
return $value;
}
/**
* {@inheritdoc}
*
* @see ColorChannelInterface::toString()
*/
public function toString(): string
{
return (string) $this->value();
}
/**
* {@inheritdoc}
*
* @see ColorChannelInterface::__toString()
*/
public function __toString(): string
{
return $this->toString();
}
}
================================================
FILE: src/Colors/Cmyk/Channels/Cyan.php
================================================
channels = [
new Cyan($c),
new Magenta($m),
new Yellow($y),
new Key($k),
];
}
/**
* {@inheritdoc}
*
* @see ColorInterface::create()
*/
public static function create(mixed $input): ColorInterface
{
return InputHandler::withDecoders([
Decoders\StringColorDecoder::class,
])->handle($input);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::colorspace()
*/
public function colorspace(): ColorspaceInterface
{
return new Colorspace();
}
/**
* {@inheritdoc}
*
* @see ColorInterface::toHex()
*/
public function toHex(string $prefix = ''): string
{
return $this->convertTo(RgbColorspace::class)->toHex($prefix);
}
/**
* Return the CMYK cyan channel
*/
public function cyan(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Cyan::class);
}
/**
* Return the CMYK magenta channel
*/
public function magenta(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Magenta::class);
}
/**
* Return the CMYK yellow channel
*/
public function yellow(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Yellow::class);
}
/**
* Return the CMYK key channel
*/
public function key(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Key::class);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::toString()
*/
public function toString(): string
{
return sprintf(
'cmyk(%d%%, %d%%, %d%%, %d%%)',
$this->cyan()->value(),
$this->magenta()->value(),
$this->yellow()->value(),
$this->key()->value()
);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isGreyscale()
*/
public function isGreyscale(): bool
{
return 0 === array_sum([
$this->cyan()->value(),
$this->magenta()->value(),
$this->yellow()->value(),
]);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isTransparent()
*/
public function isTransparent(): bool
{
return false;
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isClear()
*/
public function isClear(): bool
{
return false;
}
}
================================================
FILE: src/Colors/Cmyk/Colorspace.php
================================================
*/
public static array $channels = [
Channels\Cyan::class,
Channels\Magenta::class,
Channels\Yellow::class,
Channels\Key::class
];
/**
* {@inheritdoc}
*
* @see ColorspaceInterface::createColor()
*/
public function colorFromNormalized(array $normalized): ColorInterface
{
return new Color(...array_map(
fn(string $classname, float $value_normalized) => (new $classname(normalized: $value_normalized))->value(),
self::$channels,
$normalized,
));
}
/**
* @throws ColorException
*/
public function importColor(ColorInterface $color): ColorInterface
{
return match ($color::class) {
RgbColor::class => $this->importRgbColor($color),
HsvColor::class => $this->importRgbColor($color->convertTo(RgbColorspace::class)),
HslColor::class => $this->importRgbColor($color->convertTo(RgbColorspace::class)),
default => $color,
};
}
/**
* @throws ColorException
*/
protected function importRgbColor(ColorInterface $color): CmykColor
{
if (!($color instanceof RgbColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
$c = (255 - $color->red()->value()) / 255.0 * 100;
$m = (255 - $color->green()->value()) / 255.0 * 100;
$y = (255 - $color->blue()->value()) / 255.0 * 100;
$k = intval(round(min([$c, $m, $y])));
$c = intval(round($c - $k));
$m = intval(round($m - $k));
$y = intval(round($y - $k));
return new CmykColor($c, $m, $y, $k);
}
}
================================================
FILE: src/Colors/Cmyk/Decoders/StringColorDecoder.php
================================================
[0-9\.]+%?), ?(?P[0-9\.]+%?), ?(?P[0-9\.]+%?), ?(?P[0-9\.]+%?)\)$/i';
if (preg_match($pattern, $input, $matches) != 1) {
throw new DecoderException('Unable to decode input');
}
$values = array_map(function (string $value): int {
return intval(round(floatval(trim(str_replace('%', '', $value)))));
}, [$matches['c'], $matches['m'], $matches['y'], $matches['k']]);
return new Color(...$values);
}
}
================================================
FILE: src/Colors/Hsl/Channels/Hue.php
================================================
channels = [
new Hue($h),
new Saturation($s),
new Luminance($l),
];
}
/**
* {@inheritdoc}
*
* @see ColorInterface::colorspace()
*/
public function colorspace(): ColorspaceInterface
{
return new Colorspace();
}
/**
* {@inheritdoc}
*
* @see ColorInterface::create()
*/
public static function create(mixed $input): ColorInterface
{
return InputHandler::withDecoders([
Decoders\StringColorDecoder::class,
])->handle($input);
}
/**
* Return the Hue channel
*/
public function hue(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Hue::class);
}
/**
* Return the Saturation channel
*/
public function saturation(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Saturation::class);
}
/**
* Return the Luminance channel
*/
public function luminance(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Luminance::class);
}
public function toHex(string $prefix = ''): string
{
return $this->convertTo(RgbColorspace::class)->toHex($prefix);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::toString()
*/
public function toString(): string
{
return sprintf(
'hsl(%d, %d%%, %d%%)',
$this->hue()->value(),
$this->saturation()->value(),
$this->luminance()->value()
);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isGreyscale()
*/
public function isGreyscale(): bool
{
return $this->saturation()->value() == 0;
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isTransparent()
*/
public function isTransparent(): bool
{
return false;
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isClear()
*/
public function isClear(): bool
{
return false;
}
}
================================================
FILE: src/Colors/Hsl/Colorspace.php
================================================
*/
public static array $channels = [
Channels\Hue::class,
Channels\Saturation::class,
Channels\Luminance::class
];
/**
* {@inheritdoc}
*
* @see ColorspaceInterface::colorFromNormalized()
*/
public function colorFromNormalized(array $normalized): ColorInterface
{
return new Color(...array_map(
fn(string $classname, float $value_normalized) => (new $classname(normalized: $value_normalized))->value(),
self::$channels,
$normalized
));
}
/**
* @throws ColorException
*/
public function importColor(ColorInterface $color): ColorInterface
{
return match ($color::class) {
CmykColor::class => $this->importRgbColor($color->convertTo(RgbColorspace::class)),
RgbColor::class => $this->importRgbColor($color),
HsvColor::class => $this->importHsvColor($color),
default => $color,
};
}
/**
* @throws ColorException
*/
protected function importRgbColor(ColorInterface $color): ColorInterface
{
if (!($color instanceof RgbColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
// normalized values of rgb channels
$values = array_map(
fn(ColorChannelInterface $channel): float => $channel->normalize(),
$color->channels(),
);
// take only RGB
$values = array_slice($values, 0, 3);
// calculate Luminance
$min = min(...$values);
$max = max(...$values);
$luminance = ($max + $min) / 2;
$delta = $max - $min;
// calculate saturation
$saturation = match (true) {
$delta == 0 => 0,
default => $delta / (1 - abs(2 * $luminance - 1)),
};
// calculate hue
[$r, $g, $b] = $values;
$hue = match (true) {
($delta == 0) => 0,
($max == $r) => 60 * fmod((($g - $b) / $delta), 6),
($max == $g) => 60 * ((($b - $r) / $delta) + 2),
($max == $b) => 60 * ((($r - $g) / $delta) + 4),
default => 0,
};
$hue = ($hue + 360) % 360; // normalize hue
return new Color(
intval(round($hue)),
intval(round($saturation * 100)),
intval(round($luminance * 100)),
);
}
/**
* @throws ColorException
*/
protected function importHsvColor(ColorInterface $color): ColorInterface
{
if (!($color instanceof HsvColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
// normalized values of hsv channels
[$h, $s, $v] = array_map(
fn(ColorChannelInterface $channel): float => $channel->normalize(),
$color->channels(),
);
// calculate Luminance
$luminance = (2 - $s) * $v / 2;
// calculate Saturation
$saturation = match (true) {
$luminance == 0 => $s,
$luminance == 1 => 0,
$luminance < .5 => $s * $v / ($luminance * 2),
default => $s * $v / (2 - $luminance * 2),
};
return new Color(
intval(round($h * 360)),
intval(round($saturation * 100)),
intval(round($luminance * 100)),
);
}
}
================================================
FILE: src/Colors/Hsl/Decoders/StringColorDecoder.php
================================================
[0-9\.]+), ?(?P[0-9\.]+%?), ?(?P[0-9\.]+%?)\)$/i';
if (preg_match($pattern, $input, $matches) != 1) {
throw new DecoderException('Unable to decode input');
}
$values = array_map(function (string $value): int {
return match (strpos($value, '%')) {
false => intval(trim($value)),
default => intval(trim(str_replace('%', '', $value))),
};
}, [$matches['h'], $matches['s'], $matches['l']]);
return new Color(...$values);
}
}
================================================
FILE: src/Colors/Hsv/Channels/Hue.php
================================================
channels = [
new Hue($h),
new Saturation($s),
new Value($v),
];
}
/**
* {@inheritdoc}
*
* @see ColorInterface::colorspace()
*/
public function colorspace(): ColorspaceInterface
{
return new Colorspace();
}
/**
* {@inheritdoc}
*
* @see ColorInterface::create()
*/
public static function create(mixed $input): ColorInterface
{
return InputHandler::withDecoders([
Decoders\StringColorDecoder::class,
])->handle($input);
}
/**
* Return the Hue channel
*/
public function hue(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Hue::class);
}
/**
* Return the Saturation channel
*/
public function saturation(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Saturation::class);
}
/**
* Return the Value channel
*/
public function value(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Value::class);
}
public function toHex(string $prefix = ''): string
{
return $this->convertTo(RgbColorspace::class)->toHex($prefix);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::toString()
*/
public function toString(): string
{
return sprintf(
'hsv(%d, %d%%, %d%%)',
$this->hue()->value(),
$this->saturation()->value(),
$this->value()->value()
);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isGreyscale()
*/
public function isGreyscale(): bool
{
return $this->saturation()->value() == 0;
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isTransparent()
*/
public function isTransparent(): bool
{
return false;
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isClear()
*/
public function isClear(): bool
{
return false;
}
}
================================================
FILE: src/Colors/Hsv/Colorspace.php
================================================
*/
public static array $channels = [
Channels\Hue::class,
Channels\Saturation::class,
Channels\Value::class
];
/**
* {@inheritdoc}
*
* @see ColorspaceInterface::colorFromNormalized()
*/
public function colorFromNormalized(array $normalized): ColorInterface
{
return new Color(...array_map(
fn(string $classname, float $value_normalized) => (new $classname(normalized: $value_normalized))->value(),
self::$channels,
$normalized
));
}
/**
* @throws ColorException
*/
public function importColor(ColorInterface $color): ColorInterface
{
return match ($color::class) {
CmykColor::class => $this->importRgbColor($color->convertTo(RgbColorspace::class)),
RgbColor::class => $this->importRgbColor($color),
HslColor::class => $this->importHslColor($color),
default => $color,
};
}
/**
* @throws ColorException
*/
protected function importRgbColor(ColorInterface $color): ColorInterface
{
if (!($color instanceof RgbColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
// normalized values of rgb channels
$values = array_map(fn(ColorChannelInterface $channel): float => $channel->normalize(), $color->channels());
// take only RGB
$values = array_slice($values, 0, 3);
// calculate chroma
$min = min(...$values);
$max = max(...$values);
$chroma = $max - $min;
// calculate value
$v = 100 * $max;
if ($chroma == 0) {
// greyscale color
return new Color(0, 0, intval(round($v)));
}
// calculate saturation
$s = 100 * ($chroma / $max);
// calculate hue
[$r, $g, $b] = $values;
$h = match (true) {
($r == $min) => 3 - (($g - $b) / $chroma),
($b == $min) => 1 - (($r - $g) / $chroma),
default => 5 - (($b - $r) / $chroma),
} * 60;
return new Color(
intval(round($h)),
intval(round($s)),
intval(round($v))
);
}
/**
* @throws ColorException
*/
protected function importHslColor(ColorInterface $color): ColorInterface
{
if (!($color instanceof HslColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
// normalized values of hsl channels
[$h, $s, $l] = array_map(
fn(ColorChannelInterface $channel): float => $channel->normalize(),
$color->channels()
);
$v = $l + $s * min($l, 1 - $l);
$s = ($v == 0) ? 0 : 2 * (1 - $l / $v);
return $this->colorFromNormalized([$h, $s, $v]);
}
}
================================================
FILE: src/Colors/Hsv/Decoders/StringColorDecoder.php
================================================
[0-9\.]+), ?(?P[0-9\.]+%?), ?(?P[0-9\.]+%?)\)$/i';
if (preg_match($pattern, $input, $matches) != 1) {
throw new DecoderException('Unable to decode input');
}
$values = array_map(function (string $value): int {
return match (strpos($value, '%')) {
false => intval(trim($value)),
default => intval(trim(str_replace('%', '', $value))),
};
}, [$matches['h'], $matches['s'], $matches['v']]);
return new Color(...$values);
}
}
================================================
FILE: src/Colors/Profile.php
================================================
normalize(), 6));
}
}
================================================
FILE: src/Colors/Rgb/Channels/Blue.php
================================================
channels = [
new Red($r),
new Green($g),
new Blue($b),
new Alpha($a),
];
}
/**
* {@inheritdoc}
*
* @see ColorInterface::colorspace()
*/
public function colorspace(): ColorspaceInterface
{
return new Colorspace();
}
/**
* {@inheritdoc}
*
* @see ColorInterface::create()
*/
public static function create(mixed $input): ColorInterface
{
return InputHandler::withDecoders([
Decoders\HexColorDecoder::class,
Decoders\StringColorDecoder::class,
Decoders\TransparentColorDecoder::class,
Decoders\HtmlColornameDecoder::class,
])->handle($input);
}
/**
* Return the RGB red color channel
*/
public function red(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Red::class);
}
/**
* Return the RGB green color channel
*/
public function green(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Green::class);
}
/**
* Return the RGB blue color channel
*/
public function blue(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Blue::class);
}
/**
* Return the colors alpha channel
*/
public function alpha(): ColorChannelInterface
{
/** @throws void */
return $this->channel(Alpha::class);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::toHex()
*/
public function toHex(string $prefix = ''): string
{
if ($this->isTransparent()) {
return sprintf(
'%s%02x%02x%02x%02x',
$prefix,
$this->red()->value(),
$this->green()->value(),
$this->blue()->value(),
$this->alpha()->value()
);
}
return sprintf(
'%s%02x%02x%02x',
$prefix,
$this->red()->value(),
$this->green()->value(),
$this->blue()->value()
);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::toString()
*/
public function toString(): string
{
if ($this->isTransparent()) {
return sprintf(
'rgba(%d, %d, %d, %.1F)',
$this->red()->value(),
$this->green()->value(),
$this->blue()->value(),
$this->alpha()->normalize(),
);
}
return sprintf(
'rgb(%d, %d, %d)',
$this->red()->value(),
$this->green()->value(),
$this->blue()->value()
);
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isGreyscale()
*/
public function isGreyscale(): bool
{
$values = [$this->red()->value(), $this->green()->value(), $this->blue()->value()];
return count(array_unique($values, SORT_REGULAR)) === 1;
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isTransparent()
*/
public function isTransparent(): bool
{
return $this->alpha()->value() < $this->alpha()->max();
}
/**
* {@inheritdoc}
*
* @see ColorInterface::isClear()
*/
public function isClear(): bool
{
return $this->alpha()->value() == 0;
}
}
================================================
FILE: src/Colors/Rgb/Colorspace.php
================================================
*/
public static array $channels = [
Channels\Red::class,
Channels\Green::class,
Channels\Blue::class,
Channels\Alpha::class
];
/**
* {@inheritdoc}
*
* @see ColorspaceInterface::colorFromNormalized()
*/
public function colorFromNormalized(array $normalized): ColorInterface
{
return new Color(...array_map(
fn($classname, float $value_normalized) => (new $classname(normalized: $value_normalized))->value(),
self::$channels,
$normalized,
));
}
/**
* @throws ColorException
*/
public function importColor(ColorInterface $color): ColorInterface
{
return match ($color::class) {
CmykColor::class => $this->importCmykColor($color),
HsvColor::class => $this->importHsvColor($color),
HslColor::class => $this->importHslColor($color),
default => $color,
};
}
/**
* @throws ColorException
*/
protected function importCmykColor(ColorInterface $color): ColorInterface
{
if (!($color instanceof CmykColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
return new Color(
(int) (255 * (1 - $color->cyan()->normalize()) * (1 - $color->key()->normalize())),
(int) (255 * (1 - $color->magenta()->normalize()) * (1 - $color->key()->normalize())),
(int) (255 * (1 - $color->yellow()->normalize()) * (1 - $color->key()->normalize())),
);
}
/**
* @throws ColorException
*/
protected function importHsvColor(ColorInterface $color): ColorInterface
{
if (!($color instanceof HsvColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
$chroma = $color->value()->normalize() * $color->saturation()->normalize();
$hue = $color->hue()->normalize() * 6;
$x = $chroma * (1 - abs(fmod($hue, 2) - 1));
// connect channel values
$values = match (true) {
$hue < 1 => [$chroma, $x, 0],
$hue < 2 => [$x, $chroma, 0],
$hue < 3 => [0, $chroma, $x],
$hue < 4 => [0, $x, $chroma],
$hue < 5 => [$x, 0, $chroma],
default => [$chroma, 0, $x],
};
// add to each value
$values = array_map(fn(float|int $value): float => $value + $color->value()->normalize() - $chroma, $values);
$values[] = 1; // append alpha channel value
return $this->colorFromNormalized($values);
}
/**
* @throws ColorException
*/
protected function importHslColor(ColorInterface $color): ColorInterface
{
if (!($color instanceof HslColor)) {
throw new ColorException('Unabled to import color of type ' . $color::class . '.');
}
// normalized values of hsl channels
[$h, $s, $l] = array_map(
fn(ColorChannelInterface $channel): float => $channel->normalize(),
$color->channels()
);
$c = (1 - abs(2 * $l - 1)) * $s;
$x = $c * (1 - abs(fmod($h * 6, 2) - 1));
$m = $l - $c / 2;
$values = match (true) {
$h < 1 / 6 => [$c, $x, 0],
$h < 2 / 6 => [$x, $c, 0],
$h < 3 / 6 => [0, $c, $x],
$h < 4 / 6 => [0, $x, $c],
$h < 5 / 6 => [$x, 0, $c],
default => [$c, 0, $x],
};
$values = array_map(fn(float|int $value): float => $value + $m, $values);
$values[] = 1; // append alpha channel value
return $this->colorFromNormalized($values);
}
}
================================================
FILE: src/Colors/Rgb/Decoders/HexColorDecoder.php
================================================
[a-f\d]{3}(?:[a-f\d]?|(?:[a-f\d]{3}(?:[a-f\d]{2})?)?)\b)$/i';
if (preg_match($pattern, $input, $matches) != 1) {
throw new DecoderException('Unable to decode input');
}
$values = match (strlen($matches['hex'])) {
3, 4 => str_split($matches['hex']),
6, 8 => str_split($matches['hex'], 2),
default => throw new DecoderException('Unable to decode input'),
};
$values = array_map(function (string $value): float|int {
return match (strlen($value)) {
1 => hexdec($value . $value),
2 => hexdec($value),
default => throw new DecoderException('Unable to decode input'),
};
}, $values);
return new Color(...$values);
}
}
================================================
FILE: src/Colors/Rgb/Decoders/HtmlColornameDecoder.php
================================================
*/
protected static array $names = [
'lightsalmon' => '#ffa07a',
'salmon' => '#fa8072',
'darksalmon' => '#e9967a',
'lightcoral' => '#f08080',
'indianred' => '#cd5c5c',
'crimson' => '#dc143c',
'firebrick' => '#b22222',
'red' => '#ff0000',
'darkred' => '#8b0000',
'coral' => '#ff7f50',
'tomato' => '#ff6347',
'orangered' => '#ff4500',
'gold' => '#ffd700',
'orange' => '#ffa500',
'darkorange' => '#ff8c00',
'lightyellow' => '#ffffe0',
'lemonchiffon' => '#fffacd',
'lightgoldenrodyellow' => '#fafad2',
'papayawhip' => '#ffefd5',
'moccasin' => '#ffe4b5',
'peachpuff' => '#ffdab9',
'palegoldenrod' => '#eee8aa',
'khaki' => '#f0e68c',
'darkkhaki' => '#bdb76b',
'yellow' => '#ffff00',
'lawngreen' => '#7cfc00',
'chartreuse' => '#7fff00',
'limegreen' => '#32cd32',
'lime' => '#00ff00',
'forestgreen' => '#228b22',
'green' => '#008000',
'darkgreen' => '#006400',
'greenyellow' => '#adff2f',
'yellowgreen' => '#9acd32',
'springgreen' => '#00ff7f',
'mediumspringgreen' => '#00fa9a',
'lightgreen' => '#90ee90',
'palegreen' => '#98fb98',
'darkseagreen' => '#8fbc8f',
'mediumseagre' => 'en #3cb371',
'seagreen' => '#2e8b57',
'olive' => '#808000',
'darkolivegreen' => '#556b2f',
'olivedrab' => '#6b8e23',
'lightcyan' => '#e0ffff',
'cyan' => '#00ffff',
'aqua' => '#00ffff',
'aquamarine' => '#7fffd4',
'mediumaquamarine' => '#66cdaa',
'paleturquoise' => '#afeeee',
'turquoise' => '#40e0d0',
'mediumturquoise' => '#48d1cc',
'darkturquoise' => '#00ced1',
'lightseagreen' => '#20b2aa',
'cadetblue' => '#5f9ea0',
'darkcyan' => '#008b8b',
'teal' => '#008080',
'powderblue' => '#b0e0e6',
'lightblue' => '#add8e6',
'lightskyblue' => '#87cefa',
'skyblue' => '#87ceeb',
'deepskyblue' => '#00bfff',
'lightsteelblue' => '#b0c4de',
'dodgerblue' => '#1e90ff',
'cornflowerblue' => '#6495ed',
'steelblue' => '#4682b4',
'royalblue' => '#4169e1',
'blue' => '#0000ff',
'mediumblue' => '#0000cd',
'darkblue' => '#00008b',
'navy' => '#000080',
'midnightblue' => '#191970',
'mediumslateblue' => '#7b68ee',
'slateblue' => '#6a5acd',
'darkslateblue' => '#483d8b',
'lavender' => '#e6e6fa',
'thistle' => '#d8bfd8',
'plum' => '#dda0dd',
'violet' => '#ee82ee',
'orchid' => '#da70d6',
'fuchsia' => '#ff00ff',
'magenta' => '#ff00ff',
'mediumorchid' => '#ba55d3',
'mediumpurple' => '#9370db',
'blueviolet' => '#8a2be2',
'darkviolet' => '#9400d3',
'darkorchid' => '#9932cc',
'darkmagenta' => '#8b008b',
'purple' => '#800080',
'indigo' => '#4b0082',
'pink' => '#ffc0cb',
'lightpink' => '#ffb6c1',
'hotpink' => '#ff69b4',
'deeppink' => '#ff1493',
'palevioletred' => '#db7093',
'mediumvioletred' => '#c71585',
'white' => '#ffffff',
'snow' => '#fffafa',
'honeydew' => '#f0fff0',
'mintcream' => '#f5fffa',
'azure' => '#f0ffff',
'aliceblue' => '#f0f8ff',
'ghostwhite' => '#f8f8ff',
'whitesmoke' => '#f5f5f5',
'seashell' => '#fff5ee',
'beige' => '#f5f5dc',
'oldlace' => '#fdf5e6',
'floralwhite' => '#fffaf0',
'ivory' => '#fffff0',
'antiquewhite' => '#faebd7',
'linen' => '#faf0e6',
'lavenderblush' => '#fff0f5',
'mistyrose' => '#ffe4e1',
'gainsboro' => '#dcdcdc',
'lightgray' => '#d3d3d3',
'silver' => '#c0c0c0',
'darkgray' => '#a9a9a9',
'gray' => '#808080',
'dimgray' => '#696969',
'lightslategray' => '#778899',
'slategray' => '#708090',
'darkslategray' => '#2f4f4f',
'black' => '#000000',
'cornsilk' => '#fff8dc',
'blanchedalmond' => '#ffebcd',
'bisque' => '#ffe4c4',
'navajowhite' => '#ffdead',
'wheat' => '#f5deb3',
'burlywood' => '#deb887',
'tan' => '#d2b48c',
'rosybrown' => '#bc8f8f',
'sandybrown' => '#f4a460',
'goldenrod' => '#daa520',
'peru' => '#cd853f',
'chocolate' => '#d2691e',
'saddlebrown' => '#8b4513',
'sienna' => '#a0522d',
'brown' => '#a52a2a',
'maroon' => '#800000',
];
/**
* Decode html color names
*/
public function decode(mixed $input): ImageInterface|ColorInterface
{
if (!is_string($input)) {
throw new DecoderException('Unable to decode input');
}
if (!array_key_exists(strtolower($input), static::$names)) {
throw new DecoderException('Unable to decode input');
}
return parent::decode(static::$names[strtolower($input)]);
}
}
================================================
FILE: src/Colors/Rgb/Decoders/StringColorDecoder.php
================================================
[0-9\.]+%?), ?(?P[0-9\.]+%?), ?(?P[0-9\.]+%?)' .
'(?:, ?(?P(?:1)|(?:1\.0*)|(?:0)|(?:0?\.\d+%?)|(?:\d{1,3}%)))?\)$/i';
if (preg_match($pattern, $input, $matches) != 1) {
throw new DecoderException('Unable to decode input');
}
// rgb values
$values = array_map(function (string $value): int {
return match (strpos($value, '%')) {
false => intval(trim($value)),
default => intval(round(floatval(trim(str_replace('%', '', $value))) / 100 * 255)),
};
}, [$matches['r'], $matches['g'], $matches['b']]);
// alpha value
if (array_key_exists('a', $matches)) {
$values[] = match (true) {
strpos($matches['a'], '%') => round(intval(trim(str_replace('%', '', $matches['a']))) / 2.55),
default => intval(round(floatval(trim($matches['a'])) * 255)),
};
}
return new Color(...$values);
}
}
================================================
FILE: src/Colors/Rgb/Decoders/TransparentColorDecoder.php
================================================
prepareOptions($options) as $name => $value) {
if (!property_exists($this, $name)) {
throw new InputException('Property ' . $name . ' does not exists for ' . $this::class . '.');
}
$this->{$name} = $value;
}
return $this;
}
/**
* This method makes it possible to call self::setOptions() with a single
* array instead of named parameters
*
* @param array $options
* @return array
*/
private function prepareOptions(array $options): array
{
if ($options === []) {
return $options;
}
if (count($options) > 1) {
return $options;
}
if (!array_key_exists(0, $options)) {
return $options;
}
if (!is_array($options[0])) {
return $options;
}
return $options[0];
}
}
================================================
FILE: src/Decoders/Base64ImageDecoder.php
================================================
PHP_MAXPATHLEN) {
return false;
}
try {
if (!@is_file($input)) {
return false;
}
} catch (Exception) {
return false;
}
return true;
}
/**
* Extract and return EXIF data from given input which can be binary image
* data or a file path.
*
* @return CollectionInterface
*/
protected function extractExifData(string $path_or_data): CollectionInterface
{
if (!function_exists('exif_read_data')) {
return new Collection();
}
try {
$source = match (true) {
$this->isFile($path_or_data) => $path_or_data, // path
default => $this->buildFilePointer($path_or_data), // data
};
// extract exif data
$data = @exif_read_data($source, null, true);
if (is_resource($source)) {
fclose($source);
}
} catch (Exception) {
$data = [];
}
return new Collection(is_array($data) ? $data : []);
}
/**
* Determine if given input is base64 encoded data
*/
protected function isValidBase64(mixed $input): bool
{
if (!is_string($input)) {
return false;
}
return base64_encode(base64_decode($input)) === str_replace(["\n", "\r"], '', $input);
}
/**
* Parse data uri
*/
protected function parseDataUri(mixed $input): object
{
$pattern = "/^data:(?P\w+\/[-+.\w]+)?" .
"(?P(;[-\w]+=[-\w]+)*)(?P;base64)?,(?P.*)/";
$result = preg_match($pattern, (string) $input, $matches);
return new class ($matches, $result)
{
/**
* @param array $matches
* @return void
*/
public function __construct(private array $matches, private int|false $result)
{
//
}
public function isValid(): bool
{
return (bool) $this->result;
}
public function mediaType(): ?string
{
if (isset($this->matches['mediatype']) && !empty($this->matches['mediatype'])) {
return $this->matches['mediatype'];
}
return null;
}
public function hasMediaType(): bool
{
return !empty($this->mediaType());
}
public function isBase64Encoded(): bool
{
return isset($this->matches['base64']) && $this->matches['base64'] === ';base64';
}
public function data(): ?string
{
if (isset($this->matches['data']) && !empty($this->matches['data'])) {
return $this->matches['data'];
}
return null;
}
};
}
}
================================================
FILE: src/Drivers/AbstractDriver.php
================================================
config = new Config();
$this->checkHealth();
}
/**
* {@inheritdoc}
*
* @see DriverInterface::config()
*/
public function config(): Config
{
return $this->config;
}
/**
* {@inheritdoc}
*
* @see DriverInterface::handleInput()
*/
public function handleInput(mixed $input, array $decoders = []): ImageInterface|ColorInterface
{
return InputHandler::withDecoders($decoders, $this)->handle($input);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::specialize()
*/
public function specialize(
ModifierInterface|AnalyzerInterface|EncoderInterface|DecoderInterface $object
): ModifierInterface|AnalyzerInterface|EncoderInterface|DecoderInterface {
// return object directly if no specializing is possible
if (!($object instanceof SpecializableInterface)) {
return $object;
}
// return directly and only attach driver if object is already specialized
if ($object instanceof SpecializedInterface) {
$object->setDriver($this);
return $object;
}
// resolve classname for specializable object
$specialized_classname = implode("\\", [
(new ReflectionClass($this))->getNamespaceName(), // driver's namespace
match (true) {
$object instanceof ModifierInterface => 'Modifiers',
$object instanceof AnalyzerInterface => 'Analyzers',
$object instanceof EncoderInterface => 'Encoders',
$object instanceof DecoderInterface => 'Decoders',
},
$object_shortname = (new ReflectionClass($object))->getShortName(),
]);
// fail if driver specialized classname does not exists
if (!class_exists($specialized_classname)) {
throw new NotSupportedException(
"Class '" . $object_shortname . "' is not supported by " . $this->id() . " driver."
);
}
// create a driver specialized object with the specializable properties of generic object
$specialized = new $specialized_classname(...$object->specializable());
// attach driver
return $specialized->setDriver($this);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::specializeMultiple()
*
* @throws NotSupportedException
* @throws DriverException
*/
public function specializeMultiple(array $objects): array
{
return array_map(
function (string|object $object): ModifierInterface|AnalyzerInterface|EncoderInterface|DecoderInterface {
return $this->specialize(
match (true) {
is_string($object) => new $object(),
is_object($object) => $object,
}
);
},
$objects
);
}
}
================================================
FILE: src/Drivers/AbstractEncoder.php
================================================
encode($this);
}
/**
* Build new file pointer, run callback with it and return result as encoded image
*
* @throws RuntimeException
*/
protected function createEncodedImage(callable $callback, ?string $mediaType = null): EncodedImage
{
$pointer = $this->buildFilePointer();
$callback($pointer);
return is_string($mediaType) ? new EncodedImage($pointer, $mediaType) : new EncodedImage($pointer);
}
}
================================================
FILE: src/Drivers/AbstractFontProcessor.php
================================================
wrapTextBlock(new TextBlock($text), $font);
$pivot = $this->buildPivot($lines, $font, $position);
$leading = $this->leading($font);
$blockWidth = $this->boxSize((string) $lines->longestLine(), $font)->width();
$x = $pivot->x();
$y = $font->hasFilename() ? $pivot->y() + $this->capHeight($font) : $pivot->y();
$xAdjustment = 0;
// adjust line positions according to alignment
foreach ($lines as $line) {
$lineBoxSize = $this->boxSize((string) $line, $font);
$lineWidth = $lineBoxSize->width() + $lineBoxSize->pivot()->x();
$xAdjustment = $font->alignment() === 'left' ? 0 : $blockWidth - $lineWidth;
$xAdjustment = $font->alignment() === 'right' ? intval(round($xAdjustment)) : $xAdjustment;
$xAdjustment = $font->alignment() === 'center' ? intval(round($xAdjustment / 2)) : $xAdjustment;
$position = new Point($x + $xAdjustment, $y);
$position->rotate($font->angle(), $pivot);
$line->setPosition($position);
$y += $leading;
}
return $lines;
}
/**
* {@inheritdoc}
*
* @see FontProcessorInterface::nativeFontSize()
*/
public function nativeFontSize(FontInterface $font): float
{
return $font->size();
}
/**
* {@inheritdoc}
*
* @see FontProcessorInterface::typographicalSize()
*/
public function typographicalSize(FontInterface $font): int
{
return $this->boxSize('Hy', $font)->height();
}
/**
* {@inheritdoc}
*
* @see FontProcessorInterface::capHeight()
*/
public function capHeight(FontInterface $font): int
{
return $this->boxSize('T', $font)->height();
}
/**
* {@inheritdoc}
*
* @see FontProcessorInterface::leading()
*/
public function leading(FontInterface $font): int
{
return intval(round($this->typographicalSize($font) * $font->lineHeight()));
}
/**
* Reformat a text block by wrapping each line before the given maximum width
*
* @throws FontException
*/
protected function wrapTextBlock(TextBlock $block, FontInterface $font): TextBlock
{
$newLines = [];
foreach ($block as $line) {
foreach ($this->wrapLine($line, $font) as $newLine) {
$newLines[] = $newLine;
}
}
return $block->setLines($newLines);
}
/**
* Check if a line exceeds the given maximum width and wrap it if necessary.
* The output will be an array of formatted lines that are all within the
* maximum width.
*
* @throws FontException
* @return array
*/
protected function wrapLine(Line $line, FontInterface $font): array
{
// no wrap width - no wrapping
if (is_null($font->wrapWidth())) {
return [$line];
}
$wrapped = [];
$formattedLine = new Line();
foreach ($line as $word) {
// calculate width of newly formatted line
$lineWidth = $this->boxSize(match ($formattedLine->count()) {
0 => $word,
default => $formattedLine . ' ' . $word,
}, $font)->width();
// decide if word fits on current line or a new line must be created
if ($line->count() === 1 || $lineWidth <= $font->wrapWidth()) {
$formattedLine->add($word);
} else {
if ($formattedLine->count() !== 0) {
$wrapped[] = $formattedLine;
}
$formattedLine = new Line($word);
}
}
$wrapped[] = $formattedLine;
return $wrapped;
}
/**
* Build pivot point of textblock according to the font settings and based on given position
*
* @throws FontException
*/
protected function buildPivot(TextBlock $block, FontInterface $font, PointInterface $position): PointInterface
{
// bounding box
$box = new Rectangle(
$this->boxSize((string) $block->longestLine(), $font)->width(),
$this->leading($font) * ($block->count() - 1) + $this->capHeight($font)
);
// set position
$box->setPivot($position);
// alignment
$box->align($font->alignment());
$box->valign($font->valignment());
$box->rotate($font->angle());
return $box->last();
}
}
================================================
FILE: src/Drivers/AbstractFrame.php
================================================
*/
public function __debugInfo(): array
{
return [
'delay' => $this->delay(),
'left' => $this->offsetLeft(),
'top' => $this->offsetTop(),
'dispose' => $this->dispose(),
];
}
}
================================================
FILE: src/Drivers/Gd/Analyzers/ColorspaceAnalyzer.php
================================================
core()->native());
}
}
================================================
FILE: src/Drivers/Gd/Analyzers/PixelColorAnalyzer.php
================================================
colorAt(
$image->colorspace(),
$image->core()->frame($this->frame_key)->native()
);
}
/**
* @throws GeometryException
* @throws ColorException
*/
protected function colorAt(ColorspaceInterface $colorspace, GdImage $gd): ColorInterface
{
$index = @imagecolorat($gd, $this->x, $this->y);
if (!imageistruecolor($gd)) {
$index = imagecolorsforindex($gd, $index);
}
if ($index === false) {
throw new GeometryException(
'The specified position is not in the valid image area.'
);
}
return $this->driver()->colorProcessor($colorspace)->nativeToColor($index);
}
}
================================================
FILE: src/Drivers/Gd/Analyzers/PixelColorsAnalyzer.php
================================================
colorspace();
foreach ($image as $frame) {
$colors->push(
parent::colorAt($colorspace, $frame->native())
);
}
return $colors;
}
}
================================================
FILE: src/Drivers/Gd/Analyzers/ResolutionAnalyzer.php
================================================
core()->native()));
}
}
================================================
FILE: src/Drivers/Gd/Analyzers/WidthAnalyzer.php
================================================
core()->native());
}
}
================================================
FILE: src/Drivers/Gd/Cloner.php
================================================
width(), $size->height());
// copy resolution to clone
$resolution = imageresolution($gd);
if (is_array($resolution) && array_key_exists(0, $resolution) && array_key_exists(1, $resolution)) {
imageresolution($clone, $resolution[0], $resolution[1]);
}
// fill with background
$processor = new ColorProcessor();
imagefill($clone, 0, 0, $processor->colorToNative($background));
imagealphablending($clone, true);
imagesavealpha($clone, true);
// set background image as transparent if alpha channel value if color is below .5
// comes into effect when the end format only supports binary transparency (like GIF)
if ($background->channel(Alpha::class)->value() < 128) {
imagecolortransparent($clone, $processor->colorToNative($background));
}
return $clone;
}
/**
* Create a clone of an GdImage that is positioned on the specified background color.
* Possible transparent areas are mixed with this color.
*
* @throws ColorException
*/
public static function cloneBlended(GdImage $gd, ColorInterface $background): GdImage
{
// create empty canvas with same size
$clone = static::cloneEmpty($gd, background: $background);
// transfer actual image to clone
imagecopy($clone, $gd, 0, 0, 0, 0, imagesx($gd), imagesy($gd));
return $clone;
}
}
================================================
FILE: src/Drivers/Gd/ColorProcessor.php
================================================
convertTo($this->colorspace);
// gd only supports rgb so the channels can be accessed directly
$r = $color->channel(Red::class)->value();
$g = $color->channel(Green::class)->value();
$b = $color->channel(Blue::class)->value();
$a = $color->channel(Alpha::class)->value();
// convert alpha value to gd alpha
// ([opaque]255-0[transparent]) to ([opaque]0-127[transparent])
$a = (int) $this->convertRange($a, 0, 255, 127, 0);
return ($a << 24) + ($r << 16) + ($g << 8) + $b;
}
/**
* {@inheritdoc}
*
* @see ColorProcessorInterface::nativeToColor()
*/
public function nativeToColor(mixed $value): ColorInterface
{
if (!is_int($value) && !is_array($value)) {
throw new ColorException('GD driver can only decode colors in integer and array format.');
}
if (is_array($value)) {
// array conversion
if (!$this->isValidArrayColor($value)) {
throw new ColorException(
'GD driver can only decode array color format array{red: int, green: int, blue: int, alpha: int}.',
);
}
$r = $value['red'];
$g = $value['green'];
$b = $value['blue'];
$a = $value['alpha'];
} else {
// integer conversion
$a = ($value >> 24) & 0xFF;
$r = ($value >> 16) & 0xFF;
$g = ($value >> 8) & 0xFF;
$b = $value & 0xFF;
}
// convert gd apha integer to intervention alpha integer
// ([opaque]0-127[transparent]) to ([opaque]255-0[transparent])
$a = (int) static::convertRange($a, 127, 0, 0, 255);
return new Color($r, $g, $b, $a);
}
/**
* Convert input in range (min) to (max) to the corresponding value
* in target range (targetMin) to (targetMax).
*/
protected function convertRange(
float|int $input,
float|int $min,
float|int $max,
float|int $targetMin,
float|int $targetMax
): float|int {
return ceil(((($input - $min) * ($targetMax - $targetMin)) / ($max - $min)) + $targetMin);
}
/**
* Check if given array is valid color format
* array{red: int, green: int, blue: int, alpha: int}
* i.e. result of imagecolorsforindex()
*
* @param array $color
*/
private function isValidArrayColor(array $color): bool
{
if (!array_key_exists('red', $color)) {
return false;
}
if (!array_key_exists('green', $color)) {
return false;
}
if (!array_key_exists('blue', $color)) {
return false;
}
if (!array_key_exists('alpha', $color)) {
return false;
}
if (!is_int($color['red'])) {
return false;
}
if (!is_int($color['green'])) {
return false;
}
if (!is_int($color['blue'])) {
return false;
}
if (!is_int($color['alpha'])) {
return false;
}
return true;
}
}
================================================
FILE: src/Drivers/Gd/Core.php
================================================
push($frame);
return $this;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::native()
*/
public function native(): mixed
{
return $this->first()->native();
}
/**
* {@inheritdoc}
*
* @see CoreInterface::setNative()
*/
public function setNative(mixed $native): self
{
$this->empty()->push(new Frame($native));
return $this;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::frame()
*/
public function frame(int $position): FrameInterface
{
$frame = $this->getAtPosition($position);
if (!($frame instanceof FrameInterface)) {
throw new AnimationException('Frame #' . $position . ' could not be found in the image.');
}
return $frame;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::loops()
*/
public function loops(): int
{
return $this->loops;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::setLoops()
*/
public function setLoops(int $loops): self
{
$this->loops = $loops;
return $this;
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::first()
*/
public function first(): FrameInterface
{
return parent::first();
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::last()
*/
public function last(): FrameInterface
{
return parent::last();
}
/**
* Clone instance
*/
public function __clone(): void
{
foreach ($this->items as $key => $frame) {
$this->items[$key] = clone $frame;
}
}
}
================================================
FILE: src/Drivers/Gd/Decoders/AbstractDecoder.php
================================================
isValidBase64($input)) {
throw new DecoderException('Unable to decode input');
}
return parent::decode(base64_decode((string) $input));
}
}
================================================
FILE: src/Drivers/Gd/Decoders/BinaryImageDecoder.php
================================================
isGifFormat($input)) {
true => $this->decodeGif($input),
default => $this->decodeBinary($input),
};
}
/**
* Decode image from given binary data
*
* @throws RuntimeException
*/
private function decodeBinary(string $input): ImageInterface
{
$gd = @imagecreatefromstring($input);
if ($gd === false) {
throw new DecoderException('Unable to decode input');
}
// create image instance
$image = parent::decode($gd);
// get media type
$mediaType = $this->getMediaTypeByBinary($input);
// extract & set exif data for appropriate formats
if (in_array($mediaType->format(), [Format::JPEG, Format::TIFF])) {
$image->setExif($this->extractExifData($input));
}
// set mediaType on origin
$image->origin()->setMediaType($mediaType);
// adjust image orientation
if ($this->driver()->config()->autoOrientation) {
$image->modify(new AlignRotationModifier());
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Decoders/DataUriImageDecoder.php
================================================
parseDataUri($input);
if (!$uri->isValid()) {
throw new DecoderException('Unable to decode input');
}
if ($uri->isBase64Encoded()) {
return parent::decode(base64_decode($uri->data()));
}
return parent::decode(urldecode($uri->data()));
}
}
================================================
FILE: src/Drivers/Gd/Decoders/EncodedImageObjectDecoder.php
================================================
toString());
}
}
================================================
FILE: src/Drivers/Gd/Decoders/FilePathImageDecoder.php
================================================
isFile($input)) {
throw new DecoderException('Unable to decode input');
}
// detect media (mime) type
$mediaType = $this->getMediaTypeByFilePath($input);
$image = match ($mediaType->format()) {
// gif files might be animated and therefore cannot
// be handled by the standard GD decoder.
Format::GIF => $this->decodeGif($input),
default => parent::decode(match ($mediaType->format()) {
Format::JPEG => @imagecreatefromjpeg($input),
Format::WEBP => @imagecreatefromwebp($input),
Format::PNG => @imagecreatefrompng($input),
Format::AVIF => @imagecreatefromavif($input),
Format::BMP => @imagecreatefrombmp($input),
default => throw new DecoderException('Unable to decode input'),
}),
};
// set file path & mediaType on origin
$image->origin()->setFilePath($input);
$image->origin()->setMediaType($mediaType);
// extract exif for the appropriate formats
if ($mediaType->format() === Format::JPEG) {
$image->setExif($this->extractExifData($input));
}
// adjust image orientation
if ($this->driver()->config()->autoOrientation) {
$image->modify(new AlignRotationModifier());
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Decoders/FilePointerImageDecoder.php
================================================
driver(),
new Core([
new Frame($input)
])
);
}
/**
* Decode image from given GIF source which can be either a file path or binary data
*
* Depending on the configuration, this is taken over by the native GD function
* or, if animations are required, by our own extended decoder.
*
* @throws RuntimeException
*/
protected function decodeGif(mixed $input): ImageInterface
{
// create non-animated image depending on config
if (!$this->driver()->config()->decodeAnimation) {
$native = match (true) {
$this->isGifFormat($input) => @imagecreatefromstring($input),
default => @imagecreatefromgif($input),
};
if ($native === false) {
throw new DecoderException('Unable to decode input.');
}
$image = self::decode($native);
$image->origin()->setMediaType('image/gif');
return $image;
}
try {
// create empty core
$core = new Core();
$gif = GifDecoder::decode($input);
$splitter = GifSplitter::create($gif)->split();
$delays = $splitter->getDelays();
// set loops on core
if ($loops = $gif->getMainApplicationExtension()?->getLoops()) {
$core->setLoops($loops);
}
// add GDImage instances to core
foreach ($splitter->coalesceToResources() as $key => $native) {
$core->push(
new Frame($native, $delays[$key] / 100)
);
}
} catch (Exception $e) {
throw new DecoderException($e->getMessage(), $e->getCode(), $e);
}
// create (possibly) animated image
$image = new Image($this->driver(), $core);
// set media type
$image->origin()->setMediaType('image/gif');
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Decoders/SplFileInfoImageDecoder.php
================================================
getRealPath());
}
}
================================================
FILE: src/Drivers/Gd/Driver.php
================================================
core->add(
$this->driver->handleInput($source)->core()->first()->setDelay($delay)
);
return $this;
}
/**
* @throws RuntimeException
*/
public function __invoke(): ImageInterface
{
return new Image(
$this->driver,
$this->core
);
}
};
$init($animation);
return call_user_func($animation);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::colorProcessor()
*/
public function colorProcessor(ColorspaceInterface $colorspace): ColorProcessorInterface
{
return new ColorProcessor($colorspace);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::fontProcessor()
*/
public function fontProcessor(): FontProcessorInterface
{
return new FontProcessor();
}
/**
* {@inheritdoc}
*
* @see DriverInterface::supports()
*/
public function supports(string|Format|FileExtension|MediaType $identifier): bool
{
return match (Format::tryCreate($identifier)) {
Format::JPEG => boolval(imagetypes() & IMG_JPEG),
Format::WEBP => boolval(imagetypes() & IMG_WEBP),
Format::GIF => boolval(imagetypes() & IMG_GIF),
Format::PNG => boolval(imagetypes() & IMG_PNG),
Format::AVIF => boolval(imagetypes() & IMG_AVIF),
Format::BMP => boolval(imagetypes() & IMG_BMP),
default => false,
};
}
/**
* Return version of GD library
*/
public static function version(): string
{
return gd_info()['GD Version'];
}
}
================================================
FILE: src/Drivers/Gd/Encoders/AvifEncoder.php
================================================
createEncodedImage(function ($pointer) use ($image): void {
imageavif($image->core()->native(), $pointer, $this->quality);
}, 'image/avif');
}
}
================================================
FILE: src/Drivers/Gd/Encoders/BmpEncoder.php
================================================
createEncodedImage(function ($pointer) use ($image): void {
imagebmp($image->core()->native(), $pointer, false);
}, 'image/bmp');
}
}
================================================
FILE: src/Drivers/Gd/Encoders/GifEncoder.php
================================================
isAnimated()) {
return $this->encodeAnimated($image);
}
$gd = Cloner::clone($image->core()->native());
return $this->createEncodedImage(function ($pointer) use ($gd): void {
imageinterlace($gd, $this->interlaced);
imagegif($gd, $pointer);
}, 'image/gif');
}
/**
* @throws RuntimeException
*/
protected function encodeAnimated(ImageInterface $image): EncodedImage
{
try {
$builder = GifBuilder::canvas(
$image->width(),
$image->height()
);
foreach ($image as $frame) {
$builder->addFrame(
source: $this->encode($frame->toImage($image->driver()))->toFilePointer(),
delay: $frame->delay(),
interlaced: $this->interlaced
);
}
$builder->setLoops($image->loops());
return new EncodedImage($builder->encode(), 'image/gif');
} catch (Exception $e) {
throw new EncoderException($e->getMessage(), $e->getCode(), $e);
}
}
}
================================================
FILE: src/Drivers/Gd/Encoders/JpegEncoder.php
================================================
driver()->handleInput(
$this->driver()->config()->blendingColor
);
$output = Cloner::cloneBlended(
$image->core()->native(),
background: $blendingColor
);
return $this->createEncodedImage(function ($pointer) use ($output): void {
imageinterlace($output, $this->progressive);
imagejpeg($output, $pointer, $this->quality);
}, 'image/jpeg');
}
}
================================================
FILE: src/Drivers/Gd/Encoders/PngEncoder.php
================================================
prepareOutput($image);
return $this->createEncodedImage(function ($pointer) use ($output): void {
imageinterlace($output, $this->interlaced);
imagepng($output, $pointer, -1);
}, 'image/png');
}
/**
* Prepare given image instance for PNG format output according to encoder settings
*
* @throws RuntimeException
* @throws ColorException
* @throws AnimationException
*/
private function prepareOutput(ImageInterface $image): GdImage
{
if ($this->indexed) {
$output = clone $image;
$output->reduceColors(255);
return $output->core()->native();
}
return Cloner::clone($image->core()->native());
}
}
================================================
FILE: src/Drivers/Gd/Encoders/WebpEncoder.php
================================================
quality === 100 && defined('IMG_WEBP_LOSSLESS') ? IMG_WEBP_LOSSLESS : $this->quality;
return $this->createEncodedImage(function ($pointer) use ($image, $quality): void {
imagewebp($image->core()->native(), $pointer, $quality);
}, 'image/webp');
}
}
================================================
FILE: src/Drivers/Gd/FontProcessor.php
================================================
hasFilename()) {
// calculate box size from gd font
$box = new Rectangle(0, 0);
$chars = mb_strlen($text);
if ($chars > 0) {
$box->setWidth(
$chars * $this->gdCharacterWidth((int) $font->filename())
);
$box->setHeight(
$this->gdCharacterHeight((int) $font->filename())
);
}
return $box;
}
// build full path to font file to make sure to pass absolute path to imageftbbox()
// because of issues with different GD version behaving differently when passing
// relative paths to imageftbbox()
$fontPath = realpath($font->filename());
if ($fontPath === false) {
throw new FontException('Font file ' . $font->filename() . ' does not exist.');
}
// calculate box size from ttf font file with angle 0
$box = imageftbbox(
size: $this->nativeFontSize($font),
angle: 0,
font_filename: $fontPath,
string: $text,
);
if ($box === false) {
throw new FontException('Unable to calculate box size of font ' . $font->filename() . '.');
}
// build size from points
return new Rectangle(
width: intval(abs($box[6] - $box[4])), // difference of upper-left-x and upper-right-x
height: intval(abs($box[7] - $box[1])), // difference if upper-left-y and lower-left-y
pivot: new Point($box[6], $box[7]), // position of upper-left corner
);
}
/**
* {@inheritdoc}
*
* @see FontProcessorInterface::nativeFontSize()
*/
public function nativeFontSize(FontInterface $font): float
{
return floatval(round($font->size() * .76, 6));
}
/**
* {@inheritdoc}
*
* @see FontProcessorInterface::leading()
*/
public function leading(FontInterface $font): int
{
return (int) round(parent::leading($font) * .8);
}
/**
* Return width of a single character
*/
protected function gdCharacterWidth(int $gdfont): int
{
return $gdfont + 4;
}
/**
* Return height of a single character
*/
protected function gdCharacterHeight(int $gdfont): int
{
return match ($gdfont) {
2, 3 => 14,
4, 5 => 16,
default => 8,
};
}
}
================================================
FILE: src/Drivers/Gd/Frame.php
================================================
native = $native;
return $this;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::native()
*/
public function native(): GdImage
{
return $this->native;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::size()
*/
public function size(): SizeInterface
{
return new Rectangle(imagesx($this->native), imagesy($this->native));
}
/**
* {@inheritdoc}
*
* @see FrameInterface::delay()
*/
public function delay(): float
{
return $this->delay;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::setDelay()
*/
public function setDelay(float $delay): FrameInterface
{
$this->delay = $delay;
return $this;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::dispose()
*/
public function dispose(): int
{
return $this->dispose;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::setDispose()
*
* @throws InputException
*/
public function setDispose(int $dispose): FrameInterface
{
if (!in_array($dispose, [0, 1, 2, 3])) {
throw new InputException('Value for argument $dispose must be 0, 1, 2 or 3.');
}
$this->dispose = $dispose;
return $this;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::setOffset()
*/
public function setOffset(int $left, int $top): FrameInterface
{
$this->offset_left = $left;
$this->offset_top = $top;
return $this;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::offsetLeft()
*/
public function offsetLeft(): int
{
return $this->offset_left;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::setOffsetLeft()
*/
public function setOffsetLeft(int $offset): FrameInterface
{
$this->offset_left = $offset;
return $this;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::offsetTop()
*/
public function offsetTop(): int
{
return $this->offset_top;
}
/**
* {@inheritdoc}
*
* @see FrameInterface::setOffsetTop()
*/
public function setOffsetTop(int $offset): FrameInterface
{
$this->offset_top = $offset;
return $this;
}
/**
* This workaround helps cloning GdImages which is currently not possible.
*
* @throws ColorException
*/
public function __clone(): void
{
$this->native = Cloner::clone($this->native);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/AlignRotationModifier.php
================================================
exif('IFD0.Orientation')) {
2 => $image->flop(),
3 => $image->rotate(180),
4 => $image->rotate(180)->flop(),
5 => $image->rotate(270)->flop(),
6 => $image->rotate(270),
7 => $image->rotate(90)->flop(),
8 => $image->rotate(90),
default => $image
};
return $this->markAligned($image);
}
/**
* Set exif data of image to top-left orientation, marking the image as
* aligned and making sure the rotation correction process is not
* performed again.
*/
private function markAligned(ImageInterface $image): ImageInterface
{
$exif = $image->exif()->map(function ($item) {
if (is_array($item) && array_key_exists('Orientation', $item)) {
$item['Orientation'] = 1;
return $item;
}
return $item;
});
return $image->setExif($exif);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/BlendTransparencyModifier.php
================================================
blendingColor($this->driver());
foreach ($image as $frame) {
// create new canvas with blending color as background
$modified = Cloner::cloneBlended(
$frame->native(),
background: $blendingColor
);
// set new gd image
$frame->setNative($modified);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/BlurModifier.php
================================================
amount; $i++) {
imagefilter($frame->native(), IMG_FILTER_GAUSSIAN_BLUR);
}
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/BrightnessModifier.php
================================================
native(), IMG_FILTER_BRIGHTNESS, intval($this->level * 2.55));
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ColorizeModifier.php
================================================
red * 2.55);
$green = (int) round($this->green * 2.55);
$blue = (int) round($this->blue * 2.55);
foreach ($image as $frame) {
imagefilter($frame->native(), IMG_FILTER_COLORIZE, $red, $green, $blue);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ColorspaceModifier.php
================================================
targetColorspace() instanceof RgbColorspace)) {
throw new NotSupportedException(
'Only RGB colorspace is supported by GD driver.'
);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ContainModifier.php
================================================
getCropSize($image);
$resize = $this->getResizeSize($image);
$background = $this->driver()->handleInput($this->background);
$blendingColor = $this->driver()->handleInput(
$this->driver()->config()->blendingColor
);
foreach ($image as $frame) {
$this->modify($frame, $crop, $resize, $background, $blendingColor);
}
return $image;
}
/**
* @throws ColorException
*/
protected function modify(
FrameInterface $frame,
SizeInterface $crop,
SizeInterface $resize,
ColorInterface $background,
ColorInterface $blendingColor
): void {
// create new gd image
$modified = Cloner::cloneEmpty($frame->native(), $resize, $background);
// make image area transparent to keep transparency
// even if background-color is set
$transparent = imagecolorallocatealpha(
$modified,
$blendingColor->channel(Red::class)->value(),
$blendingColor->channel(Green::class)->value(),
$blendingColor->channel(Blue::class)->value(),
127,
);
imagealphablending($modified, false); // do not blend / just overwrite
imagecolortransparent($modified, $transparent);
imagefilledrectangle(
$modified,
$crop->pivot()->x(),
$crop->pivot()->y(),
$crop->pivot()->x() + $crop->width() - 1,
$crop->pivot()->y() + $crop->height() - 1,
$transparent
);
// copy image from original with blending alpha
imagealphablending($modified, true);
imagecopyresampled(
$modified,
$frame->native(),
$crop->pivot()->x(),
$crop->pivot()->y(),
0,
0,
$crop->width(),
$crop->height(),
$frame->size()->width(),
$frame->size()->height()
);
// set new content as resource
$frame->setNative($modified);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ContrastModifier.php
================================================
native(), IMG_FILTER_CONTRAST, ($this->level * -1));
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/CoverDownModifier.php
================================================
resizeDown($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/CoverModifier.php
================================================
getCropSize($image);
$resize = $this->getResizeSize($crop);
foreach ($image as $frame) {
$this->modifyFrame($frame, $crop, $resize);
}
return $image;
}
/**
* @throws ColorException
*/
protected function modifyFrame(FrameInterface $frame, SizeInterface $crop, SizeInterface $resize): void
{
// create new image
$modified = Cloner::cloneEmpty($frame->native(), $resize);
// copy content from resource
imagecopyresampled(
$modified,
$frame->native(),
0,
0,
$crop->pivot()->x(),
$crop->pivot()->y(),
$resize->width(),
$resize->height(),
$crop->width(),
$crop->height()
);
// set new content as resource
$frame->setNative($modified);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/CropModifier.php
================================================
size();
$crop = $this->crop($image);
$background = $this->driver()->handleInput($this->background);
foreach ($image as $frame) {
$this->cropFrame($frame, $originalSize, $crop, $background);
}
return $image;
}
/**
* @throws ColorException
*/
protected function cropFrame(
FrameInterface $frame,
SizeInterface $originalSize,
SizeInterface $resizeTo,
ColorInterface $background
): void {
// create new image with transparent background
$modified = Cloner::cloneEmpty($frame->native(), $resizeTo, $background);
// define offset
$offset_x = $resizeTo->pivot()->x() + $this->offset_x;
$offset_y = $resizeTo->pivot()->y() + $this->offset_y;
// define target width & height
$targetWidth = min($resizeTo->width(), $originalSize->width());
$targetHeight = min($resizeTo->height(), $originalSize->height());
$targetWidth = $targetWidth < $originalSize->width() ? $targetWidth + $offset_x : $targetWidth;
$targetHeight = $targetHeight < $originalSize->height() ? $targetHeight + $offset_y : $targetHeight;
// don't alpha blend for copy operation to keep transparent areas of original image
imagealphablending($modified, false);
// copy content from resource
imagecopyresampled(
$modified,
$frame->native(),
$offset_x * -1,
$offset_y * -1,
0,
0,
$targetWidth,
$targetHeight,
$targetWidth,
$targetHeight
);
// set new content as resource
$frame->setNative($modified);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/DrawBezierModifier.php
================================================
drawable->count() !== 3 && $this->drawable->count() !== 4) {
throw new GeometryException('You must specify either 3 or 4 points to create a bezier curve');
}
[$polygon, $polygon_border_segments] = $this->calculateBezierPoints();
if ($this->drawable->hasBackgroundColor() || $this->drawable->hasBorder()) {
imagealphablending($frame->native(), true);
imageantialias($frame->native(), true);
}
if ($this->drawable->hasBackgroundColor()) {
$background_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
);
imagesetthickness($frame->native(), 0);
imagefilledpolygon(
$frame->native(),
$polygon,
$background_color
);
}
if ($this->drawable->hasBorder() && $this->drawable->borderSize() > 0) {
$border_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
);
if ($this->drawable->borderSize() === 1) {
imagesetthickness($frame->native(), $this->drawable->borderSize());
$count = count($polygon);
for ($i = 0; $i < $count; $i += 2) {
if (array_key_exists($i + 2, $polygon) && array_key_exists($i + 3, $polygon)) {
imageline(
$frame->native(),
$polygon[$i],
$polygon[$i + 1],
$polygon[$i + 2],
$polygon[$i + 3],
$border_color
);
}
}
} else {
$polygon_border_segments_total = count($polygon_border_segments);
for ($i = 0; $i < $polygon_border_segments_total; $i += 1) {
imagefilledpolygon(
$frame->native(),
$polygon_border_segments[$i],
$border_color
);
}
}
}
}
return $image;
}
/**
* Calculate interpolation points for quadratic beziers using the Bernstein polynomial form
*
* @return array{'x': float, 'y': float}
*/
private function calculateQuadraticBezierInterpolationPoint(float $t = 0.05): array
{
$remainder = 1 - $t;
$control_point_1_multiplier = $remainder * $remainder;
$control_point_2_multiplier = $remainder * $t * 2;
$control_point_3_multiplier = $t * $t;
$x = (
$this->drawable->first()->x() * $control_point_1_multiplier +
$this->drawable->second()->x() * $control_point_2_multiplier +
$this->drawable->last()->x() * $control_point_3_multiplier
);
$y = (
$this->drawable->first()->y() * $control_point_1_multiplier +
$this->drawable->second()->y() * $control_point_2_multiplier +
$this->drawable->last()->y() * $control_point_3_multiplier
);
return ['x' => $x, 'y' => $y];
}
/**
* Calculate interpolation points for cubic beziers using the Bernstein polynomial form
*
* @return array{'x': float, 'y': float}
*/
private function calculateCubicBezierInterpolationPoint(float $t = 0.05): array
{
$remainder = 1 - $t;
$t_squared = $t * $t;
$remainder_squared = $remainder * $remainder;
$control_point_1_multiplier = $remainder_squared * $remainder;
$control_point_2_multiplier = $remainder_squared * $t * 3;
$control_point_3_multiplier = $t_squared * $remainder * 3;
$control_point_4_multiplier = $t_squared * $t;
$x = (
$this->drawable->first()->x() * $control_point_1_multiplier +
$this->drawable->second()->x() * $control_point_2_multiplier +
$this->drawable->third()->x() * $control_point_3_multiplier +
$this->drawable->last()->x() * $control_point_4_multiplier
);
$y = (
$this->drawable->first()->y() * $control_point_1_multiplier +
$this->drawable->second()->y() * $control_point_2_multiplier +
$this->drawable->third()->y() * $control_point_3_multiplier +
$this->drawable->last()->y() * $control_point_4_multiplier
);
return ['x' => $x, 'y' => $y];
}
/**
* Calculate the points needed to draw a quadratic or cubic bezier with optional border/stroke
*
* @throws GeometryException
* @return array{0: array, 1: array}
*/
private function calculateBezierPoints(): array
{
if ($this->drawable->count() !== 3 && $this->drawable->count() !== 4) {
throw new GeometryException('You must specify either 3 or 4 points to create a bezier curve');
}
$polygon = [];
$inner_polygon = [];
$outer_polygon = [];
$polygon_border_segments = [];
// define ratio t; equivalent to 5 percent distance along edge
$t = 0.05;
$polygon[] = $this->drawable->first()->x();
$polygon[] = $this->drawable->first()->y();
for ($i = $t; $i < 1; $i += $t) {
if ($this->drawable->count() === 3) {
$ip = $this->calculateQuadraticBezierInterpolationPoint($i);
} elseif ($this->drawable->count() === 4) {
$ip = $this->calculateCubicBezierInterpolationPoint($i);
}
$polygon[] = (int) $ip['x'];
$polygon[] = (int) $ip['y'];
}
$polygon[] = $this->drawable->last()->x();
$polygon[] = $this->drawable->last()->y();
if ($this->drawable->hasBorder() && $this->drawable->borderSize() > 1) {
// create the border/stroke effect by calculating two new curves with offset positions
// from the main polygon and then connecting the inner/outer curves to create separate
// 4-point polygon segments
$polygon_total_points = count($polygon);
$offset = ($this->drawable->borderSize() / 2);
for ($i = 0; $i < $polygon_total_points; $i += 2) {
if (array_key_exists($i + 2, $polygon) && array_key_exists($i + 3, $polygon)) {
$dx = $polygon[$i + 2] - $polygon[$i];
$dy = $polygon[$i + 3] - $polygon[$i + 1];
$dxy_sqrt = ($dx * $dx + $dy * $dy) ** 0.5;
// inner polygon
$scale = $offset / $dxy_sqrt;
$ox = -$dy * $scale;
$oy = $dx * $scale;
$inner_polygon[] = $ox + $polygon[$i];
$inner_polygon[] = $oy + $polygon[$i + 1];
$inner_polygon[] = $ox + $polygon[$i + 2];
$inner_polygon[] = $oy + $polygon[$i + 3];
// outer polygon
$scale = -$offset / $dxy_sqrt;
$ox = -$dy * $scale;
$oy = $dx * $scale;
$outer_polygon[] = $ox + $polygon[$i];
$outer_polygon[] = $oy + $polygon[$i + 1];
$outer_polygon[] = $ox + $polygon[$i + 2];
$outer_polygon[] = $oy + $polygon[$i + 3];
}
}
$inner_polygon_total_points = count($inner_polygon);
for ($i = 0; $i < $inner_polygon_total_points; $i += 2) {
if (array_key_exists($i + 2, $inner_polygon) && array_key_exists($i + 3, $inner_polygon)) {
$polygon_border_segments[] = [
$inner_polygon[$i],
$inner_polygon[$i + 1],
$outer_polygon[$i],
$outer_polygon[$i + 1],
$outer_polygon[$i + 2],
$outer_polygon[$i + 3],
$inner_polygon[$i + 2],
$inner_polygon[$i + 3],
];
}
}
}
return [$polygon, $polygon_border_segments];
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/DrawEllipseModifier.php
================================================
drawable->hasBorder()) {
imagealphablending($frame->native(), true);
// slightly smaller ellipse to keep 1px bordered edges clean
if ($this->drawable->hasBackgroundColor()) {
imagefilledellipse(
$frame->native(),
$this->drawable()->position()->x(),
$this->drawable->position()->y(),
$this->drawable->width() - 1,
$this->drawable->height() - 1,
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
)
);
}
// gd's imageellipse ignores imagesetthickness
// so i use imagearc with 360 degrees instead.
imagesetthickness(
$frame->native(),
$this->drawable->borderSize(),
);
imagearc(
$frame->native(),
$this->drawable()->position()->x(),
$this->drawable()->position()->y(),
$this->drawable->width(),
$this->drawable->height(),
0,
360,
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
)
);
} else {
imagealphablending($frame->native(), true);
imagesetthickness($frame->native(), 0);
imagefilledellipse(
$frame->native(),
$this->drawable()->position()->x(),
$this->drawable()->position()->y(),
$this->drawable->width(),
$this->drawable->height(),
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
)
);
}
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/DrawLineModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
);
foreach ($image as $frame) {
imagealphablending($frame->native(), true);
imageantialias($frame->native(), true);
imagesetthickness($frame->native(), $this->drawable->width());
imageline(
$frame->native(),
$this->drawable->start()->x(),
$this->drawable->start()->y(),
$this->drawable->end()->x(),
$this->drawable->end()->y(),
$color
);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/DrawPixelModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->color)
);
foreach ($image as $frame) {
imagealphablending($frame->native(), true);
imagesetpixel(
$frame->native(),
$this->position->x(),
$this->position->y(),
$color
);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/DrawPolygonModifier.php
================================================
drawable->hasBackgroundColor()) {
imagealphablending($frame->native(), true);
imagesetthickness($frame->native(), 0);
imagefilledpolygon(
$frame->native(),
$this->drawable->toArray(),
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
)
);
}
if ($this->drawable->hasBorder()) {
imagealphablending($frame->native(), true);
imagesetthickness($frame->native(), $this->drawable->borderSize());
imagepolygon(
$frame->native(),
$this->drawable->toArray(),
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
)
);
}
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/DrawRectangleModifier.php
================================================
drawable->position();
foreach ($image as $frame) {
// draw background
if ($this->drawable->hasBackgroundColor()) {
imagealphablending($frame->native(), true);
imagesetthickness($frame->native(), 0);
imagefilledrectangle(
$frame->native(),
$position->x(),
$position->y(),
$position->x() + $this->drawable->width(),
$position->y() + $this->drawable->height(),
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
)
);
}
// draw border
if ($this->drawable->hasBorder()) {
imagealphablending($frame->native(), true);
imagesetthickness($frame->native(), $this->drawable->borderSize());
imagerectangle(
$frame->native(),
$position->x(),
$position->y(),
$position->x() + $this->drawable->width(),
$position->y() + $this->drawable->height(),
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
)
);
}
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/FillModifier.php
================================================
color($image);
foreach ($image as $frame) {
if ($this->hasPosition()) {
$this->floodFillWithColor($frame, $color);
} else {
$this->fillAllWithColor($frame, $color);
}
}
return $image;
}
/**
* @throws RuntimeException
*/
private function color(ImageInterface $image): int
{
return $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->color)
);
}
private function floodFillWithColor(FrameInterface $frame, int $color): void
{
imagefill(
$frame->native(),
$this->position->x(),
$this->position->y(),
$color
);
}
private function fillAllWithColor(FrameInterface $frame, int $color): void
{
imagealphablending($frame->native(), true);
imagefilledrectangle(
$frame->native(),
0,
0,
$frame->size()->width() - 1,
$frame->size()->height() - 1,
$color
);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/FlipModifier.php
================================================
native(), IMG_FLIP_VERTICAL);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/FlopModifier.php
================================================
native(), IMG_FLIP_HORIZONTAL);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/GammaModifier.php
================================================
native(), 1, $this->gamma);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/GreyscaleModifier.php
================================================
native(), IMG_FILTER_GRAYSCALE);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/InvertModifier.php
================================================
native(), IMG_FILTER_NEGATE);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/PadModifier.php
================================================
size()
->containMax(
$this->width,
$this->height
)
->alignPivotTo(
$this->getResizeSize($image),
$this->position
);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/PixelateModifier.php
================================================
native(), IMG_FILTER_PIXELATE, $this->size, true);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/PlaceModifier.php
================================================
driver()->handleInput($this->element);
$position = $this->getPosition($image, $watermark);
foreach ($image as $frame) {
imagealphablending($frame->native(), true);
if ($this->opacity === 100) {
$this->placeOpaque($frame, $watermark, $position);
} else {
$this->placeTransparent($frame, $watermark, $position);
}
}
return $image;
}
/**
* Insert watermark with 100% opacity
*
* @throws RuntimeException
*/
private function placeOpaque(FrameInterface $frame, ImageInterface $watermark, PointInterface $position): void
{
imagecopy(
$frame->native(),
$watermark->core()->native(),
$position->x(),
$position->y(),
0,
0,
$watermark->width(),
$watermark->height()
);
}
/**
* Insert watermark transparent with current opacity
*
* Unfortunately, the original PHP function imagecopymerge does not work reliably.
* For example, any transparency of the image to be inserted is not applied correctly.
* For this reason, a new GDImage is created into which the original image is inserted
* in the first step and the watermark is inserted with 100% opacity in the second
* step. This combination is then transferred to the original image again with the
* respective opacity.
*
* Please note: Unfortunately, there is still an edge case, when a transparent image
* is placed on a transparent background, the "double" transparent areas appear opaque!
*
* @throws RuntimeException
*/
private function placeTransparent(FrameInterface $frame, ImageInterface $watermark, PointInterface $position): void
{
$cut = imagecreatetruecolor($watermark->width(), $watermark->height());
imagecopy(
$cut,
$frame->native(),
0,
0,
$position->x(),
$position->y(),
imagesx($cut),
imagesy($cut)
);
imagecopy(
$cut,
$watermark->core()->native(),
0,
0,
0,
0,
imagesx($cut),
imagesy($cut)
);
imagecopymerge(
$frame->native(),
$cut,
$position->x(),
$position->y(),
0,
0,
$watermark->width(),
$watermark->height(),
$this->opacity
);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ProfileModifier.php
================================================
limit <= 0) {
throw new InputException('Quantization limit must be greater than 0.');
}
// no color reduction if the limit is higher than the colors in the img
$colorCount = imagecolorstotal($image->core()->native());
if ($colorCount > 0 && $this->limit > $colorCount) {
return $image;
}
$width = $image->width();
$height = $image->height();
$background = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->background)
);
$blendingColor = $this->driver()->handleInput(
$this->driver()->config()->blendingColor
);
foreach ($image as $frame) {
// create new image for color quantization
$reduced = Cloner::cloneEmpty($frame->native(), background: $blendingColor);
// fill with background
imagefill($reduced, 0, 0, $background);
// set transparency
imagecolortransparent($reduced, $background);
// copy original image (colors are limited automatically in the copy process)
imagecopy($reduced, $frame->native(), 0, 0, 0, 0, $width, $height);
// gd library does not support color quantization directly therefore the
// colors are decrease by transforming the image to a palette version
imagetruecolortopalette($reduced, true, $this->limit);
$frame->setNative($reduced);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/RemoveAnimationModifier.php
================================================
core()->setNative(
$this->selectedFrame($image)->native()
);
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ResizeCanvasModifier.php
================================================
cropSize($image);
$image->modify(new CropModifier(
$cropSize->width(),
$cropSize->height(),
$cropSize->pivot()->x(),
$cropSize->pivot()->y(),
$this->background,
));
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ResizeCanvasRelativeModifier.php
================================================
size()->resizeDown($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ResizeModifier.php
================================================
getAdjustedSize($image);
foreach ($image as $frame) {
$this->resizeFrame($frame, $resizeTo);
}
return $image;
}
/**
* @throws ColorException
*/
private function resizeFrame(FrameInterface $frame, SizeInterface $resizeTo): void
{
// create empty canvas in target size
$modified = Cloner::cloneEmpty($frame->native(), $resizeTo);
// copy content from resource
imagecopyresampled(
$modified,
$frame->native(),
$resizeTo->pivot()->x(),
$resizeTo->pivot()->y(),
0,
0,
$resizeTo->width(),
$resizeTo->height(),
$frame->size()->width(),
$frame->size()->height()
);
// set new content as resource
$frame->setNative($modified);
}
/**
* Return the size the modifier will resize to
*
* @throws RuntimeException
* @throws GeometryException
*/
protected function getAdjustedSize(ImageInterface $image): SizeInterface
{
return $image->size()->resize($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ResolutionModifier.php
================================================
x));
$y = intval(round($this->y));
foreach ($image as $frame) {
imageresolution($frame->native(), $x, $y);
}
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/RotateModifier.php
================================================
driver()->handleInput($this->background);
foreach ($image as $frame) {
$this->modifyFrame($frame, $background);
}
return $image;
}
/**
* Apply rotation modification on given frame, given background
* color is used for newly create image areas
*
* @throws ColorException
*/
protected function modifyFrame(FrameInterface $frame, ColorInterface $background): void
{
// get transparent color from frame core
$transparent = match ($transparent = imagecolortransparent($frame->native())) {
-1 => imagecolorallocatealpha(
$frame->native(),
$background->channel(Red::class)->value(),
$background->channel(Green::class)->value(),
$background->channel(Blue::class)->value(),
127
),
default => $transparent,
};
// rotate original image against transparent background
$rotated = imagerotate(
$frame->native(),
$this->rotationAngle(),
$transparent
);
// create size from original after rotation
$container = (new Rectangle(
imagesx($rotated),
imagesy($rotated),
))->movePivot('center');
// create size from original and rotate points
$cutout = (new Rectangle(
imagesx($frame->native()),
imagesy($frame->native()),
$container->pivot()
))->align('center')
->valign('center')
->rotate($this->rotationAngle() * -1);
// create new gd image
$modified = Cloner::cloneEmpty($frame->native(), $container, $background);
// draw the cutout on new gd image to have a transparent
// background where the rotated image will be placed
imagealphablending($modified, false);
imagefilledpolygon(
$modified,
$cutout->toArray(),
imagecolortransparent($modified)
);
// place rotated image on new gd image
imagealphablending($modified, true);
imagecopy(
$modified,
$rotated,
0,
0,
0,
0,
imagesx($rotated),
imagesy($rotated)
);
$frame->setNative($modified);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ScaleDownModifier.php
================================================
size()->scaleDown($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/ScaleModifier.php
================================================
size()->scale($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/SharpenModifier.php
================================================
matrix();
foreach ($image as $frame) {
imageconvolution($frame->native(), $matrix, 1, 0);
}
return $image;
}
/**
* Create matrix to be used by imageconvolution()
*
* @return array>
*/
private function matrix(): array
{
$min = $this->amount >= 10 ? $this->amount * -0.01 : 0;
$max = $this->amount * -0.025;
$abs = ((4 * $min + 4 * $max) * -1) + 1;
return [
[$min, $max, $min],
[$max, $abs, $max],
[$min, $max, $min]
];
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/SliceAnimationModifier.php
================================================
offset >= $image->count()) {
throw new AnimationException('Offset is not in the range of frames.');
}
$image->core()->slice($this->offset, $this->length);
return $image;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/TextModifier.php
================================================
driver()->fontProcessor();
$lines = $fontProcessor->textBlock($this->text, $this->font, $this->position);
// decode text colors
$textColor = $this->gdTextColor($image);
$strokeColor = $this->gdStrokeColor($image);
// build full path to font file to make sure to pass absolute path to imageftbbox()
// because of issues with different GD version behaving differently when passing
// relative paths to imagettftext()
$fontPath = $this->font->hasFilename() ? realpath($this->font->filename()) : false;
if ($this->font->hasFilename() && $fontPath === false) {
throw new FontException('Font file ' . $this->font->filename() . ' does not exist.');
}
foreach ($image as $frame) {
imagealphablending($frame->native(), true);
if ($this->font->hasFilename()) {
foreach ($lines as $line) {
foreach ($this->strokeOffsets($this->font) as $offset) {
imagettftext(
image: $frame->native(),
size: $fontProcessor->nativeFontSize($this->font),
angle: $this->font->angle() * -1,
x: $line->position()->x() + $offset->x(),
y: $line->position()->y() + $offset->y(),
color: $strokeColor,
font_filename: $fontPath,
text: (string) $line
);
}
imagettftext(
image: $frame->native(),
size: $fontProcessor->nativeFontSize($this->font),
angle: $this->font->angle() * -1,
x: $line->position()->x(),
y: $line->position()->y(),
color: $textColor,
font_filename: $fontPath,
text: (string) $line
);
}
} else {
foreach ($lines as $line) {
foreach ($this->strokeOffsets($this->font) as $offset) {
imagestring(
$frame->native(),
$this->gdFont(),
$line->position()->x() + $offset->x(),
$line->position()->y() + $offset->y(),
(string) $line,
$strokeColor
);
}
imagestring(
$frame->native(),
$this->gdFont(),
$line->position()->x(),
$line->position()->y(),
(string) $line,
$textColor
);
}
}
}
return $image;
}
/**
* Decode text color in GD compatible format
*
* @throws RuntimeException
* @throws ColorException
*/
protected function gdTextColor(ImageInterface $image): int
{
return $this
->driver()
->colorProcessor($image->colorspace())
->colorToNative(parent::textColor());
}
/**
* Decode color for stroke (outline) effect in GD compatible format
*
* @throws RuntimeException
* @throws ColorException
*/
protected function gdStrokeColor(ImageInterface $image): int
{
if (!$this->font->hasStrokeEffect()) {
return 0;
}
$color = parent::strokeColor();
if ($color->isTransparent()) {
throw new ColorException(
'The stroke color must be fully opaque.'
);
}
return $this
->driver()
->colorProcessor($image->colorspace())
->colorToNative($color);
}
/**
* Return GD's internal font size (if no ttf file is set)
*/
private function gdFont(): int
{
if (is_numeric($this->font->filename())) {
return intval($this->font->filename());
}
return 1;
}
}
================================================
FILE: src/Drivers/Gd/Modifiers/TrimModifier.php
================================================
isAnimated()) {
throw new NotSupportedException('Trim modifier cannot be applied to animated images.');
}
// apply tolerance with a min. value of .5 because the default tolerance of '0' should
// already trim away similar colors which is not the case with imagecropauto.
$trimmed = imagecropauto(
$image->core()->native(),
IMG_CROP_THRESHOLD,
max([.5, $this->tolerance / 10]),
$this->trimColor($image)
);
// if the tolerance is very high, it is possible that no image is left.
// imagick returns a 1x1 pixel image in this case. this does the same.
if ($trimmed === false) {
$trimmed = $this->driver()->createImage(1, 1)->core()->native();
}
$image->core()->setNative($trimmed);
return $image;
}
/**
* Create an average color from the colors of the four corner points of the given image
*
* @throws RuntimeException
* @throws AnimationException
*/
private function trimColor(ImageInterface $image): int
{
// trim color base
$red = 0;
$green = 0;
$blue = 0;
// corner coordinates
$size = $image->size();
$cornerPoints = [
new Point(0, 0),
new Point($size->width() - 1, 0),
new Point(0, $size->height() - 1),
new Point($size->width() - 1, $size->height() - 1),
];
// create an average color to be used in trim operation
foreach ($cornerPoints as $pos) {
$cornerColor = imagecolorat($image->core()->native(), $pos->x(), $pos->y());
$rgb = imagecolorsforindex($image->core()->native(), $cornerColor);
$red += round(round(($rgb['red'] / 51)) * 51);
$green += round(round(($rgb['green'] / 51)) * 51);
$blue += round(round(($rgb['blue'] / 51)) * 51);
}
$red = (int) round($red / 4);
$green = (int) round($green / 4);
$blue = (int) round($blue / 4);
return imagecolorallocate($image->core()->native(), $red, $green, $blue);
}
}
================================================
FILE: src/Drivers/Imagick/Analyzers/ColorspaceAnalyzer.php
================================================
core()->native()->getImageColorspace()) {
Imagick::COLORSPACE_CMYK => new CmykColorspace(),
default => new RgbColorspace(),
};
}
}
================================================
FILE: src/Drivers/Imagick/Analyzers/HeightAnalyzer.php
================================================
core()->native()->getImageHeight();
}
}
================================================
FILE: src/Drivers/Imagick/Analyzers/PixelColorAnalyzer.php
================================================
colorAt(
$image->colorspace(),
$image->core()->frame($this->frame_key)->native()
);
}
/**
* @throws ColorException
*/
protected function colorAt(ColorspaceInterface $colorspace, Imagick $imagick): ColorInterface
{
return $this->driver()
->colorProcessor($colorspace)
->nativeToColor(
$imagick->getImagePixelColor($this->x, $this->y)
);
}
}
================================================
FILE: src/Drivers/Imagick/Analyzers/PixelColorsAnalyzer.php
================================================
colorspace();
foreach ($image as $frame) {
$colors->push(
parent::colorAt($colorspace, $frame->native())
);
}
return $colors;
}
}
================================================
FILE: src/Drivers/Imagick/Analyzers/ProfileAnalyzer.php
================================================
core()->native()->getImageProfiles('icc');
if (!array_key_exists('icc', $profiles)) {
throw new ColorException('No ICC profile found in image.');
}
return new Profile($profiles['icc']);
}
}
================================================
FILE: src/Drivers/Imagick/Analyzers/ResolutionAnalyzer.php
================================================
core()->native();
$imageResolution = $imagick->getImageResolution();
return new Resolution(
$imageResolution['x'],
$imageResolution['y'],
$imagick->getImageUnits(),
);
}
}
================================================
FILE: src/Drivers/Imagick/Analyzers/WidthAnalyzer.php
================================================
core()->native()->getImageWidth();
}
}
================================================
FILE: src/Drivers/Imagick/ColorProcessor.php
================================================
convertTo($this->colorspace)
);
}
public function nativeToColor(mixed $native): ColorInterface
{
return match ($this->colorspace::class) {
CmykColorspace::class => $this->colorspace->colorFromNormalized([
$native->getColorValue(Imagick::COLOR_CYAN),
$native->getColorValue(Imagick::COLOR_MAGENTA),
$native->getColorValue(Imagick::COLOR_YELLOW),
$native->getColorValue(Imagick::COLOR_BLACK),
]),
default => $this->colorspace->colorFromNormalized([
$native->getColorValue(Imagick::COLOR_RED),
$native->getColorValue(Imagick::COLOR_GREEN),
$native->getColorValue(Imagick::COLOR_BLUE),
$native->getColorValue(Imagick::COLOR_ALPHA),
]),
};
}
}
================================================
FILE: src/Drivers/Imagick/Core.php
================================================
*/
class Core implements CoreInterface, Iterator
{
protected int $iteratorIndex = 0;
/**
* Create new core instance
*
* @return void
*/
public function __construct(protected Imagick $imagick)
{
//
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::has()
*/
public function has(int|string $key): bool
{
try {
$result = $this->imagick->setIteratorIndex($key);
} catch (ImagickException) {
return false;
}
return $result;
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::push()
*/
public function push(mixed $item): CollectionInterface
{
return $this->add($item);
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::get()
*/
public function get(int|string $key, mixed $default = null): mixed
{
try {
$this->imagick->setIteratorIndex($key);
} catch (ImagickException) {
return $default;
}
return new Frame($this->imagick->current());
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::getAtPosition()
*/
public function getAtPosition(int $key = 0, mixed $default = null): mixed
{
return $this->get($key, $default);
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::empty()
*/
public function empty(): CollectionInterface
{
$this->imagick->clear();
return $this;
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::slice()
*/
public function slice(int $offset, ?int $length = null): CollectionInterface
{
$allowed_indexes = [];
$length = is_null($length) ? $this->count() : $length;
for ($i = $offset; $i < $offset + $length; $i++) {
$allowed_indexes[] = $i;
}
$sliced = new Imagick();
foreach ($this->imagick as $key => $native) {
if (in_array($key, $allowed_indexes)) {
$sliced->addImage($native->getImage());
}
}
$sliced = $sliced->coalesceImages();
$sliced->setImageIterations($this->imagick->getImageIterations());
$this->imagick = $sliced;
return $this;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::add()
*/
public function add(FrameInterface $frame): CoreInterface
{
$imagick = $frame->native();
$imagick->setImageDelay(
(int) round($frame->delay() * 100)
);
$imagick->setImageDispose($frame->dispose());
$size = $frame->size();
$imagick->setImagePage(
$size->width(),
$size->height(),
$frame->offsetLeft(),
$frame->offsetTop()
);
$this->imagick->addImage($imagick);
return $this;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::count()
*/
public function count(): int
{
return $this->imagick->getNumberImages();
}
/**
* {@inheritdoc}
*
* @see Iterator::rewind()
*/
public function current(): mixed
{
$this->imagick->setIteratorIndex($this->iteratorIndex);
return new Frame($this->imagick->current());
}
/**
* {@inheritdoc}
*
* @see Iterator::rewind()
*/
public function next(): void
{
$this->iteratorIndex += 1;
}
/**
* {@inheritdoc}
*
* @see Iterator::rewind()
*/
public function key(): mixed
{
return $this->iteratorIndex;
}
/**
* {@inheritdoc}
*
* @see Iterator::rewind()
*/
public function valid(): bool
{
try {
$result = $this->imagick->setIteratorIndex($this->iteratorIndex);
} catch (ImagickException) {
return false;
}
return $result;
}
/**
* {@inheritdoc}
*
* @see Iterator::rewind()
*/
public function rewind(): void
{
$this->iteratorIndex = 0;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::native()
*/
public function native(): mixed
{
return $this->imagick;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::setNative()
*/
public function setNative(mixed $native): CoreInterface
{
$this->imagick = $native;
return $this;
}
/**
* {@inheritdoc}
*
* @see CoreInterface::frame()
*/
public function frame(int $position): FrameInterface
{
foreach ($this->imagick as $core) {
if ($core->getIteratorIndex() === $position) {
return new Frame($core);
}
}
throw new AnimationException('Frame #' . $position . ' could not be found in the image.');
}
/**
* {@inheritdoc}
*
* @see CoreInterface::loops()
*/
public function loops(): int
{
return $this->imagick->getImageIterations();
}
/**
* {@inheritdoc}
*
* @see CoreInterface::setLoops()
*/
public function setLoops(int $loops): CoreInterface
{
$this->imagick = $this->imagick->coalesceImages();
$this->imagick->setImageIterations($loops);
return $this;
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::first()
*/
public function first(): FrameInterface
{
return $this->frame(0);
}
/**
* {@inheritdoc}
*
* @see CollectableInterface::last()
*/
public function last(): FrameInterface
{
return $this->frame($this->count() - 1);
}
/**
* {@inheritdoc}
*
* @see CollectionInterface::toArray()
*/
public function toArray(): array
{
$frames = [];
foreach ($this as $frame) {
$frames[] = $frame;
}
return $frames;
}
/**
* Clone instance
*/
public function __clone(): void
{
$this->imagick = clone $this->imagick;
}
}
================================================
FILE: src/Drivers/Imagick/Decoders/Base64ImageDecoder.php
================================================
isValidBase64($input)) {
throw new DecoderException('Unable to decode input');
}
return parent::decode(base64_decode((string) $input));
}
}
================================================
FILE: src/Drivers/Imagick/Decoders/BinaryImageDecoder.php
================================================
readImageBlob($input);
} catch (ImagickException) {
throw new DecoderException('Unable to decode input');
}
// decode image
$image = parent::decode($imagick);
// get media type enum from string media type
$format = Format::tryCreate($image->origin()->mediaType());
// extract exif data for appropriate formats
if (in_array($format, [Format::JPEG, Format::TIFF])) {
$image->setExif($this->extractExifData($input));
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Decoders/DataUriImageDecoder.php
================================================
parseDataUri($input);
if (!$uri->isValid()) {
throw new DecoderException('Unable to decode input');
}
if ($uri->isBase64Encoded()) {
return parent::decode(base64_decode($uri->data()));
}
return parent::decode(urldecode($uri->data()));
}
}
================================================
FILE: src/Drivers/Imagick/Decoders/EncodedImageObjectDecoder.php
================================================
toString());
}
}
================================================
FILE: src/Drivers/Imagick/Decoders/FilePathImageDecoder.php
================================================
isFile($input)) {
throw new DecoderException('Unable to decode input');
}
try {
$imagick = new Imagick();
$imagick->readImage($input);
} catch (ImagickException) {
throw new DecoderException('Unable to decode input');
}
// decode image
$image = parent::decode($imagick);
// set file path on origin
$image->origin()->setFilePath($input);
// extract exif data for the appropriate formats
if (in_array($imagick->getImageFormat(), ['JPEG', 'TIFF', 'TIF'])) {
$image->setExif($this->extractExifData($input));
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Decoders/FilePointerImageDecoder.php
================================================
getImageFormat() !== 'JPEG') {
$input = $input->coalesceImages();
}
// turn images with colorspace 'GRAY' into 'SRGB' to avoid working on
// greyscale colorspace images as this results images loosing color
// information when placed into this image.
if ($input->getImageColorspace() == Imagick::COLORSPACE_GRAY) {
$input->setImageColorspace(Imagick::COLORSPACE_SRGB);
}
// create image object
$image = new Image(
$this->driver(),
new Core($input)
);
// discard animation depending on config
if (!$this->driver()->config()->decodeAnimation) {
$image->modify(new RemoveAnimationModifier());
}
// adjust image rotatation
if ($this->driver()->config()->autoOrientation) {
$image->modify(new AlignRotationModifier());
}
// set media type on origin
$image->origin()->setMediaType($input->getImageMimeType());
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Decoders/SplFileInfoImageDecoder.php
================================================
getRealPath());
}
}
================================================
FILE: src/Drivers/Imagick/Driver.php
================================================
newImage($width, $height, $background, 'png');
$imagick->setType(Imagick::IMGTYPE_UNDEFINED);
$imagick->setImageType(Imagick::IMGTYPE_UNDEFINED);
$imagick->setColorspace(Imagick::COLORSPACE_SRGB);
$imagick->setImageResolution(96, 96);
$imagick->setImageBackgroundColor($background);
return new Image($this, new Core($imagick));
}
/**
* {@inheritdoc}
*
* @see DriverInterface::createAnimation()
*
* @throws RuntimeException
*/
public function createAnimation(callable $init): ImageInterface
{
$imagick = new Imagick();
$imagick->setFormat('gif');
$animation = new class ($this, $imagick)
{
public function __construct(
protected DriverInterface $driver,
public Imagick $imagick
) {
//
}
/**
* @throws RuntimeException
*/
public function add(mixed $source, float $delay = 1): self
{
$native = $this->driver->handleInput($source)->core()->native();
$native->setImageDelay(intval(round($delay * 100)));
$this->imagick->addImage($native);
return $this;
}
/**
* @throws RuntimeException
*/
public function __invoke(): ImageInterface
{
return new Image(
$this->driver,
new Core($this->imagick)
);
}
};
$init($animation);
return call_user_func($animation);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::colorProcessor()
*/
public function colorProcessor(ColorspaceInterface $colorspace): ColorProcessorInterface
{
return new ColorProcessor($colorspace);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::fontProcessor()
*/
public function fontProcessor(): FontProcessorInterface
{
return new FontProcessor();
}
/**
* {@inheritdoc}
*
* @see DriverInterface::supports()
*/
public function supports(string|Format|FileExtension|MediaType $identifier): bool
{
try {
$format = Format::create($identifier);
} catch (NotSupportedException) {
return false;
}
return count(Imagick::queryFormats($format->name)) >= 1;
}
/**
* Return version of ImageMagick library
*
* @throws DriverException
*/
public static function version(): string
{
$pattern = '/^ImageMagick (?P(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)' .
'(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?' .
'(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)/';
if (preg_match($pattern, Imagick::getVersion()['versionString'], $matches) !== 1) {
throw new DriverException('Unable to read ImageMagick version number.');
}
return $matches['version'];
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/AvifEncoder.php
================================================
strip || (is_null($this->strip) && $this->driver()->config()->strip)) {
$image->modify(new StripMetaModifier());
}
$imagick = $image->core()->native();
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompression($compression);
$imagick->setImageCompression($compression);
$imagick->setCompressionQuality($this->quality);
$imagick->setImageCompressionQuality($this->quality);
return new EncodedImage($imagick->getImagesBlob(), 'image/avif');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/BmpEncoder.php
================================================
core()->native();
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompression($compression);
$imagick->setImageCompression($compression);
return new EncodedImage($imagick->getImagesBlob(), 'image/bmp');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/GifEncoder.php
================================================
core()->native();
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompression($compression);
$imagick->setImageCompression($compression);
if ($this->interlaced) {
$imagick->setInterlaceScheme(Imagick::INTERLACE_LINE);
}
return new EncodedImage($imagick->getImagesBlob(), 'image/gif');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/HeicEncoder.php
================================================
strip || (is_null($this->strip) && $this->driver()->config()->strip)) {
$image->modify(new StripMetaModifier());
}
$imagick = $image->core()->native();
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompressionQuality($this->quality);
$imagick->setImageCompressionQuality($this->quality);
return new EncodedImage($imagick->getImagesBlob(), 'image/heic');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/Jpeg2000Encoder.php
================================================
strip || (is_null($this->strip) && $this->driver()->config()->strip)) {
$image->modify(new StripMetaModifier());
}
$imagick = $image->core()->native();
$imagick->setImageBackgroundColor('white');
$imagick->setBackgroundColor('white');
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompression($compression);
$imagick->setImageCompression($compression);
$imagick->setCompressionQuality($this->quality);
$imagick->setImageCompressionQuality($this->quality);
return new EncodedImage($imagick->getImagesBlob(), 'image/jp2');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/JpegEncoder.php
================================================
driver()->handleInput(
$this->driver()->config()->blendingColor
);
// resolve blending color because jpeg has no transparency
$background = $this->driver()
->colorProcessor($image->colorspace())
->colorToNative($blendingColor);
// set alpha value to 1 because Imagick renders
// possible full transparent colors as black
$background->setColorValue(Imagick::COLOR_ALPHA, 1);
// strip meta data
if ($this->strip || (is_null($this->strip) && $this->driver()->config()->strip)) {
$image->modify(new StripMetaModifier());
}
/** @var Imagick $imagick */
$imagick = $image->core()->native();
$imagick->setImageBackgroundColor($background);
$imagick->setBackgroundColor($background);
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompression($compression);
$imagick->setImageCompression($compression);
$imagick->setCompressionQuality($this->quality);
$imagick->setImageCompressionQuality($this->quality);
$imagick->setImageAlphaChannel(Imagick::ALPHACHANNEL_REMOVE);
if ($this->progressive) {
$imagick->setInterlaceScheme(Imagick::INTERLACE_PLANE);
}
return new EncodedImage($imagick->getImagesBlob(), 'image/jpeg');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/PngEncoder.php
================================================
indexed) {
// reduce colors
$output = clone $image;
$output->reduceColors(256);
$output = $output->core()->native();
$output->setFormat('PNG');
$output->setImageFormat('PNG');
} else {
$output = clone $image->core()->native();
$output->setFormat('PNG32');
$output->setImageFormat('PNG32');
}
$output->setCompression(Imagick::COMPRESSION_ZIP);
$output->setImageCompression(Imagick::COMPRESSION_ZIP);
if ($this->interlaced) {
$output->setInterlaceScheme(Imagick::INTERLACE_LINE);
}
return new EncodedImage($output->getImagesBlob(), 'image/png');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/TiffEncoder.php
================================================
strip || (is_null($this->strip) && $this->driver()->config()->strip)) {
$image->modify(new StripMetaModifier());
}
$imagick = $image->core()->native();
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompression($imagick->getImageCompression());
$imagick->setImageCompression($imagick->getImageCompression());
$imagick->setCompressionQuality($this->quality);
$imagick->setImageCompressionQuality($this->quality);
return new EncodedImage($imagick->getImagesBlob(), 'image/tiff');
}
}
================================================
FILE: src/Drivers/Imagick/Encoders/WebpEncoder.php
================================================
strip || (is_null($this->strip) && $this->driver()->config()->strip)) {
$image->modify(new StripMetaModifier());
}
$imagick = $image->core()->native();
$imagick->setImageBackgroundColor(new ImagickPixel('transparent'));
if (!$image->isAnimated()) {
$imagick = $imagick->mergeImageLayers(Imagick::LAYERMETHOD_MERGE);
}
$imagick->setFormat($format);
$imagick->setImageFormat($format);
$imagick->setCompression($compression);
$imagick->setImageCompression($compression);
$imagick->setImageCompressionQuality($this->quality);
if ($this->quality === 100) {
$imagick->setOption('webp:lossless', 'true');
}
return new EncodedImage($imagick->getImagesBlob(), 'image/webp');
}
}
================================================
FILE: src/Drivers/Imagick/FontProcessor.php
================================================
toImagickDraw($font);
$dimensions = (new Imagick())->queryFontMetrics($draw, $text);
return new Rectangle(
intval(round($dimensions['textWidth'])),
intval(round($dimensions['ascender'] + $dimensions['descender'])),
);
}
/**
* Imagick::annotateImage() needs an ImagickDraw object - this method takes
* the font object as the base and adds an optional passed color to the new
* ImagickDraw object.
*
* @throws FontException
* @throws ImagickDrawException
* @throws ImagickException
*/
public function toImagickDraw(FontInterface $font, ?ImagickPixel $color = null): ImagickDraw
{
if (!$font->hasFilename()) {
throw new FontException('No font file specified.');
}
$draw = new ImagickDraw();
$draw->setStrokeAntialias(true);
$draw->setTextAntialias(true);
$draw->setFont($font->filename());
$draw->setFontSize($this->nativeFontSize($font));
$draw->setTextAlignment(Imagick::ALIGN_LEFT);
if ($color instanceof ImagickPixel) {
$draw->setFillColor($color);
}
return $draw;
}
}
================================================
FILE: src/Drivers/Imagick/Frame.php
================================================
native->setImageBackgroundColor($background);
$this->native->setBackgroundColor($background);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::toImage()
*/
public function toImage(DriverInterface $driver): ImageInterface
{
return new Image($driver, new Core($this->native()));
}
/**
* {@inheritdoc}
*
* @see DriverInterface::setNative()
*/
public function setNative(mixed $native): FrameInterface
{
$this->native = $native;
return $this;
}
/**
* {@inheritdoc}
*
* @see DriverInterface::native()
*/
public function native(): Imagick
{
return $this->native;
}
/**
* {@inheritdoc}
*
* @see DriverInterface::size()
*/
public function size(): SizeInterface
{
return new Rectangle(
$this->native->getImageWidth(),
$this->native->getImageHeight()
);
}
/**
* {@inheritdoc}
*
* @see DriverInterface::delay()
*/
public function delay(): float
{
return $this->native->getImageDelay() / 100;
}
/**
* {@inheritdoc}
*
* @see DriverInterface::setDelay()
*/
public function setDelay(float $delay): FrameInterface
{
$this->native->setImageDelay(intval(round($delay * 100)));
return $this;
}
/**
* {@inheritdoc}
*
* @see DriverInterface::dispose()
*/
public function dispose(): int
{
return $this->native->getImageDispose();
}
/**
* {@inheritdoc}
*
* @see DriverInterface::setDispose()
*
* @throws InputException
*/
public function setDispose(int $dispose): FrameInterface
{
if (!in_array($dispose, [0, 1, 2, 3])) {
throw new InputException('Value for argument $dispose must be 0, 1, 2 or 3.');
}
$this->native->setImageDispose($dispose);
return $this;
}
/**
* {@inheritdoc}
*
* @see DriverInterface::setOffset()
*/
public function setOffset(int $left, int $top): FrameInterface
{
$this->native->setImagePage(
$this->native->getImageWidth(),
$this->native->getImageHeight(),
$left,
$top
);
return $this;
}
/**
* {@inheritdoc}
*
* @see DriverInterface::offsetLeft()
*/
public function offsetLeft(): int
{
return $this->native->getImagePage()['x'];
}
/**
* {@inheritdoc}
*
* @see DriverInterface::setOffsetLeft()
*/
public function setOffsetLeft(int $offset): FrameInterface
{
return $this->setOffset($offset, $this->offsetTop());
}
/**
* {@inheritdoc}
*
* @see DriverInterface::offsetTop()
*/
public function offsetTop(): int
{
return $this->native->getImagePage()['y'];
}
/**
* {@inheritdoc}
*
* @see DriverInterface::setOffsetTop()
*/
public function setOffsetTop(int $offset): FrameInterface
{
return $this->setOffset($this->offsetLeft(), $offset);
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/AlignRotationModifier.php
================================================
core()->native()->getImageOrientation()) {
case Imagick::ORIENTATION_TOPRIGHT: // 2
$image->core()->native()->flopImage();
break;
case Imagick::ORIENTATION_BOTTOMRIGHT: // 3
$image->core()->native()->rotateImage("#000", 180);
break;
case Imagick::ORIENTATION_BOTTOMLEFT: // 4
$image->core()->native()->rotateImage("#000", 180);
$image->core()->native()->flopImage();
break;
case Imagick::ORIENTATION_LEFTTOP: // 5
$image->core()->native()->rotateImage("#000", -270);
$image->core()->native()->flopImage();
break;
case Imagick::ORIENTATION_RIGHTTOP: // 6
$image->core()->native()->rotateImage("#000", -270);
break;
case Imagick::ORIENTATION_RIGHTBOTTOM: // 7
$image->core()->native()->rotateImage("#000", -90);
$image->core()->native()->flopImage();
break;
case Imagick::ORIENTATION_LEFTBOTTOM: // 8
$image->core()->native()->rotateImage("#000", -90);
break;
}
// set new orientation in image
$image->core()->native()->setImageOrientation(Imagick::ORIENTATION_TOPLEFT);
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/BlendTransparencyModifier.php
================================================
blendingColor($this->driver());
// get imagickpixel from blending color
$pixel = $this->driver()
->colorProcessor($image->colorspace())
->colorToNative($blendingColor);
// merge transparent areas with the background color
foreach ($image as $frame) {
$frame->native()->setImageBackgroundColor($pixel);
$frame->native()->setImageAlphaChannel(Imagick::ALPHACHANNEL_REMOVE);
$frame->native()->mergeImageLayers(Imagick::LAYERMETHOD_FLATTEN);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/BlurModifier.php
================================================
native()->blurImage($this->amount, 0.5 * $this->amount);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/BrightnessModifier.php
================================================
native()->modulateImage(100 + $this->level, 100, 100);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ColorizeModifier.php
================================================
normalizeLevel($this->red);
$green = $this->normalizeLevel($this->green);
$blue = $this->normalizeLevel($this->blue);
foreach ($image as $frame) {
$qrange = $frame->native()->getQuantumRange();
$frame->native()->levelImage(0, $red, $qrange['quantumRangeLong'], Imagick::CHANNEL_RED);
$frame->native()->levelImage(0, $green, $qrange['quantumRangeLong'], Imagick::CHANNEL_GREEN);
$frame->native()->levelImage(0, $blue, $qrange['quantumRangeLong'], Imagick::CHANNEL_BLUE);
}
return $image;
}
private function normalizeLevel(int $level): int
{
return $level > 0 ? intval(round($level / 5)) : intval(round(($level + 100) / 100));
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ColorspaceModifier.php
================================================
*/
protected static array $mapping = [
RgbColorspace::class => Imagick::COLORSPACE_SRGB,
CmykColorspace::class => Imagick::COLORSPACE_CMYK,
];
public function apply(ImageInterface $image): ImageInterface
{
$colorspace = $this->targetColorspace();
$imagick = $image->core()->native();
$imagick->transformImageColorspace(
$this->getImagickColorspace($colorspace)
);
return $image;
}
/**
* @throws NotSupportedException
*/
private function getImagickColorspace(ColorspaceInterface $colorspace): int
{
if (!array_key_exists($colorspace::class, self::$mapping)) {
throw new NotSupportedException('Given colorspace is not supported.');
}
return self::$mapping[$colorspace::class];
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ContainModifier.php
================================================
getCropSize($image);
$resize = $this->getResizeSize($image);
$transparent = new ImagickPixel('transparent');
$background = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->background)
);
foreach ($image as $frame) {
$frame->native()->scaleImage(
$crop->width(),
$crop->height(),
);
$frame->native()->setBackgroundColor($transparent);
$frame->native()->setImageBackgroundColor($transparent);
$frame->native()->extentImage(
$resize->width(),
$resize->height(),
$crop->pivot()->x() * -1,
$crop->pivot()->y() * -1
);
if ($resize->width() > $crop->width()) {
// fill new emerged background
$draw = new ImagickDraw();
$draw->setFillColor($background);
$delta = abs($crop->pivot()->x());
if ($delta > 0) {
$draw->rectangle(
0,
0,
$delta - 1,
$resize->height()
);
}
$draw->rectangle(
$crop->width() + $delta,
0,
$resize->width(),
$resize->height()
);
$frame->native()->drawImage($draw);
}
if ($resize->height() > $crop->height()) {
// fill new emerged background
$draw = new ImagickDraw();
$draw->setFillColor($background);
$delta = abs($crop->pivot()->y());
if ($delta > 0) {
$draw->rectangle(
0,
0,
$resize->width(),
$delta - 1
);
}
$draw->rectangle(
0,
$crop->height() + $delta,
$resize->width(),
$resize->height()
);
$frame->native()->drawImage($draw);
}
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ContrastModifier.php
================================================
native()->sigmoidalContrastImage($this->level > 0, abs($this->level / 4), 0);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/CoverDownModifier.php
================================================
resizeDown($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/CoverModifier.php
================================================
getCropSize($image);
$resize = $this->getResizeSize($crop);
foreach ($image as $frame) {
$frame->native()->cropImage(
$crop->width(),
$crop->height(),
$crop->pivot()->x(),
$crop->pivot()->y()
);
$frame->native()->scaleImage(
$resize->width(),
$resize->height()
);
$frame->native()->setImagePage(0, 0, 0, 0);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/CropModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->background)
);
// create empty container imagick to rebuild core
$imagick = new Imagick();
// save resolution to add it later
$resolution = $image->resolution()->perInch();
// define position of the image on the new canvas
$crop = $this->crop($image);
$position = [
($crop->pivot()->x() + $this->offset_x) * -1,
($crop->pivot()->y() + $this->offset_y) * -1,
];
foreach ($image as $frame) {
// create new frame canvas with modifiers background
$canvas = new Imagick();
$canvas->newImage($crop->width(), $crop->height(), $background, 'png');
$canvas->setImageResolution($resolution->x(), $resolution->y());
$canvas->setImageAlphaChannel(Imagick::ALPHACHANNEL_SET); // or ALPHACHANNEL_ACTIVATE?
// set animation details
if ($image->isAnimated()) {
$canvas->setImageDelay($frame->native()->getImageDelay());
$canvas->setImageIterations($frame->native()->getImageIterations());
$canvas->setImageDispose($frame->native()->getImageDispose());
}
// make the rectangular position of the original image transparent
// so that we can later place the original on top. this preserves
// the transparency of the original and shows the background color
// of the modifier in the other areas. if the original image has no
// transparent area the rectangular transparency will be covered by
// the original.
$clearer = new Imagick();
$clearer->newImage(
$frame->native()->getImageWidth(),
$frame->native()->getImageHeight(),
new ImagickPixel('black'),
);
$canvas->compositeImage($clearer, Imagick::COMPOSITE_DSTOUT, ...$position);
// place original frame content onto prepared frame canvas
$canvas->compositeImage($frame->native(), Imagick::COMPOSITE_DEFAULT, ...$position);
// add newly built frame to container imagick
$imagick->addImage($canvas);
}
// replace imagick in the original image
$image->core()->setNative($imagick);
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/DrawBezierModifier.php
================================================
drawable->count() !== 3 && $this->drawable->count() !== 4) {
throw new GeometryException('You must specify either 3 or 4 points to create a bezier curve');
}
$drawing = new ImagickDraw();
if ($this->drawable->hasBackgroundColor()) {
$background_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
);
} else {
$background_color = 'transparent';
}
$drawing->setFillColor($background_color);
if ($this->drawable->hasBorder() && $this->drawable->borderSize() > 0) {
$border_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
);
$drawing->setStrokeColor($border_color);
$drawing->setStrokeWidth($this->drawable->borderSize());
}
$drawing->pathStart();
$drawing->pathMoveToAbsolute(
$this->drawable->first()->x(),
$this->drawable->first()->y()
);
if ($this->drawable->count() === 3) {
$drawing->pathCurveToQuadraticBezierAbsolute(
$this->drawable->second()->x(),
$this->drawable->second()->y(),
$this->drawable->last()->x(),
$this->drawable->last()->y()
);
} elseif ($this->drawable->count() === 4) {
$drawing->pathCurveToAbsolute(
$this->drawable->second()->x(),
$this->drawable->second()->y(),
$this->drawable->third()->x(),
$this->drawable->third()->y(),
$this->drawable->last()->x(),
$this->drawable->last()->y()
);
}
$drawing->pathFinish();
foreach ($image as $frame) {
$frame->native()->drawImage($drawing);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/DrawEllipseModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
);
$border_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
);
foreach ($image as $frame) {
$drawing = new ImagickDraw();
$drawing->setFillColor($background_color);
if ($this->drawable->hasBorder()) {
$drawing->setStrokeWidth($this->drawable->borderSize());
$drawing->setStrokeColor($border_color);
}
$drawing->ellipse(
$this->drawable->position()->x(),
$this->drawable->position()->y(),
$this->drawable->width() / 2,
$this->drawable->height() / 2,
0,
360
);
$frame->native()->drawImage($drawing);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/DrawLineModifier.php
================================================
setStrokeWidth($this->drawable->width());
$drawing->setFillOpacity(0);
$drawing->setStrokeColor(
$this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
)
);
$drawing->line(
$this->drawable->start()->x(),
$this->drawable->start()->y(),
$this->drawable->end()->x(),
$this->drawable->end()->y(),
);
foreach ($image as $frame) {
$frame->native()->drawImage($drawing);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/DrawPixelModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->color)
);
$pixel = new ImagickDraw();
$pixel->setFillColor($color);
$pixel->point($this->position->x(), $this->position->y());
foreach ($image as $frame) {
$frame->native()->drawImage($pixel);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/DrawPolygonModifier.php
================================================
setFillColor(new ImagickPixel('transparent')); // defaults to no backgroundColor
if ($this->drawable->hasBackgroundColor()) {
$background_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
);
$drawing->setFillColor($background_color);
}
if ($this->drawable->hasBorder()) {
$border_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
);
$drawing->setStrokeColor($border_color);
$drawing->setStrokeWidth($this->drawable->borderSize());
}
$drawing->polygon($this->points());
foreach ($image as $frame) {
$frame->native()->drawImage($drawing);
}
return $image;
}
/**
* Return points of drawable in processable form for ImagickDraw
*
* @return array>
*/
private function points(): array
{
$points = [];
foreach ($this->drawable as $point) {
$points[] = ['x' => $point->x(), 'y' => $point->y()];
}
return $points;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/DrawRectangleModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->backgroundColor()
);
$border_color = $this->driver()->colorProcessor($image->colorspace())->colorToNative(
$this->borderColor()
);
$drawing->setFillColor($background_color);
if ($this->drawable->hasBorder()) {
$drawing->setStrokeColor($border_color);
$drawing->setStrokeWidth($this->drawable->borderSize());
}
// build rectangle
$drawing->rectangle(
$this->drawable->position()->x(),
$this->drawable->position()->y(),
$this->drawable->position()->x() + $this->drawable->width(),
$this->drawable->position()->y() + $this->drawable->height()
);
foreach ($image as $frame) {
$frame->native()->drawImage($drawing);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/FillModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->color)
);
foreach ($image->core()->native() as $frame) {
if ($this->hasPosition()) {
$this->floodFillWithColor($frame, $pixel);
} else {
$this->fillAllWithColor($frame, $pixel);
}
}
return $image;
}
private function floodFillWithColor(Imagick $frame, ImagickPixel $pixel): void
{
$target = $frame->getImagePixelColor(
$this->position->x(),
$this->position->y()
);
$frame->floodfillPaintImage(
$pixel,
100,
$target,
$this->position->x(),
$this->position->y(),
false,
Imagick::CHANNEL_ALL
);
}
private function fillAllWithColor(Imagick $frame, ImagickPixel $pixel): void
{
$draw = new ImagickDraw();
$draw->setFillColor($pixel);
$draw->rectangle(0, 0, $frame->getImageWidth(), $frame->getImageHeight());
$frame->drawImage($draw);
// deactive alpha channel when image was filled with opaque color
if ($pixel->getColorValue(Imagick::COLOR_ALPHA) == 1) {
$frame->setImageAlphaChannel(Imagick::ALPHACHANNEL_DEACTIVATE);
}
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/FlipModifier.php
================================================
native()->flipImage();
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/FlopModifier.php
================================================
native()->flopImage();
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/GammaModifier.php
================================================
native()->gammaImage($this->gamma);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/GreyscaleModifier.php
================================================
native()->modulateImage(100, 0, 100);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/InvertModifier.php
================================================
native()->negateImage(false);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/PadModifier.php
================================================
size()
->containMax(
$this->width,
$this->height
)
->alignPivotTo(
$this->getResizeSize($image),
$this->position
);
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/PixelateModifier.php
================================================
pixelateFrame($frame);
}
return $image;
}
protected function pixelateFrame(FrameInterface $frame): void
{
$size = $frame->size();
$frame->native()->scaleImage(
(int) round(max(1, $size->width() / $this->size)),
(int) round(max(1, $size->height() / $this->size))
);
$frame->native()->scaleImage($size->width(), $size->height());
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/PlaceModifier.php
================================================
driver()->handleInput($this->element);
$position = $this->getPosition($image, $watermark);
// set opacity of watermark
if ($this->opacity < 100) {
$watermark->core()->native()->setImageAlphaChannel(Imagick::ALPHACHANNEL_SET);
$watermark->core()->native()->evaluateImage(
Imagick::EVALUATE_DIVIDE,
$this->opacity > 0 ? 100 / $this->opacity : 1000,
Imagick::CHANNEL_ALPHA,
);
}
foreach ($image as $frame) {
$frame->native()->compositeImage(
$watermark->core()->native(),
Imagick::COMPOSITE_DEFAULT,
$position->x(),
$position->y()
);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ProfileModifier.php
================================================
core()->native();
$result = $imagick->profileImage('icc', (string) $this->profile);
if ($result === false) {
throw new ColorException('ICC color profile could not be set.');
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ProfileRemovalModifier.php
================================================
core()->native();
$result = $imagick->profileImage('icc', null);
if ($result === false) {
throw new ColorException('ICC color profile could not be removed.');
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/QuantizeColorsModifier.php
================================================
limit <= 0) {
throw new InputException('Quantization limit must be greater than 0.');
}
// no color reduction if the limit is higher than the colors in the img
if ($this->limit > $image->core()->native()->getImageColors()) {
return $image;
}
foreach ($image as $frame) {
$frame->native()->quantizeImage(
$this->limit,
$frame->native()->getImageColorspace(),
0,
false,
false
);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/RemoveAnimationModifier.php
================================================
selectedFrame($image);
$imagick->addImage($frame->native()->getImage());
// set new imagick to image
$image->core()->setNative($imagick);
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ResizeCanvasModifier.php
================================================
cropSize($image);
$image->modify(new CropModifier(
$cropSize->width(),
$cropSize->height(),
$cropSize->pivot()->x(),
$cropSize->pivot()->y(),
$this->background,
));
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ResizeCanvasRelativeModifier.php
================================================
size()->resizeDown($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ResizeModifier.php
================================================
getAdjustedSize($image);
foreach ($image as $frame) {
$frame->native()->scaleImage(
$resizeTo->width(),
$resizeTo->height()
);
}
return $image;
}
/**
* @throws RuntimeException
*/
protected function getAdjustedSize(ImageInterface $image): SizeInterface
{
return $image->size()->resize($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ResolutionModifier.php
================================================
core()->native();
$imagick->setImageResolution($this->x, $this->y);
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/RotateModifier.php
================================================
driver()->colorProcessor($image->colorspace())->colorToNative(
$this->driver()->handleInput($this->background)
);
foreach ($image as $frame) {
$frame->native()->rotateImage(
$background,
$this->rotationAngle() * -1
);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ScaleDownModifier.php
================================================
size()->scaleDown($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/ScaleModifier.php
================================================
size()->scale($this->width, $this->height);
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/SharpenModifier.php
================================================
native()->unsharpMaskImage(1, 1, $this->amount / 6.25, 0);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/SliceAnimationModifier.php
================================================
offset >= $image->count()) {
throw new AnimationException('Offset is not in the range of frames.');
}
$image->core()->slice($this->offset, $this->length);
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/StripMetaModifier.php
================================================
core()->native()->getImageProfiles('icc');
// remove meta data
$image->core()->native()->stripImage();
$image->setExif(new Collection());
if ($profiles !== []) {
// re-apply icc profiles
$image->core()->native()->profileImage("icc", $profiles['icc']);
}
return $image;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/TextModifier.php
================================================
processor()->textBlock($this->text, $this->font, $this->position);
$drawText = $this->imagickDrawText($image, $this->font);
$drawStroke = $this->imagickDrawStroke($image, $this->font);
foreach ($image as $frame) {
foreach ($lines as $line) {
foreach ($this->strokeOffsets($this->font) as $offset) {
// Draw the stroke outline under the actual text
$this->maybeDrawTextline($frame, $line, $drawStroke, $offset);
}
// Draw the actual text
$this->maybeDrawTextline($frame, $line, $drawText);
}
}
return $image;
}
/**
* Create an ImagickDraw object to draw text on the image
*
* @throws RuntimeException
* @throws ColorException
* @throws FontException
* @throws ImagickDrawException
* @throws ImagickException
*/
private function imagickDrawText(ImageInterface $image, FontInterface $font): ImagickDraw
{
$color = $this->driver()->handleInput($font->color());
if ($font->hasStrokeEffect() && $color->isTransparent()) {
throw new ColorException(
'The text color must be fully opaque when using the stroke effect.'
);
}
$color = $this->driver()->colorProcessor($image->colorspace())->colorToNative($color);
return $this->processor()->toImagickDraw($font, $color);
}
/**
* Create a ImagickDraw object to draw the outline stroke effect on the Image
*
* @throws RuntimeException
* @throws ColorException
* @throws FontException
* @throws ImagickDrawException
* @throws ImagickException
*/
private function imagickDrawStroke(ImageInterface $image, FontInterface $font): ?ImagickDraw
{
if (!$font->hasStrokeEffect()) {
return null;
}
$color = $this->driver()->handleInput($font->strokeColor());
if ($color->isTransparent()) {
throw new ColorException(
'The stroke color must be fully opaque.'
);
}
$color = $this->driver()->colorProcessor($image->colorspace())->colorToNative($color);
return $this->processor()->toImagickDraw($font, $color);
}
/**
* Maybe draw given line of text on frame instance depending on given
* ImageDraw instance. Optionally move line position by given offset.
*/
private function maybeDrawTextline(
FrameInterface $frame,
Line $textline,
?ImagickDraw $draw = null,
PointInterface $offset = new Point(),
): void {
if ($draw instanceof ImagickDraw) {
$frame->native()->annotateImage(
$draw,
$textline->position()->x() + $offset->x(),
$textline->position()->y() + $offset->y(),
$this->font->angle(),
(string) $textline
);
}
}
/**
* Return imagick font processor
*
* @throws FontException
*/
private function processor(): FontProcessor
{
$processor = $this->driver()->fontProcessor();
if (!($processor instanceof FontProcessor)) {
throw new FontException('Font processor does not match the driver.');
}
return $processor;
}
}
================================================
FILE: src/Drivers/Imagick/Modifiers/TrimModifier.php
================================================
isAnimated()) {
throw new NotSupportedException('Trim modifier cannot be applied to animated images.');
}
$imagick = $image->core()->native();
$imagick->trimImage(($this->tolerance / 100 * $imagick->getQuantum()) / 1.5);
$imagick->setImagePage(0, 0, 0, 0);
return $image;
}
}
================================================
FILE: src/Drivers/Specializable.php
================================================
analyze($this);
}
}
================================================
FILE: src/Drivers/SpecializableDecoder.php
================================================
modify($this);
}
}
================================================
FILE: src/EncodedImage.php
================================================
mediaType;
}
/**
* {@inheritdoc}
*
* @see EncodedImageInterface::mimetype()
*/
public function mimetype(): string
{
return $this->mediaType();
}
/**
* {@inheritdoc}
*
* @see EncodedImageInterface::toDataUri()
*/
public function toDataUri(): string
{
return sprintf('data:%s;base64,%s', $this->mediaType(), base64_encode((string) $this));
}
/**
* Show debug info for the current image
*
* @return array
*/
public function __debugInfo(): array
{
return [
'mediaType' => $this->mediaType(),
'size' => $this->size(),
];
}
}
================================================
FILE: src/Encoders/AutoEncoder.php
================================================
encode(
$this->encoderByMediaType(
$image->origin()->mediaType()
)
);
}
}
================================================
FILE: src/Encoders/AvifEncoder.php
================================================
*/
protected array $options = [];
/**
* Create new encoder instance to encode to format of given file extension
*
* @param null|string|FileExtension $extension Target file extension for example "png"
* @return void
*/
public function __construct(public null|string|FileExtension $extension = null, mixed ...$options)
{
$this->options = $options;
}
/**
* {@inheritdoc}
*
* @see EncoderInterface::encode()
*/
public function encode(ImageInterface $image): EncodedImageInterface
{
$extension = is_null($this->extension) ? $image->origin()->fileExtension() : $this->extension;
return $image->encode(
$this->encoderByFileExtension(
$extension
)
);
}
/**
* Create matching encoder for given file extension
*
* @throws EncoderException
*/
protected function encoderByFileExtension(null|string|FileExtension $extension): EncoderInterface
{
if (empty($extension)) {
throw new EncoderException('No encoder found for empty file extension.');
}
try {
$extension = is_string($extension) ? FileExtension::from(strtolower($extension)) : $extension;
} catch (Error) {
throw new EncoderException('No encoder found for file extension (' . $extension . ').');
}
return $extension->format()->encoder(...$this->options);
}
}
================================================
FILE: src/Encoders/FilePathEncoder.php
================================================
encode(
$this->encoderByFileExtension(
is_null($this->path) ? $image->origin()->fileExtension() : pathinfo($this->path, PATHINFO_EXTENSION)
)
);
}
}
================================================
FILE: src/Encoders/GifEncoder.php
================================================
*/
protected array $options = [];
/**
* Create new encoder instance
*
* @param null|string|MediaType $mediaType Target media type for example "image/jpeg"
* @return void
*/
public function __construct(public null|string|MediaType $mediaType = null, mixed ...$options)
{
$this->options = $options;
}
/**
* {@inheritdoc}
*
* @see EncoderInterface::encode()
*/
public function encode(ImageInterface $image): EncodedImageInterface
{
$mediaType = is_null($this->mediaType) ? $image->origin()->mediaType() : $this->mediaType;
return $image->encode(
$this->encoderByMediaType($mediaType)
);
}
/**
* Return new encoder by given media (MIME) type
*
* @throws EncoderException
*/
protected function encoderByMediaType(string|MediaType $mediaType): EncoderInterface
{
try {
$mediaType = is_string($mediaType) ? MediaType::from($mediaType) : $mediaType;
} catch (Error) {
throw new EncoderException('No encoder found for media type (' . $mediaType . ').');
}
return $mediaType->format()->encoder(...$this->options);
}
}
================================================
FILE: src/Encoders/PngEncoder.php
================================================
pointer = $this->buildFilePointer($data);
}
/**
* Create file object from path in file system
*
* @throws RuntimeException
*/
public static function fromPath(string $path): self
{
return new self(fopen($path, 'r'));
}
/**
* {@inheritdoc}
*
* @see FileInterface::save()
*/
public function save(string $filepath): void
{
$dir = pathinfo($filepath, PATHINFO_DIRNAME);
if (!is_dir($dir)) {
throw new NotWritableException(
"Can't write image to path. Directory does not exist."
);
}
if (!is_writable($dir)) {
throw new NotWritableException(
"Can't write image to path. Directory is not writable."
);
}
if (is_file($filepath) && !is_writable($filepath)) {
throw new NotWritableException(
sprintf("Can't write image. Path (%s) is not writable.", $filepath)
);
}
// write data
$saved = @file_put_contents($filepath, $this->toFilePointer());
if ($saved === false) {
throw new NotWritableException(
sprintf("Can't write image data to path (%s).", $filepath)
);
}
}
/**
* {@inheritdoc}
*
* @see FileInterface::toString()
*/
public function toString(): string
{
return stream_get_contents($this->toFilePointer(), offset: 0);
}
/**
* {@inheritdoc}
*
* @see FileInterface::toFilePointer()
*/
public function toFilePointer()
{
rewind($this->pointer);
return $this->pointer;
}
/**
* {@inheritdoc}
*
* @see FileInterface::size()
*/
public function size(): int
{
$info = fstat($this->toFilePointer());
return intval($info['size']);
}
/**
* {@inheritdoc}
*
* @see FileInterface::__toString()
*/
public function __toString(): string
{
return $this->toString();
}
}
================================================
FILE: src/FileExtension.php
================================================
fileExtension();
}
if ($identifier instanceof MediaType) {
return $identifier->fileExtension();
}
try {
$extension = self::from(strtolower($identifier));
} catch (Error) {
try {
$extension = MediaType::from(strtolower($identifier))->fileExtension();
} catch (Error) {
throw new NotSupportedException('Unable to create file extension from "' . $identifier . '".');
}
}
return $extension;
}
/**
* Try to create media type from given identifier and return null on failure
*
* @param string|Format|MediaType|FileExtension $identifier
* @return FileExtension|null
*/
public static function tryCreate(string|self|Format|MediaType $identifier): ?self
{
try {
return self::create($identifier);
} catch (NotSupportedException) {
return null;
}
}
/**
* Return the matching format for the current file extension
*/
public function format(): Format
{
return match ($this) {
self::JPEG,
self::JPG => Format::JPEG,
self::WEBP => Format::WEBP,
self::GIF => Format::GIF,
self::PNG => Format::PNG,
self::AVIF => Format::AVIF,
self::BMP => Format::BMP,
self::TIF,
self::TIFF => Format::TIFF,
self::JP2,
self::JP2K,
self::J2K,
self::JPF,
self::JPM,
self::JPG2,
self::J2C,
self::JPC,
self::JPX => Format::JP2,
self::HEIC,
self::HEIF => Format::HEIC,
};
}
/**
* Return media types for the current format
*
* @return array
*/
public function mediaTypes(): array
{
return $this->format()->mediaTypes();
}
/**
* Return the first found media type for the current format
*/
public function mediaType(): MediaType
{
return $this->format()->mediaType();
}
}
================================================
FILE: src/Format.php
================================================
format();
}
if ($identifier instanceof FileExtension) {
return $identifier->format();
}
try {
$format = MediaType::from(strtolower($identifier))->format();
} catch (Error) {
try {
$format = FileExtension::from(strtolower($identifier))->format();
} catch (Error) {
throw new NotSupportedException('Unable to create format from "' . $identifier . '".');
}
}
return $format;
}
/**
* Try to create format from given identifier and return null on failure
*
* @param string|Format|MediaType|FileExtension $identifier
* @return Format|null
*/
public static function tryCreate(string|self|MediaType|FileExtension $identifier): ?self
{
try {
return self::create($identifier);
} catch (NotSupportedException) {
return null;
}
}
/**
* Return the possible media (MIME) types for the current format
*
* @return array
*/
public function mediaTypes(): array
{
return array_filter(
MediaType::cases(),
fn(MediaType $mediaType): bool => $mediaType->format() === $this
);
}
/**
* Return the first found media type for the current format
*/
public function mediaType(): MediaType
{
$types = $this->mediaTypes();
return reset($types);
}
/**
* Return the possible file extension for the current format
*
* @return array
*/
public function fileExtensions(): array
{
return array_filter(
FileExtension::cases(),
fn(FileExtension $fileExtension): bool => $fileExtension->format() === $this
);
}
/**
* Return the first found file extension for the current format
*/
public function fileExtension(): FileExtension
{
$extensions = $this->fileExtensions();
return reset($extensions);
}
/**
* Create an encoder instance with given options that matches the format
*/
public function encoder(mixed ...$options): EncoderInterface
{
// get classname of target encoder from current format
$classname = match ($this) {
self::AVIF => AvifEncoder::class,
self::BMP => BmpEncoder::class,
self::GIF => GifEncoder::class,
self::HEIC => HeicEncoder::class,
self::JP2 => Jpeg2000Encoder::class,
self::JPEG => JpegEncoder::class,
self::PNG => PngEncoder::class,
self::TIFF => TiffEncoder::class,
self::WEBP => WebpEncoder::class,
};
// get parameters of target encoder
$parameters = [];
$reflectionClass = new ReflectionClass($classname);
if ($constructor = $reflectionClass->getConstructor()) {
$parameters = array_map(
fn(ReflectionParameter $parameter): string => $parameter->getName(),
$constructor->getParameters(),
);
}
// filter out unavailable options of target encoder
$options = array_filter(
$options,
fn(mixed $key): bool => in_array($key, $parameters),
ARRAY_FILTER_USE_KEY,
);
return new $classname(...$options);
}
}
================================================
FILE: src/Geometry/Bezier.php
================================================
* @implements ArrayAccess
*/
class Bezier implements IteratorAggregate, Countable, ArrayAccess, DrawableInterface
{
use HasBorder;
use HasBackgroundColor;
/**
* Create new bezier instance
*
* @param array $points
* @return void
*/
public function __construct(
protected array $points = [],
protected PointInterface $pivot = new Point()
) {
//
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::position()
*/
public function position(): PointInterface
{
return $this->pivot;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::setPosition()
*/
public function setPosition(PointInterface $position): DrawableInterface
{
$this->pivot = $position;
return $this;
}
/**
* Implement iteration through all points of bezier
*
* @return Traversable
*/
public function getIterator(): Traversable
{
return new ArrayIterator($this->points);
}
/**
* Return current pivot point
*/
public function pivot(): PointInterface
{
return $this->pivot;
}
/**
* Change pivot point to given point
*/
public function setPivot(PointInterface $pivot): self
{
$this->pivot = $pivot;
return $this;
}
/**
* Return first control point of bezier
*/
public function first(): ?PointInterface
{
if ($point = reset($this->points)) {
return $point;
}
return null;
}
/**
* Return second control point of bezier
*/
public function second(): ?PointInterface
{
if (array_key_exists(1, $this->points)) {
return $this->points[1];
}
return null;
}
/**
* Return third control point of bezier
*/
public function third(): ?PointInterface
{
if (array_key_exists(2, $this->points)) {
return $this->points[2];
}
return null;
}
/**
* Return last control point of bezier
*/
public function last(): ?PointInterface
{
if ($point = end($this->points)) {
return $point;
}
return null;
}
/**
* Return bezier's point count
*/
public function count(): int
{
return count($this->points);
}
/**
* Determine if point exists at given offset
*/
public function offsetExists(mixed $offset): bool
{
return array_key_exists($offset, $this->points);
}
/**
* Return point at given offset
*/
public function offsetGet(mixed $offset): mixed
{
return $this->points[$offset];
}
/**
* Set point at given offset
*/
public function offsetSet(mixed $offset, mixed $value): void
{
$this->points[$offset] = $value;
}
/**
* Unset offset at given offset
*/
public function offsetUnset(mixed $offset): void
{
unset($this->points[$offset]);
}
/**
* Add given point to bezier
*/
public function addPoint(PointInterface $point): self
{
$this->points[] = $point;
return $this;
}
/**
* Return array of all x/y values of all points of bezier
*
* @return array
*/
public function toArray(): array
{
$coordinates = [];
foreach ($this->points as $point) {
$coordinates[] = $point->x();
$coordinates[] = $point->y();
}
return $coordinates;
}
}
================================================
FILE: src/Geometry/Circle.php
================================================
setWidth($diameter);
$this->setHeight($diameter);
return $this;
}
/**
* Get diameter of circle
*/
public function diameter(): int
{
return $this->width();
}
/**
* Set radius of circle
*/
public function setRadius(int $radius): self
{
return $this->setDiameter(intval($radius * 2));
}
/**
* Get radius of circle
*/
public function radius(): int
{
return intval(round($this->diameter() / 2));
}
}
================================================
FILE: src/Geometry/Ellipse.php
================================================
pivot;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::setPosition()
*/
public function setPosition(PointInterface $position): self
{
$this->pivot = $position;
return $this;
}
/**
* Return pivot point of Ellipse
*/
public function pivot(): PointInterface
{
return $this->pivot;
}
/**
* Set size of Ellipse
*/
public function setSize(int $width, int $height): self
{
return $this->setWidth($width)->setHeight($height);
}
/**
* Set width of Ellipse
*/
public function setWidth(int $width): self
{
$this->width = $width;
return $this;
}
/**
* Set height of Ellipse
*/
public function setHeight(int $height): self
{
$this->height = $height;
return $this;
}
/**
* Get width of Ellipse
*/
public function width(): int
{
return $this->width;
}
/**
* Get height of Ellipse
*/
public function height(): int
{
return $this->height;
}
}
================================================
FILE: src/Geometry/Factories/BezierFactory.php
================================================
bezier = is_a($init, Bezier::class) ? $init : new Bezier([]);
if (is_callable($init)) {
$init($this);
}
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::init()
*/
public static function init(null|Closure|DrawableInterface $init = null): self
{
return new self($init);
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::create()
*/
public function create(): DrawableInterface
{
return $this->bezier;
}
/**
* Add a point to the bezier to be produced
*/
public function point(int $x, int $y): self
{
$this->bezier->addPoint(new Point($x, $y));
return $this;
}
/**
* Set the background color of the bezier to be produced
*/
public function background(mixed $color): self
{
$this->bezier->setBackgroundColor($color);
return $this;
}
/**
* Set the border color & border size of the bezier to be produced
*/
public function border(mixed $color, int $size = 1): self
{
$this->bezier->setBorder($color, $size);
return $this;
}
/**
* Produce the bezier
*/
public function __invoke(): Bezier
{
return $this->bezier;
}
}
================================================
FILE: src/Geometry/Factories/CircleFactory.php
================================================
circle = is_a($init, Circle::class) ? $init : new Circle(0);
$this->circle->setPosition($pivot);
if (is_callable($init)) {
$init($this);
}
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::init()
*/
public static function init(null|Closure|DrawableInterface $init = null): self
{
return new self(init: $init);
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::create()
*/
public function create(): DrawableInterface
{
return $this->circle;
}
/**
* Set the radius of the circle to be produced
*/
public function radius(int $radius): self
{
$this->circle->setSize($radius * 2, $radius * 2);
return $this;
}
/**
* Set the diameter of the circle to be produced
*/
public function diameter(int $diameter): self
{
$this->circle->setSize($diameter, $diameter);
return $this;
}
/**
* Set the background color of the circle to be produced
*/
public function background(mixed $color): self
{
$this->circle->setBackgroundColor($color);
return $this;
}
/**
* Set the border color & border size of the ellipse to be produced
*/
public function border(mixed $color, int $size = 1): self
{
$this->circle->setBorder($color, $size);
return $this;
}
/**
* Produce the circle
*/
public function __invoke(): Circle
{
return $this->circle;
}
}
================================================
FILE: src/Geometry/Factories/Drawable.php
================================================
ellipse = is_a($init, Ellipse::class) ? $init : new Ellipse(0, 0);
$this->ellipse->setPosition($pivot);
if (is_callable($init)) {
$init($this);
}
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::init()
*/
public static function init(null|Closure|DrawableInterface $init = null): self
{
return new self(init: $init);
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::create()
*/
public function create(): DrawableInterface
{
return $this->ellipse;
}
/**
* Set the size of the ellipse to be produced
*/
public function size(int $width, int $height): self
{
$this->ellipse->setSize($width, $height);
return $this;
}
/**
* Set the width of the ellipse to be produced
*/
public function width(int $width): self
{
$this->ellipse->setWidth($width);
return $this;
}
/**
* Set the height of the ellipse to be produced
*/
public function height(int $height): self
{
$this->ellipse->setHeight($height);
return $this;
}
/**
* Set the background color of the ellipse to be produced
*/
public function background(mixed $color): self
{
$this->ellipse->setBackgroundColor($color);
return $this;
}
/**
* Set the border color & border size of the ellipse to be produced
*/
public function border(mixed $color, int $size = 1): self
{
$this->ellipse->setBorder($color, $size);
return $this;
}
/**
* Produce the ellipse
*/
public function __invoke(): Ellipse
{
return $this->ellipse;
}
}
================================================
FILE: src/Geometry/Factories/LineFactory.php
================================================
line = is_a($init, Line::class) ? $init : new Line(new Point(), new Point());
if (is_callable($init)) {
$init($this);
}
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::init()
*/
public static function init(null|Closure|DrawableInterface $init = null): self
{
return new self($init);
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::create()
*/
public function create(): DrawableInterface
{
return $this->line;
}
/**
* Set the color of the line to be produced
*/
public function color(mixed $color): self
{
$this->line->setBackgroundColor($color);
$this->line->setBorderColor($color);
return $this;
}
/**
* Set the (background) color of the line to be produced
*/
public function background(mixed $color): self
{
$this->line->setBackgroundColor($color);
$this->line->setBorderColor($color);
return $this;
}
/**
* Set the border size & border color of the line to be produced
*/
public function border(mixed $color, int $size = 1): self
{
$this->line->setBackgroundColor($color);
$this->line->setBorderColor($color);
$this->line->setWidth($size);
return $this;
}
/**
* Set the width of the line to be produced
*/
public function width(int $size): self
{
$this->line->setWidth($size);
return $this;
}
/**
* Set the coordinates of the starting point of the line to be produced
*/
public function from(int $x, int $y): self
{
$this->line->setStart(new Point($x, $y));
return $this;
}
/**
* Set the coordinates of the end point of the line to be produced
*/
public function to(int $x, int $y): self
{
$this->line->setEnd(new Point($x, $y));
return $this;
}
/**
* Produce the line
*/
public function __invoke(): Line
{
return $this->line;
}
}
================================================
FILE: src/Geometry/Factories/PolygonFactory.php
================================================
polygon = is_a($init, Polygon::class) ? $init : new Polygon([]);
if (is_callable($init)) {
$init($this);
}
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::init()
*/
public static function init(null|Closure|DrawableInterface $init = null): self
{
return new self($init);
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::create()
*/
public function create(): DrawableInterface
{
return $this->polygon;
}
/**
* Add a point to the polygon to be produced
*/
public function point(int $x, int $y): self
{
$this->polygon->addPoint(new Point($x, $y));
return $this;
}
/**
* Set the background color of the polygon to be produced
*/
public function background(mixed $color): self
{
$this->polygon->setBackgroundColor($color);
return $this;
}
/**
* Set the border color & border size of the polygon to be produced
*/
public function border(mixed $color, int $size = 1): self
{
$this->polygon->setBorder($color, $size);
return $this;
}
/**
* Produce the polygon
*/
public function __invoke(): Polygon
{
return $this->polygon;
}
}
================================================
FILE: src/Geometry/Factories/RectangleFactory.php
================================================
rectangle = is_a($init, Rectangle::class) ? $init : new Rectangle(0, 0, $pivot);
$this->rectangle->setPosition($pivot);
if (is_callable($init)) {
$init($this);
}
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::init()
*/
public static function init(null|Closure|DrawableInterface $init = null): self
{
return new self(init: $init);
}
/**
* {@inheritdoc}
*
* @see DrawableFactoryInterface::create()
*/
public function create(): DrawableInterface
{
return $this->rectangle;
}
/**
* Set the size of the rectangle to be produced
*/
public function size(int $width, int $height): self
{
$this->rectangle->setSize($width, $height);
return $this;
}
/**
* Set the width of the rectangle to be produced
*/
public function width(int $width): self
{
$this->rectangle->setWidth($width);
return $this;
}
/**
* Set the height of the rectangle to be produced
*/
public function height(int $height): self
{
$this->rectangle->setHeight($height);
return $this;
}
/**
* Set the background color of the rectangle to be produced
*/
public function background(mixed $color): self
{
$this->rectangle->setBackgroundColor($color);
return $this;
}
/**
* Set the border color & border size of the rectangle to be produced
*/
public function border(mixed $color, int $size = 1): self
{
$this->rectangle->setBorder($color, $size);
return $this;
}
/**
* Produce the rectangle
*/
public function __invoke(): Rectangle
{
return $this->rectangle;
}
}
================================================
FILE: src/Geometry/Line.php
================================================
start;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::setPosition()
*/
public function setPosition(PointInterface $position): DrawableInterface
{
$this->start = $position;
return $this;
}
/**
* Return line width
*/
public function width(): int
{
return $this->width;
}
/**
* Set line width
*/
public function setWidth(int $width): self
{
$this->width = $width;
return $this;
}
/**
* Get starting point of line
*/
public function start(): PointInterface
{
return $this->start;
}
/**
* get end point of line
*/
public function end(): PointInterface
{
return $this->end;
}
/**
* Set starting point of line
*/
public function setStart(PointInterface $start): self
{
$this->start = $start;
return $this;
}
/**
* Set starting point of line by coordinates
*/
public function from(int $x, int $y): self
{
$this->start()->setX($x);
$this->start()->setY($y);
return $this;
}
/**
* Set end point of line by coordinates
*/
public function to(int $x, int $y): self
{
$this->end()->setX($x);
$this->end()->setY($y);
return $this;
}
/**
* Set end point of line
*/
public function setEnd(PointInterface $end): self
{
$this->end = $end;
return $this;
}
}
================================================
FILE: src/Geometry/Pixel.php
================================================
background = $background;
return $this;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::backgroundColor()
*/
public function backgroundColor(): ColorInterface
{
return $this->background;
}
}
================================================
FILE: src/Geometry/Point.php
================================================
*/
class Point implements PointInterface, IteratorAggregate
{
/**
* Create new point instance
*
* @return void
*/
public function __construct(
protected int $x = 0,
protected int $y = 0
) {
//
}
/**
* {@inheritdoc}
*
* @see IteratorAggregate::getIterator()
*/
public function getIterator(): Traversable
{
return new ArrayIterator([$this->x, $this->y]);
}
/**
* {@inheritdoc}
*
* @see PointInterface::setX()
*/
public function setX(int $x): self
{
$this->x = $x;
return $this;
}
/**
* {@inheritdoc}
*
* @see PointInterface::x()
*/
public function x(): int
{
return $this->x;
}
/**
* {@inheritdoc}
*
* @see PointInterface::setY()
*/
public function setY(int $y): self
{
$this->y = $y;
return $this;
}
/**
* {@inheritdoc}
*
* @see PointInterface::y()
*/
public function y(): int
{
return $this->y;
}
/**
* {@inheritdoc}
*
* @see PointInterface::moveX()
*/
public function moveX(int $value): self
{
$this->x += $value;
return $this;
}
/**
* {@inheritdoc}
*
* @see PointInterface::moveY()
*/
public function moveY(int $value): self
{
$this->y += $value;
return $this;
}
/**
* {@inheritdoc}
*
* @see PointInterface::move()
*/
public function move(int $x, int $y): self
{
return $this->moveX($x)->moveY($y);
}
/**
* {@inheritdoc}
*
* @see PointInterface::setPosition()
*/
public function setPosition(int $x, int $y): self
{
$this->setX($x);
$this->setY($y);
return $this;
}
/**
* {@inheritdoc}
*
* @see PointInterface::rotate()
*/
public function rotate(float $angle, PointInterface $pivot): self
{
$sin = round(sin(deg2rad($angle)), 6);
$cos = round(cos(deg2rad($angle)), 6);
return $this->setPosition(
intval($cos * ($this->x() - $pivot->x()) - $sin * ($this->y() - $pivot->y()) + $pivot->x()),
intval($sin * ($this->x() - $pivot->x()) + $cos * ($this->y() - $pivot->y()) + $pivot->y())
);
}
}
================================================
FILE: src/Geometry/Polygon.php
================================================
* @implements ArrayAccess
*/
class Polygon implements IteratorAggregate, Countable, ArrayAccess, DrawableInterface
{
use HasBorder;
use HasBackgroundColor;
/**
* Create new polygon instance
*
* @param array $points
* @return void
*/
public function __construct(
protected array $points = [],
protected PointInterface $pivot = new Point()
) {
//
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::position()
*/
public function position(): PointInterface
{
return $this->pivot;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::setPosition()
*/
public function setPosition(PointInterface $position): self
{
$this->pivot = $position;
return $this;
}
/**
* Implement iteration through all points of polygon
*
* @return Traversable
*/
public function getIterator(): Traversable
{
return new ArrayIterator($this->points);
}
/**
* Return current pivot point
*/
public function pivot(): PointInterface
{
return $this->pivot;
}
/**
* Change pivot point to given point
*/
public function setPivot(PointInterface $pivot): self
{
$this->pivot = $pivot;
return $this;
}
/**
* Return first point of polygon
*/
public function first(): ?PointInterface
{
if ($point = reset($this->points)) {
return $point;
}
return null;
}
/**
* Return last point of polygon
*/
public function last(): ?PointInterface
{
if ($point = end($this->points)) {
return $point;
}
return null;
}
/**
* Return polygon's point count
*/
public function count(): int
{
return count($this->points);
}
/**
* Determine if point exists at given offset
*/
public function offsetExists(mixed $offset): bool
{
return array_key_exists($offset, $this->points);
}
/**
* Return point at given offset
*/
public function offsetGet(mixed $offset): mixed
{
return $this->points[$offset];
}
/**
* Set point at given offset
*/
public function offsetSet(mixed $offset, mixed $value): void
{
$this->points[$offset] = $value;
}
/**
* Unset offset at given offset
*/
public function offsetUnset(mixed $offset): void
{
unset($this->points[$offset]);
}
/**
* Add given point to polygon
*/
public function addPoint(PointInterface $point): self
{
$this->points[] = $point;
return $this;
}
/**
* Calculate total horizontal span of polygon
*/
public function width(): int
{
return abs($this->mostLeftPoint()->x() - $this->mostRightPoint()->x());
}
/**
* Calculate total vertical span of polygon
*/
public function height(): int
{
return abs($this->mostBottomPoint()->y() - $this->mostTopPoint()->y());
}
/**
* Return most left point of all points in polygon
*/
public function mostLeftPoint(): PointInterface
{
$points = $this->points;
usort($points, function (PointInterface $a, PointInterface $b): int {
if ($a->x() === $b->x()) {
return 0;
}
return $a->x() < $b->x() ? -1 : 1;
});
return $points[0];
}
/**
* Return most right point in polygon
*/
public function mostRightPoint(): PointInterface
{
$points = $this->points;
usort($points, function (PointInterface $a, PointInterface $b): int {
if ($a->x() === $b->x()) {
return 0;
}
return $a->x() > $b->x() ? -1 : 1;
});
return $points[0];
}
/**
* Return most top point in polygon
*/
public function mostTopPoint(): PointInterface
{
$points = $this->points;
usort($points, function (PointInterface $a, PointInterface $b): int {
if ($a->y() === $b->y()) {
return 0;
}
return $a->y() > $b->y() ? -1 : 1;
});
return $points[0];
}
/**
* Return most bottom point in polygon
*/
public function mostBottomPoint(): PointInterface
{
$points = $this->points;
usort($points, function (PointInterface $a, PointInterface $b): int {
if ($a->y() === $b->y()) {
return 0;
}
return $a->y() < $b->y() ? -1 : 1;
});
return $points[0];
}
/**
* Return point in absolute center of the polygon
*/
public function centerPoint(): PointInterface
{
return new Point(
$this->mostRightPoint()->x() - (intval(round($this->width() / 2))),
$this->mostTopPoint()->y() - (intval(round($this->height() / 2)))
);
}
/**
* Align all points of polygon horizontally to given position around pivot point
*/
public function align(string $position): self
{
switch (strtolower($position)) {
case 'center':
case 'middle':
$diff = $this->centerPoint()->x() - $this->pivot()->x();
break;
case 'right':
$diff = $this->mostRightPoint()->x() - $this->pivot()->x();
break;
default:
case 'left':
$diff = $this->mostLeftPoint()->x() - $this->pivot()->x();
break;
}
foreach ($this->points as $point) {
$point->setX(
intval($point->x() - $diff)
);
}
return $this;
}
/**
* Align all points of polygon vertically to given position around pivot point
*/
public function valign(string $position): self
{
switch (strtolower($position)) {
case 'center':
case 'middle':
$diff = $this->centerPoint()->y() - $this->pivot()->y();
break;
case 'top':
$diff = $this->mostTopPoint()->y() - $this->pivot()->y() - $this->height();
break;
default:
case 'bottom':
$diff = $this->mostBottomPoint()->y() - $this->pivot()->y() + $this->height();
break;
}
foreach ($this->points as $point) {
$point->setY(
intval($point->y() - $diff),
);
}
return $this;
}
/**
* Rotate points of polygon around pivot point with given angle
*/
public function rotate(float $angle): self
{
$sin = sin(deg2rad($angle));
$cos = cos(deg2rad($angle));
foreach ($this->points as $point) {
// translate point to pivot
$point->setX(
intval($point->x() - $this->pivot()->x()),
);
$point->setY(
intval($point->y() - $this->pivot()->y()),
);
// rotate point
$x = $point->x() * $cos - $point->y() * $sin;
$y = $point->x() * $sin + $point->y() * $cos;
// translate point back
$point->setX(
intval($x + $this->pivot()->x()),
);
$point->setY(
intval($y + $this->pivot()->y()),
);
}
return $this;
}
/**
* Move all points by given amount on the x-axis
*/
public function movePointsX(int $amount): self
{
foreach ($this->points as $point) {
$point->moveX($amount);
}
return $this;
}
/**
* Move all points by given amount on the y-axis
*/
public function movePointsY(int $amount): self
{
foreach ($this->points as $point) {
$point->moveY($amount);
}
return $this;
}
/**
* Return array of all x/y values of all points of polygon
*
* @return array
*/
public function toArray(): array
{
$coordinates = [];
foreach ($this->points as $point) {
$coordinates[] = $point->x();
$coordinates[] = $point->y();
}
return $coordinates;
}
}
================================================
FILE: src/Geometry/Rectangle.php
================================================
addPoint(new Point($this->pivot->x(), $this->pivot->y()));
$this->addPoint(new Point($this->pivot->x() + $width, $this->pivot->y()));
$this->addPoint(new Point($this->pivot->x() + $width, $this->pivot->y() - $height));
$this->addPoint(new Point($this->pivot->x(), $this->pivot->y() - $height));
}
/**
* Set size of rectangle
*/
public function setSize(int $width, int $height): self
{
return $this->setWidth($width)->setHeight($height);
}
/**
* Set width of rectangle
*/
public function setWidth(int $width): self
{
$this[1]->setX($this[0]->x() + $width);
$this[2]->setX($this[3]->x() + $width);
return $this;
}
/**
* Set height of rectangle
*/
public function setHeight(int $height): self
{
$this[2]->setY($this[1]->y() + $height);
$this[3]->setY($this[0]->y() + $height);
return $this;
}
/**
* Return pivot point of rectangle
*/
public function pivot(): PointInterface
{
return $this->pivot;
}
/**
* Set pivot point of rectangle
*/
public function setPivot(PointInterface $pivot): self
{
$this->pivot = $pivot;
return $this;
}
/**
* Move pivot to the given position in the rectangle and adjust the new
* position by given offset values.
*/
public function movePivot(string $position, int $offset_x = 0, int $offset_y = 0): self
{
switch (strtolower($position)) {
case 'top':
case 'top-center':
case 'top-middle':
case 'center-top':
case 'middle-top':
$x = intval(round($this->width() / 2)) + $offset_x;
$y = $offset_y;
break;
case 'top-right':
case 'right-top':
$x = $this->width() - $offset_x;
$y = $offset_y;
break;
case 'left':
case 'left-center':
case 'left-middle':
case 'center-left':
case 'middle-left':
$x = $offset_x;
$y = intval(round($this->height() / 2)) + $offset_y;
break;
case 'right':
case 'right-center':
case 'right-middle':
case 'center-right':
case 'middle-right':
$x = $this->width() - $offset_x;
$y = intval(round($this->height() / 2)) + $offset_y;
break;
case 'bottom-left':
case 'left-bottom':
$x = $offset_x;
$y = $this->height() - $offset_y;
break;
case 'bottom':
case 'bottom-center':
case 'bottom-middle':
case 'center-bottom':
case 'middle-bottom':
$x = intval(round($this->width() / 2)) + $offset_x;
$y = $this->height() - $offset_y;
break;
case 'bottom-right':
case 'right-bottom':
$x = $this->width() - $offset_x;
$y = $this->height() - $offset_y;
break;
case 'center':
case 'middle':
case 'center-center':
case 'middle-middle':
$x = intval(round($this->width() / 2)) + $offset_x;
$y = intval(round($this->height() / 2)) + $offset_y;
break;
default:
case 'top-left':
case 'left-top':
$x = $offset_x;
$y = $offset_y;
break;
}
$this->pivot->setPosition($x, $y);
return $this;
}
/**
* Align pivot relative to given size at given position
*/
public function alignPivotTo(SizeInterface $size, string $position): self
{
$reference = new self($size->width(), $size->height());
$reference->movePivot($position);
$this->movePivot($position)->setPivot(
$reference->relativePositionTo($this)
);
return $this;
}
/**
* Return relative position to given rectangle
*/
public function relativePositionTo(SizeInterface $rectangle): PointInterface
{
return new Point(
$this->pivot()->x() - $rectangle->pivot()->x(),
$this->pivot()->y() - $rectangle->pivot()->y()
);
}
/**
* Return aspect ration of rectangle
*/
public function aspectRatio(): float
{
return $this->width() / $this->height();
}
/**
* Determine if rectangle fits into given rectangle
*/
public function fitsInto(SizeInterface $size): bool
{
if ($this->width() > $size->width()) {
return false;
}
if ($this->height() > $size->height()) {
return false;
}
return true;
}
/**
* Determine if rectangle has landscape format
*/
public function isLandscape(): bool
{
return $this->width() > $this->height();
}
/**
* Determine if rectangle has landscape format
*/
public function isPortrait(): bool
{
return $this->width() < $this->height();
}
/**
* Return most top left point of rectangle
*/
public function topLeftPoint(): PointInterface
{
return $this->points[0];
}
/**
* Return bottom right point of rectangle
*/
public function bottomRightPoint(): PointInterface
{
return $this->points[2];
}
/**
* @see SizeInterface::resize()
*
* @throws GeometryException
*/
public function resize(?int $width = null, ?int $height = null): SizeInterface
{
return $this->resizer($width, $height)->resize($this);
}
/**
* @see SizeInterface::resizeDown()
*
* @throws GeometryException
*/
public function resizeDown(?int $width = null, ?int $height = null): SizeInterface
{
return $this->resizer($width, $height)->resizeDown($this);
}
/**
* @see SizeInterface::scale()
*
* @throws GeometryException
*/
public function scale(?int $width = null, ?int $height = null): SizeInterface
{
return $this->resizer($width, $height)->scale($this);
}
/**
* @see SizeInterface::scaleDown()
*
* @throws GeometryException
*/
public function scaleDown(?int $width = null, ?int $height = null): SizeInterface
{
return $this->resizer($width, $height)->scaleDown($this);
}
/**
* @see SizeInterface::cover()
*
* @throws GeometryException
*/
public function cover(int $width, int $height): SizeInterface
{
return $this->resizer($width, $height)->cover($this);
}
/**
* @see SizeInterface::contain()
*
* @throws GeometryException
*/
public function contain(int $width, int $height): SizeInterface
{
return $this->resizer($width, $height)->contain($this);
}
/**
* @see SizeInterface::containMax()
*
* @throws GeometryException
*/
public function containMax(int $width, int $height): SizeInterface
{
return $this->resizer($width, $height)->containDown($this);
}
/**
* Create resizer instance with given target size
*
* @throws GeometryException
*/
protected function resizer(?int $width = null, ?int $height = null): RectangleResizer
{
return new RectangleResizer($width, $height);
}
/**
* Show debug info for the current rectangle
*
* @return array
*/
public function __debugInfo(): array
{
return [
'width' => $this->width(),
'height' => $this->height(),
'pivot' => $this->pivot,
];
}
}
================================================
FILE: src/Geometry/Tools/RectangleResizer.php
================================================
width);
}
/**
* Return target width of resizer if available
*/
protected function getTargetWidth(): ?int
{
return $this->hasTargetWidth() ? $this->width : null;
}
/**
* Determine if resize has target height
*/
protected function hasTargetHeight(): bool
{
return is_int($this->height);
}
/**
* Return target width of resizer if available
*/
protected function getTargetHeight(): ?int
{
return $this->hasTargetHeight() ? $this->height : null;
}
/**
* Return target size object
*
* @throws GeometryException
*/
protected function getTargetSize(): SizeInterface
{
if (!$this->hasTargetWidth() || !$this->hasTargetHeight()) {
throw new GeometryException('Target size needs width and height.');
}
return new Rectangle($this->width, $this->height);
}
/**
* Set target width of resizer
*/
public function toWidth(int $width): self
{
$this->width = $width;
return $this;
}
/**
* Set target height of resizer
*/
public function toHeight(int $height): self
{
$this->height = $height;
return $this;
}
/**
* Set target size to given size object
*/
public function toSize(SizeInterface $size): self
{
$this->width = $size->width();
$this->height = $size->height();
return $this;
}
/**
* Get proportinal width
*/
protected function getProportionalWidth(SizeInterface $size): int
{
if (!$this->hasTargetHeight()) {
return $size->width();
}
return max([1, (int) round($this->height * $size->aspectRatio())]);
}
/**
* Get proportinal height
*/
protected function getProportionalHeight(SizeInterface $size): int
{
if (!$this->hasTargetWidth()) {
return $size->height();
}
return max([1, (int) round($this->width / $size->aspectRatio())]);
}
/**
* Resize given size to target size of the resizer
*/
public function resize(SizeInterface $size): SizeInterface
{
$resized = new Rectangle($size->width(), $size->height());
if ($width = $this->getTargetWidth()) {
$resized->setWidth($width);
}
if ($height = $this->getTargetHeight()) {
$resized->setHeight($height);
}
return $resized;
}
/**
* Resize given size to target size of the resizer but do not exceed original size
*/
public function resizeDown(SizeInterface $size): SizeInterface
{
$resized = new Rectangle($size->width(), $size->height());
if ($width = $this->getTargetWidth()) {
$resized->setWidth(
min($width, $size->width())
);
}
if ($height = $this->getTargetHeight()) {
$resized->setHeight(
min($height, $size->height())
);
}
return $resized;
}
/**
* Resize given size to target size proportinally
*/
public function scale(SizeInterface $size): SizeInterface
{
$resized = new Rectangle($size->width(), $size->height());
if ($this->hasTargetWidth() && $this->hasTargetHeight()) {
$resized->setWidth(min(
$this->getProportionalWidth($size),
$this->getTargetWidth()
));
$resized->setHeight(min(
$this->getProportionalHeight($size),
$this->getTargetHeight()
));
} elseif ($this->hasTargetWidth()) {
$resized->setWidth($this->getTargetWidth());
$resized->setHeight($this->getProportionalHeight($size));
} elseif ($this->hasTargetHeight()) {
$resized->setWidth($this->getProportionalWidth($size));
$resized->setHeight($this->getTargetHeight());
}
return $resized;
}
/**
* Resize given size to target size proportinally but do not exceed original size
*/
public function scaleDown(SizeInterface $size): SizeInterface
{
$resized = new Rectangle($size->width(), $size->height());
if ($this->hasTargetWidth() && $this->hasTargetHeight()) {
$resized->setWidth(min(
$this->getProportionalWidth($size),
$this->getTargetWidth(),
$size->width()
));
$resized->setHeight(min(
$this->getProportionalHeight($size),
$this->getTargetHeight(),
$size->height()
));
} elseif ($this->hasTargetWidth()) {
$resized->setWidth(min(
$this->getTargetWidth(),
$size->width()
));
$resized->setHeight(min(
$this->getProportionalHeight($size),
$size->height()
));
} elseif ($this->hasTargetHeight()) {
$resized->setWidth(min(
$this->getProportionalWidth($size),
$size->width()
));
$resized->setHeight(min(
$this->getTargetHeight(),
$size->height()
));
}
return $resized;
}
/**
* Scale given size to cover target size
*
* @param SizeInterface $size Size to be resized
* @throws GeometryException
*/
public function cover(SizeInterface $size): SizeInterface
{
$resized = new Rectangle($size->width(), $size->height());
// auto height
$resized->setWidth($this->getTargetWidth());
$resized->setHeight($this->getProportionalHeight($size));
if ($resized->fitsInto($this->getTargetSize())) {
// auto width
$resized->setWidth($this->getProportionalWidth($size));
$resized->setHeight($this->getTargetHeight());
}
return $resized;
}
/**
* Scale given size to contain target size
*
* @param SizeInterface $size Size to be resized
* @throws GeometryException
*/
public function contain(SizeInterface $size): SizeInterface
{
$resized = new Rectangle($size->width(), $size->height());
// auto height
$resized->setWidth($this->getTargetWidth());
$resized->setHeight($this->getProportionalHeight($size));
if (!$resized->fitsInto($this->getTargetSize())) {
// auto width
$resized->setWidth($this->getProportionalWidth($size));
$resized->setHeight($this->getTargetHeight());
}
return $resized;
}
/**
* Scale given size to contain target size but prevent upsizing
*
* @param SizeInterface $size Size to be resized
* @throws GeometryException
*/
public function containDown(SizeInterface $size): SizeInterface
{
$resized = new Rectangle($size->width(), $size->height());
// auto height
$resized->setWidth(
min($size->width(), $this->getTargetWidth())
);
$resized->setHeight(
min($size->height(), $this->getProportionalHeight($size))
);
if (!$resized->fitsInto($this->getTargetSize())) {
// auto width
$resized->setWidth(
min($size->width(), $this->getProportionalWidth($size))
);
$resized->setHeight(
min($size->height(), $this->getTargetHeight())
);
}
return $resized;
}
/**
* Crop target size out of given size at given position (i.e. move the pivot point)
*/
public function crop(SizeInterface $size, string $position = 'top-left'): SizeInterface
{
return $this->resize($size)->alignPivotTo(
$size->movePivot($position),
$position
);
}
}
================================================
FILE: src/Geometry/Traits/HasBackgroundColor.php
================================================
backgroundColor = $color;
return $this;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::backgroundColor()
*/
public function backgroundColor(): mixed
{
return $this->backgroundColor;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::hasBackgroundColor()
*/
public function hasBackgroundColor(): bool
{
return !empty($this->backgroundColor);
}
}
================================================
FILE: src/Geometry/Traits/HasBorder.php
================================================
setBorderSize($size)->setBorderColor($color);
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::setBorderSize()
*/
public function setBorderSize(int $size): self
{
$this->borderSize = $size;
return $this;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::borderSize()
*/
public function borderSize(): int
{
return $this->borderSize;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::setBorderColor()
*/
public function setBorderColor(mixed $color): self
{
$this->borderColor = $color;
return $this;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::borderColor()
*/
public function borderColor(): mixed
{
return $this->borderColor;
}
/**
* {@inheritdoc}
*
* @see DrawableInterface::hasBorder()
*/
public function hasBorder(): bool
{
return $this->borderSize > 0 && !is_null($this->borderColor);
}
}
================================================
FILE: src/Image.php
================================================
origin = new Origin();
}
/**
* {@inheritdoc}
*
* @see ImageInterface::driver()
*/
public function driver(): DriverInterface
{
return $this->driver;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::core()
*/
public function core(): CoreInterface
{
return $this->core;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::origin()
*/
public function origin(): Origin
{
return $this->origin;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::setOrigin()
*/
public function setOrigin(Origin $origin): ImageInterface
{
$this->origin = $origin;
return $this;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::count()
*/
public function count(): int
{
return $this->core->count();
}
/**
* Implementation of IteratorAggregate
*
* @return Traversable
*/
public function getIterator(): Traversable
{
return $this->core;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::isAnimated()
*/
public function isAnimated(): bool
{
return $this->count() > 1;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::removeAnimation(
*/
public function removeAnimation(int|string $position = 0): ImageInterface
{
return $this->modify(new RemoveAnimationModifier($position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::sliceAnimation()
*/
public function sliceAnimation(int $offset = 0, ?int $length = null): ImageInterface
{
return $this->modify(new SliceAnimationModifier($offset, $length));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::loops()
*/
public function loops(): int
{
return $this->core->loops();
}
/**
* {@inheritdoc}
*
* @see ImageInterface::setLoops()
*/
public function setLoops(int $loops): ImageInterface
{
$this->core->setLoops($loops);
return $this;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::exif()
*/
public function exif(?string $query = null): mixed
{
return is_null($query) ? $this->exif : $this->exif->get($query);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::setExif()
*/
public function setExif(CollectionInterface $exif): ImageInterface
{
$this->exif = $exif;
return $this;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::modify()
*/
public function modify(ModifierInterface $modifier): ImageInterface
{
return $this->driver->specialize($modifier)->apply($this);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::analyze()
*/
public function analyze(AnalyzerInterface $analyzer): mixed
{
return $this->driver->specialize($analyzer)->analyze($this);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::encode()
*/
public function encode(EncoderInterface $encoder = new AutoEncoder()): EncodedImageInterface
{
return $this->driver->specialize($encoder)->encode($this);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::save()
*/
public function save(?string $path = null, mixed ...$options): ImageInterface
{
$path = is_null($path) ? $this->origin()->filePath() : $path;
if (is_null($path)) {
throw new EncoderException('Could not determine file path to save.');
}
try {
// try to determine encoding format by file extension of the path
$encoded = $this->encodeByPath($path, ...$options);
} catch (EncoderException) {
// fallback to encoding format by media type
$encoded = $this->encodeByMediaType(null, ...$options);
}
$encoded->save($path);
return $this;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::width()
*/
public function width(): int
{
return $this->analyze(new WidthAnalyzer());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::height()
*/
public function height(): int
{
return $this->analyze(new HeightAnalyzer());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::size()
*/
public function size(): SizeInterface
{
return new Rectangle($this->width(), $this->height());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::colorspace()
*/
public function colorspace(): ColorspaceInterface
{
return $this->analyze(new ColorspaceAnalyzer());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::setColorspace()
*/
public function setColorspace(string|ColorspaceInterface $colorspace): ImageInterface
{
return $this->modify(new ColorspaceModifier($colorspace));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::resolution()
*/
public function resolution(): ResolutionInterface
{
return $this->analyze(new ResolutionAnalyzer());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::setResolution()
*/
public function setResolution(float $x, float $y): ImageInterface
{
return $this->modify(new ResolutionModifier($x, $y));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::pickColor()
*/
public function pickColor(int $x, int $y, int $frame_key = 0): ColorInterface
{
return $this->analyze(new PixelColorAnalyzer($x, $y, $frame_key));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::pickColors()
*/
public function pickColors(int $x, int $y): CollectionInterface
{
return $this->analyze(new PixelColorsAnalyzer($x, $y));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::blendingColor()
*/
public function blendingColor(): ColorInterface
{
return $this->driver()->handleInput(
$this->driver()->config()->blendingColor
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::setBlendingColor()
*/
public function setBlendingColor(mixed $color): ImageInterface
{
$this->driver()->config()->setOptions(
blendingColor: $this->driver()->handleInput($color)
);
return $this;
}
/**
* {@inheritdoc}
*
* @see ImageInterface::blendTransparency()
*/
public function blendTransparency(mixed $color = null): ImageInterface
{
return $this->modify(new BlendTransparencyModifier($color));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::profile()
*/
public function profile(): ProfileInterface
{
return $this->analyze(new ProfileAnalyzer());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::setProfile()
*/
public function setProfile(ProfileInterface $profile): ImageInterface
{
return $this->modify(new ProfileModifier($profile));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::removeProfile()
*/
public function removeProfile(): ImageInterface
{
return $this->modify(new ProfileRemovalModifier());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::reduceColors()
*/
public function reduceColors(int $limit, mixed $background = 'transparent'): ImageInterface
{
return $this->modify(new QuantizeColorsModifier($limit, $background));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::sharpen()
*/
public function sharpen(int $amount = 10): ImageInterface
{
return $this->modify(new SharpenModifier($amount));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::invert()
*/
public function invert(): ImageInterface
{
return $this->modify(new InvertModifier());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::pixelate()
*/
public function pixelate(int $size): ImageInterface
{
return $this->modify(new PixelateModifier($size));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::greyscale()
*/
public function greyscale(): ImageInterface
{
return $this->modify(new GreyscaleModifier());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::brightness()
*/
public function brightness(int $level): ImageInterface
{
return $this->modify(new BrightnessModifier($level));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::contrast()
*/
public function contrast(int $level): ImageInterface
{
return $this->modify(new ContrastModifier($level));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::gamma()
*/
public function gamma(float $gamma): ImageInterface
{
return $this->modify(new GammaModifier($gamma));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::colorize()
*/
public function colorize(int $red = 0, int $green = 0, int $blue = 0): ImageInterface
{
return $this->modify(new ColorizeModifier($red, $green, $blue));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::flip()
*/
public function flip(): ImageInterface
{
return $this->modify(new FlipModifier());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::flop()
*/
public function flop(): ImageInterface
{
return $this->modify(new FlopModifier());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::blur()
*/
public function blur(int $amount = 5): ImageInterface
{
return $this->modify(new BlurModifier($amount));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::rotate()
*/
public function rotate(float $angle, mixed $background = 'ffffff'): ImageInterface
{
return $this->modify(new RotateModifier($angle, $background));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::orient()
*/
public function orient(): ImageInterface
{
return $this->modify(new AlignRotationModifier());
}
/**
* {@inheritdoc}
*
* @see ImageInterface::text()
*/
public function text(string $text, int $x, int $y, callable|Closure|FontInterface $font): ImageInterface
{
return $this->modify(
new TextModifier(
$text,
new Point($x, $y),
call_user_func(new FontFactory($font)),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::resize()
*/
public function resize(?int $width = null, ?int $height = null): ImageInterface
{
return $this->modify(new ResizeModifier($width, $height));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::resizeDown()
*/
public function resizeDown(?int $width = null, ?int $height = null): ImageInterface
{
return $this->modify(new ResizeDownModifier($width, $height));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::scale()
*/
public function scale(?int $width = null, ?int $height = null): ImageInterface
{
return $this->modify(new ScaleModifier($width, $height));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::scaleDown()
*/
public function scaleDown(?int $width = null, ?int $height = null): ImageInterface
{
return $this->modify(new ScaleDownModifier($width, $height));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::cover()
*/
public function cover(int $width, int $height, string $position = 'center'): ImageInterface
{
return $this->modify(new CoverModifier($width, $height, $position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::coverDown()
*/
public function coverDown(int $width, int $height, string $position = 'center'): ImageInterface
{
return $this->modify(new CoverDownModifier($width, $height, $position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::resizeCanvas()
*/
public function resizeCanvas(
?int $width = null,
?int $height = null,
mixed $background = 'ffffff',
string $position = 'center'
): ImageInterface {
return $this->modify(new ResizeCanvasModifier($width, $height, $background, $position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::resizeCanvasRelative()
*/
public function resizeCanvasRelative(
?int $width = null,
?int $height = null,
mixed $background = 'ffffff',
string $position = 'center'
): ImageInterface {
return $this->modify(new ResizeCanvasRelativeModifier($width, $height, $background, $position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::padDown()
*/
public function pad(
int $width,
int $height,
mixed $background = 'ffffff',
string $position = 'center'
): ImageInterface {
return $this->modify(new PadModifier($width, $height, $background, $position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::pad()
*/
public function contain(
int $width,
int $height,
mixed $background = 'ffffff',
string $position = 'center'
): ImageInterface {
return $this->modify(new ContainModifier($width, $height, $background, $position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::crop()
*/
public function crop(
int $width,
int $height,
int $offset_x = 0,
int $offset_y = 0,
mixed $background = 'ffffff',
string $position = 'top-left'
): ImageInterface {
return $this->modify(new CropModifier($width, $height, $offset_x, $offset_y, $background, $position));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::trim()
*/
public function trim(int $tolerance = 0): ImageInterface
{
return $this->modify(new TrimModifier($tolerance));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::place()
*/
public function place(
mixed $element,
string $position = 'top-left',
int $offset_x = 0,
int $offset_y = 0,
int $opacity = 100
): ImageInterface {
return $this->modify(new PlaceModifier($element, $position, $offset_x, $offset_y, $opacity));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::fill()
*/
public function fill(mixed $color, ?int $x = null, ?int $y = null): ImageInterface
{
return $this->modify(
new FillModifier(
$color,
is_null($x) || is_null($y) ? null : new Point($x, $y),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::drawPixel()
*/
public function drawPixel(int $x, int $y, mixed $color): ImageInterface
{
return $this->modify(new DrawPixelModifier(new Point($x, $y), $color));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::drawRectangle()
*/
public function drawRectangle(int $x, int $y, callable|Closure|Rectangle $init): ImageInterface
{
return $this->modify(
new DrawRectangleModifier(
call_user_func(new RectangleFactory(new Point($x, $y), $init)),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::drawEllipse()
*/
public function drawEllipse(int $x, int $y, callable|Closure|Ellipse $init): ImageInterface
{
return $this->modify(
new DrawEllipseModifier(
call_user_func(new EllipseFactory(new Point($x, $y), $init)),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::drawCircle()
*/
public function drawCircle(int $x, int $y, callable|Closure|Circle $init): ImageInterface
{
return $this->modify(
new DrawEllipseModifier(
call_user_func(new CircleFactory(new Point($x, $y), $init)),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::drawPolygon()
*/
public function drawPolygon(callable|Closure|Polygon $init): ImageInterface
{
return $this->modify(
new DrawPolygonModifier(
call_user_func(new PolygonFactory($init)),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::drawLine()
*/
public function drawLine(callable|Closure|Line $init): ImageInterface
{
return $this->modify(
new DrawLineModifier(
call_user_func(new LineFactory($init)),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::drawBezier()
*/
public function drawBezier(callable|Closure|Bezier $init): ImageInterface
{
return $this->modify(
new DrawBezierModifier(
call_user_func(new BezierFactory($init)),
),
);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::encodeByMediaType()
*/
public function encodeByMediaType(null|string|MediaType $type = null, mixed ...$options): EncodedImageInterface
{
return $this->encode(new MediaTypeEncoder($type, ...$options));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::encodeByExtension()
*/
public function encodeByExtension(
null|string|FileExtension $extension = null,
mixed ...$options
): EncodedImageInterface {
return $this->encode(new FileExtensionEncoder($extension, ...$options));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::encodeByPath()
*/
public function encodeByPath(?string $path = null, mixed ...$options): EncodedImageInterface
{
return $this->encode(new FilePathEncoder($path, ...$options));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toJpeg()
*/
public function toJpeg(mixed ...$options): EncodedImageInterface
{
return $this->encode(new JpegEncoder(...$options));
}
/**
* Alias of self::toJpeg()
*
* @throws RuntimeException
*/
public function toJpg(mixed ...$options): EncodedImageInterface
{
return $this->toJpeg(...$options);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toJpeg()
*/
public function toJpeg2000(mixed ...$options): EncodedImageInterface
{
return $this->encode(new Jpeg2000Encoder(...$options));
}
/**
* ALias of self::toJpeg2000()
*
* @throws RuntimeException
*/
public function toJp2(mixed ...$options): EncodedImageInterface
{
return $this->toJpeg2000(...$options);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toPng()
*/
public function toPng(mixed ...$options): EncodedImageInterface
{
return $this->encode(new PngEncoder(...$options));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toGif()
*/
public function toGif(mixed ...$options): EncodedImageInterface
{
return $this->encode(new GifEncoder(...$options));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toWebp()
*/
public function toWebp(mixed ...$options): EncodedImageInterface
{
return $this->encode(new WebpEncoder(...$options));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toBitmap()
*/
public function toBitmap(mixed ...$options): EncodedImageInterface
{
return $this->encode(new BmpEncoder(...$options));
}
/**
* Alias if self::toBitmap()
*
* @throws RuntimeException
*/
public function toBmp(mixed ...$options): EncodedImageInterface
{
return $this->toBitmap(...$options);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toAvif()
*/
public function toAvif(mixed ...$options): EncodedImageInterface
{
return $this->encode(new AvifEncoder(...$options));
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toTiff()
*/
public function toTiff(mixed ...$options): EncodedImageInterface
{
return $this->encode(new TiffEncoder(...$options));
}
/**
* Alias of self::toTiff()
*
* @throws RuntimeException
*/
public function toTif(mixed ...$options): EncodedImageInterface
{
return $this->toTiff(...$options);
}
/**
* {@inheritdoc}
*
* @see ImageInterface::toHeic()
*/
public function toHeic(mixed ...$options): EncodedImageInterface
{
return $this->encode(new HeicEncoder(...$options));
}
/**
* Show debug info for the current image
*
* @return array
*/
public function __debugInfo(): array
{
try {
return [
'width' => $this->width(),
'height' => $this->height(),
];
} catch (RuntimeException) {
return [];
}
}
/**
* Clone image
*/
public function __clone(): void
{
$this->driver = clone $this->driver;
$this->core = clone $this->core;
$this->exif = clone $this->exif;
}
}
================================================
FILE: src/ImageManager.php
================================================
driver = $this->resolveDriver($driver, ...$options);
}
/**
* Create image manager with given driver
*
* @link https://image.intervention.io/v3/basics/configuration-drivers#static-constructor
*
* @throws DriverException
* @throws InputException
*/
public static function withDriver(string|DriverInterface $driver, mixed ...$options): self
{
return new self(self::resolveDriver($driver, ...$options));
}
/**
* Create image manager with GD driver
*
* @link https://image.intervention.io/v3/basics/configuration-drivers#static-gd-driver-constructor
*
* @throws DriverException
* @throws InputException
*/
public static function gd(mixed ...$options): self
{
return self::withDriver(new GdDriver(), ...$options);
}
/**
* Create image manager with Imagick driver
*
* @link https://image.intervention.io/v3/basics/configuration-drivers#static-imagick-driver-constructor
*
* @throws DriverException
* @throws InputException
*/
public static function imagick(mixed ...$options): self
{
return self::withDriver(new ImagickDriver(), ...$options);
}
/**
* {@inheritdoc}
*
* @see ImageManagerInterface::create()
*/
public function create(int $width, int $height): ImageInterface
{
return $this->driver->createImage($width, $height);
}
/**
* {@inheritdoc}
*
* @see ImageManagerInterface::read()
*/
public function read(mixed $input, string|array|DecoderInterface $decoders = []): ImageInterface
{
return $this->driver->handleInput(
$input,
match (true) {
is_string($decoders), is_a($decoders, DecoderInterface::class) => [$decoders],
default => $decoders,
}
);
}
/**
* {@inheritdoc}
*
* @see ImageManagerInterface::animate()
*/
public function animate(callable $init): ImageInterface
{
return $this->driver->createAnimation($init);
}
/**
* {@inheritdoc}
*
* @see ImageManagerInterface::driver()
*/
public function driver(): DriverInterface
{
return $this->driver;
}
/**
* Return driver object from given input which might be driver classname or instance of DriverInterface
*
* @throws DriverException
* @throws InputException
*/
private static function resolveDriver(string|DriverInterface $driver, mixed ...$options): DriverInterface
{
$driver = match (true) {
$driver instanceof DriverInterface => $driver,
class_exists($driver) => new $driver(),
default => throw new DriverException(
'Unable to resolve driver. Argment must be either an instance of ' .
DriverInterface::class . '::class or a qualified namespaced name of the driver class.',
),
};
if (!$driver instanceof DriverInterface) {
throw new DriverException(
'Unable to resolve driver. Driver object must implement ' . DriverInterface::class . '.',
);
}
$driver->config()->setOptions(...$options);
return $driver;
}
}
================================================
FILE: src/InputHandler.php
================================================
*/
protected array $decoders = [
NativeObjectDecoder::class,
ImageObjectDecoder::class,
ColorObjectDecoder::class,
RgbHexColorDecoder::class,
RgbStringColorDecoder::class,
CmykStringColorDecoder::class,
HsvStringColorDecoder::class,
HslStringColorDecoder::class,
TransparentColorDecoder::class,
HtmlColornameDecoder::class,
FilePointerImageDecoder::class,
FilePathImageDecoder::class,
SplFileInfoImageDecoder::class,
BinaryImageDecoder::class,
DataUriImageDecoder::class,
Base64ImageDecoder::class,
EncodedImageObjectDecoder::class,
];
/**
* Driver with which the decoder classes are specialized
*/
protected ?DriverInterface $driver = null;
/**
* Create new input handler instance with given decoder classnames
*
* @param array $decoders
* @return void
*/
public function __construct(array $decoders = [], ?DriverInterface $driver = null)
{
$this->decoders = count($decoders) ? $decoders : $this->decoders;
$this->driver = $driver;
}
/**
* Static factory method
*
* @param array $decoders
*/
public static function withDecoders(array $decoders, ?DriverInterface $driver = null): self
{
return new self($decoders, $driver);
}
/**
* {@inheritdoc}
*
* @see InputHandlerInterface::handle()
*/
public function handle(mixed $input): ImageInterface|ColorInterface
{
foreach ($this->decoders as $decoder) {
try {
// decode with driver specialized decoder
return $this->resolve($decoder)->decode($input);
} catch (DecoderException | NotSupportedException $e) {
// try next decoder
}
}
if (isset($e)) {
throw new ($e::class)($e->getMessage());
}
throw new DecoderException('Unable to decode input.');
}
/**
* Resolve the given classname to an decoder object
*
* @throws DriverException
* @throws NotSupportedException
*/
private function resolve(string|DecoderInterface $decoder): DecoderInterface
{
if (($decoder instanceof DecoderInterface) && empty($this->driver)) {
return $decoder;
}
if (($decoder instanceof DecoderInterface) && !empty($this->driver)) {
return $this->driver->specialize($decoder);
}
if (empty($this->driver)) {
return new $decoder();
}
return $this->driver->specialize(new $decoder());
}
}
================================================
FILE: src/Interfaces/AnalyzerInterface.php
================================================
*/
interface CollectionInterface extends Traversable
{
/**
* Determine if the collection has item at given key
*/
public function has(int|string $key): bool;
/**
* Add item to collection
*
* @return CollectionInterface
*/
public function push(mixed $item): self;
/**
* Return item for given key or return default is key does not exist
*/
public function get(int|string $key, mixed $default = null): mixed;
/**
* Return item at given numeric position starting at 0
*/
public function getAtPosition(int $key = 0, mixed $default = null): mixed;
/**
* Return first item in collection
*/
public function first(): mixed;
/**
* Return last item in collection
*/
public function last(): mixed;
/**
* Return item count of collection
*/
public function count(): int;
/**
* Empty collection
*
* @return CollectionInterface
*/
public function empty(): self;
/**
* Transform collection as array
*
* @return array
*/
public function toArray(): array;
/**
* Extract items based on given values and discard the rest.
*
* @return CollectionInterface
*/
public function slice(int $offset, ?int $length = 0): self;
}
================================================
FILE: src/Interfaces/ColorChannelInterface.php
================================================
*/
public function toArray(): array;
/**
* Cast color object to hex encoded web color
*/
public function toHex(string $prefix = ''): string;
/**
* Return array of all color channels
*
* @return array
*/
public function channels(): array;
/**
* Return array of normalized color channel values
*
* @return array
*/
public function normalize(): array;
/**
* Retrieve the color channel by its classname
*
* @throws ColorException
*/
public function channel(string $classname): ColorChannelInterface;
/**
* Convert color to given colorspace
*/
public function convertTo(string|ColorspaceInterface $colorspace): self;
/**
* Determine if the current color is gray
*/
public function isGreyscale(): bool;
/**
* Determine if the current color is (semi) transparent
*/
public function isTransparent(): bool;
/**
* Determine whether the current color is completely transparent
*/
public function isClear(): bool;
/**
* Cast color object to string
*/
public function __toString(): string;
}
================================================
FILE: src/Interfaces/ColorProcessorInterface.php
================================================
$normalized
*/
public function colorFromNormalized(array $normalized): ColorInterface;
}
================================================
FILE: src/Interfaces/CoreInterface.php
================================================
*/
public function setNative(mixed $native): self;
/**
* Count number of frames of animated image core
*/
public function count(): int;
/**
* Return frame of given position in an animated image
*
* @throws AnimationException
*/
public function frame(int $position): FrameInterface;
/**
* Add new frame to core
*
* @return CoreInterface
*/
public function add(FrameInterface $frame): self;
/**
* Return number of repetitions of an animated image
*/
public function loops(): int;
/**
* Set the number of repetitions for an animation. Where a
* value of 0 means infinite repetition.
*
* @return CoreInterface
*/
public function setLoops(int $loops): self;
/**
* Get first frame in core
*
* @throws AnimationException
*/
public function first(): FrameInterface;
/**
* Get last frame in core
*
* @throws AnimationException
*/
public function last(): FrameInterface;
}
================================================
FILE: src/Interfaces/DecoderInterface.php
================================================
$objects
* @throws NotSupportedException
* @throws DriverException
* @return array