diff --git a/src/Asset/Controller/Image/CustomStreamController.php b/src/Asset/Controller/Image/CustomStreamController.php new file mode 100644 index 000000000..e20acf30a --- /dev/null +++ b/src/Asset/Controller/Image/CustomStreamController.php @@ -0,0 +1,117 @@ +value)] + #[Get( + path: self::PREFIX . '/assets/{id}/image/stream/custom', + operationId: 'asset_image_stream_custom', + description: 'asset_image_stream_custom_description', + summary: 'asset_image_stream_custom_summary', + tags: [Tags::Assets->name] + )] + #[IdParameter(type: 'image')] + #[MimeTypeParameter] + #[ResizeModeParameter] + #[ImageConfigParameter('width', 140)] + #[ImageConfigParameter('height')] + #[ImageConfigParameter('quality')] + #[ImageConfigParameter('dpi')] + #[ContainParameter] + #[FrameParameter] + #[CoverParameter] + #[ForceResizeParameter] + #[SuccessResponse( + description: 'asset_image_stream_custom_success_response', + content: [new MediaType('image/*')], + headers: [new ContentDisposition(HttpResponseHeaders::INLINE_TYPE->value)] + )] + #[DefaultResponses([ + HttpResponseCodes::UNAUTHORIZED, + HttpResponseCodes::NOT_FOUND, + ])] + public function streamCustomThumbnailConfig( + int $id, + #[MapQueryString] ImageDownloadConfigParameter $parameters + ): StreamedResponse { + $asset = $this->assetService->getAssetElement( + $this->securityService->getCurrentUser(), + $id + ); + + return $this->binaryService->streamImageThumbnailFromConfig($asset, $parameters); + } +} diff --git a/src/Asset/MappedParameter/ImageDownloadConfigParameter.php b/src/Asset/MappedParameter/ImageDownloadConfigParameter.php index 27ac702cc..f4b3ab8e1 100644 --- a/src/Asset/MappedParameter/ImageDownloadConfigParameter.php +++ b/src/Asset/MappedParameter/ImageDownloadConfigParameter.php @@ -32,7 +32,12 @@ public function __construct( private ?int $width = null, private ?int $height = null, private ?int $quality = null, - private ?int $dpi = null + private ?int $dpi = null, + private ?string $positioning = 'center', + private string $cover = 'false', + private string $frame = 'false', + private string $contain = 'false', + private string $forceResize = 'false', ) { if (!in_array($this->mimeType, [MimeTypes::JPEG->value, MimeTypes::PNG->value], true)) { throw new InvalidArgumentException('Invalid mime type' . $this->mimeType); @@ -41,6 +46,27 @@ public function __construct( if (!in_array($this->resizeMode, ResizeModes::ALLOWED_MODES)) { throw new InvalidArgumentException('Invalid resize mode ' . $this->resizeMode); } + + if ($this->resizeMode === ResizeModes::SCALE_BY_HEIGHT && !$this->isValidHeight()) { + throw new InvalidArgumentException( + 'Height must be set and non-negative when using scale by width resize mode' + ); + } + + if ($this->resizeMode === ResizeModes::SCALE_BY_WIDTH && !$this->isValidWidth()) { + throw new InvalidArgumentException( + 'Width must be set and non-negative when using scale by width resize mode' + ); + } + + if ( + (!$this->isValidWidth() || !$this->isValidHeight()) && + ($this->hasFrame() || $this->hasCover() || $this->hasContain() || $this->resizeMode === ResizeModes::RESIZE) + ) { + throw new InvalidArgumentException( + 'Width, height must be set and non-negative when using frame, cover, contain or resize' + ); + } } public function getMimeType(): string @@ -72,4 +98,69 @@ public function getDpi(): ?int { return $this->dpi; } + + public function getCoverTransformation(): array + { + return [ + ... $this->getBaseTransformationValues(), + 'positioning' => $this->getPositioning(), + ]; + } + + public function getFrameTransformation(): array + { + return $this->getBaseTransformationValues(); + } + + public function getContainTransformation(): array + { + return $this->getBaseTransformationValues(); + } + + public function getPositioning(): ?string + { + return $this->positioning; + } + + public function getForceResize(): bool + { + return $this->forceResize === 'true'; // TODO: symfony 7.1 will support bool type + } + + public function hasCover(): bool + { + // TODO: symfony 7.1 will support bool type + return $this->cover === 'true'; + } + + public function hasFrame(): bool + { + // TODO: symfony 7.1 will support bool type + return $this->frame === 'true'; + } + + public function hasContain(): bool + { + // TODO: symfony 7.1 will support bool type + return $this->contain === 'true'; + } + + private function isValidWidth(): bool + { + return $this->width !== null && $this->width > 0; + } + + private function isValidHeight(): bool + { + return $this->height !== null && $this->height > 0; + } + + private function getBaseTransformationValues(): array + { + return [ + 'width' => $this->getWidth(), + 'height' => $this->getHeight(), + 'forceResize' => $this->getForceResize(), + ]; + } } diff --git a/src/Asset/OpenApi/Attribute/Parameter/Query/ContainParameter.php b/src/Asset/OpenApi/Attribute/Parameter/Query/ContainParameter.php new file mode 100644 index 000000000..5ea3263a1 --- /dev/null +++ b/src/Asset/OpenApi/Attribute/Parameter/Query/ContainParameter.php @@ -0,0 +1,35 @@ +getType()); + } + + return $this->getStreamedResponse( + $this->thumbnailService->getThumbnailFromConfiguration($image, $configParameter), + HttpResponseHeaders::INLINE_TYPE->value + ); + } + /** * @throws ElementProcessingNotCompletedException * @throws InvalidElementTypeException diff --git a/src/Asset/Service/BinaryServiceInterface.php b/src/Asset/Service/BinaryServiceInterface.php index 37dbb819a..c3143f88b 100644 --- a/src/Asset/Service/BinaryServiceInterface.php +++ b/src/Asset/Service/BinaryServiceInterface.php @@ -17,6 +17,7 @@ namespace Pimcore\Bundle\StudioBackendBundle\Asset\Service; use League\Flysystem\FilesystemException; +use Pimcore\Bundle\StudioBackendBundle\Asset\MappedParameter\ImageDownloadConfigParameter; use Pimcore\Bundle\StudioBackendBundle\Asset\MappedParameter\VideoImageStreamConfigParameter; use Pimcore\Bundle\StudioBackendBundle\Exception\Api\ElementProcessingNotCompletedException; use Pimcore\Bundle\StudioBackendBundle\Exception\Api\ElementStreamResourceNotFoundException; @@ -47,6 +48,14 @@ public function downloadVideoByThumbnail( */ public function streamPreviewImageThumbnail(Asset $image): StreamedResponse; + /** + * @throws InvalidElementTypeException|InvalidThumbnailException + */ + public function streamImageThumbnailFromConfig( + Asset $image, + ImageDownloadConfigParameter $configParameter + ): StreamedResponse; + /** * @throws ElementProcessingNotCompletedException * @throws InvalidElementTypeException diff --git a/src/Asset/Service/ThumbnailService.php b/src/Asset/Service/ThumbnailService.php index cbf4cbb5c..7eb514fa6 100644 --- a/src/Asset/Service/ThumbnailService.php +++ b/src/Asset/Service/ThumbnailService.php @@ -168,6 +168,18 @@ private function getImageThumbnailConfig( } } + if ($parameters->hasCover()) { + $thumbnailConfig->addItem('cover', $parameters->getCoverTransformation()); + } + + if ($parameters->hasFrame()) { + $thumbnailConfig->addItem('frame', $parameters->getFrameTransformation()); + } + + if ($parameters->hasContain()) { + $thumbnailConfig->addItem('contain', $parameters->getContainTransformation()); + } + return $thumbnailConfig; } diff --git a/translations/studio_api_docs.en.yaml b/translations/studio_api_docs.en.yaml index 526f20efd..f1392204f 100644 --- a/translations/studio_api_docs.en.yaml +++ b/translations/studio_api_docs.en.yaml @@ -120,6 +120,11 @@ asset_image_download_custom_description: | The {id} must be an ID of existing asset image asset_image_download_custom_success_response: Custom image binary file asset_image_download_custom_summary: Download custom image by ID and configuration +asset_image_stream_custom_summary: Stream custom image thumbnail by ID and configuration +asset_image_stream_custom_description: | + Stream image asset thumbnail based on the provided {id} and configuration parameters.
+ The {id} must be an ID of existing asset image +asset_image_stream_custom_success_response: Image asset stream based on custom thumbnail configuration asset_image_stream_preview_description: | Stream image asset preview based on the provided {id}.
The {id} must be an ID of existing asset image