diff --git a/Neos.Media/Classes/Domain/Model/Adjustment/MarkPointAdjustment.php b/Neos.Media/Classes/Domain/Model/Adjustment/MarkPointAdjustment.php new file mode 100644 index 00000000000..989bb67aee9 --- /dev/null +++ b/Neos.Media/Classes/Domain/Model/Adjustment/MarkPointAdjustment.php @@ -0,0 +1,94 @@ +x = $x; + } + + public function setY(int $y): void + { + $this->y = $y; + } + + public function setRadius(int $radius): void + { + $this->radius = $radius; + } + + public function setThickness(int $thickness): void + { + $this->thickness = $thickness; + } + + public function setColor(string $color): void + { + $this->color = $color; + } + + + public function applyToImage(ImagineImageInterface $image) + { + $palette = new Palette\RGB(); + $color = $palette->color($this->color); + $image->draw() + ->circle( + new Point($this->x, $this->y), + $this->radius, + $color, + false, + $this->thickness + ) + ; + + return $image; + } + + public function canBeApplied(ImagineImageInterface $image) + { + if (is_null($this->x) || is_null($this->y) || is_null($this->radius)) { + return false; + } + return true; + } +} diff --git a/Neos.Media/Classes/Domain/Model/Adjustment/ResizeDimensionCalculator.php b/Neos.Media/Classes/Domain/Model/Adjustment/ResizeDimensionCalculator.php new file mode 100644 index 00000000000..033ea9b222d --- /dev/null +++ b/Neos.Media/Classes/Domain/Model/Adjustment/ResizeDimensionCalculator.php @@ -0,0 +1,252 @@ +getWidth() > $maximumWidth) { + $newDimensions = $newDimensions->widen($maximumWidth); + } + + if ($maximumHeight !== null && $newDimensions->getHeight() > $maximumHeight) { + $newDimensions = $newDimensions->heighten($maximumHeight); + } + + return $newDimensions; + } + + /** + * @param BoxInterface $originalDimensions + * @param integer $requestedWidth + * @param integer $requestedHeight + * @param bool $allowUpScaling + * @param string $ratioMode + * @return BoxInterface + */ + protected static function calculateWithFixedDimensions(BoxInterface $originalDimensions, int $requestedWidth, int $requestedHeight, bool $allowUpScaling = false, string $ratioMode = ImageInterface::RATIOMODE_INSET): BoxInterface + { + if ($ratioMode === ImageInterface::RATIOMODE_OUTBOUND) { + return self::calculateOutboundBox($originalDimensions, $requestedWidth, $requestedHeight, $allowUpScaling); + } + + $newDimensions = clone $originalDimensions; + + $ratios = [ + $requestedWidth / $originalDimensions->getWidth(), + $requestedHeight / $originalDimensions->getHeight() + ]; + + $ratio = min($ratios); + $newDimensions = $newDimensions->scale($ratio); + + if ($allowUpScaling === false && $originalDimensions->contains($newDimensions) === false) { + return clone $originalDimensions; + } + + return $newDimensions; + } + + /** + * Calculate the final dimensions for an outbound box. usually exactly the requested width and height unless that + * would require upscaling and it is not allowed. + * + * @param BoxInterface $originalDimensions + * @param integer $requestedWidth + * @param integer $requestedHeight + * @param bool $allowUpScaling + * @return BoxInterface + */ + protected static function calculateOutboundBox(BoxInterface $originalDimensions, int $requestedWidth, int $requestedHeight, bool $allowUpScaling): BoxInterface + { + $newDimensions = new Box($requestedWidth, $requestedHeight); + + if ($allowUpScaling === true || $originalDimensions->contains($newDimensions) === true) { + return $newDimensions; + } + + // We need to make sure that the new dimensions are such that no upscaling is needed. + $ratios = [ + $originalDimensions->getWidth() / $requestedWidth, + $originalDimensions->getHeight() / $requestedHeight + ]; + + $ratio = min($ratios); + $newDimensions = $newDimensions->scale($ratio); + + return $newDimensions; + } + + /** + * Calculates new dimensions with a requested width applied. Takes upscaling into consideration. + * + * @param BoxInterface $originalDimensions + * @param integer $requestedWidth + * @param bool $allowUpScaling + * @return BoxInterface + */ + protected static function calculateScalingToWidth(BoxInterface $originalDimensions, int $requestedWidth, bool $allowUpScaling): BoxInterface + { + if ($allowUpScaling === false && $requestedWidth >= $originalDimensions->getWidth()) { + return $originalDimensions; + } + + $newDimensions = clone $originalDimensions; + $newDimensions = $newDimensions->widen($requestedWidth); + + return $newDimensions; + } + + /** + * Calculates new dimensions with a requested height applied. Takes upscaling into consideration. + * + * @param BoxInterface $originalDimensions + * @param integer $requestedHeight + * @param bool $allowUpScaling + * @return BoxInterface + */ + protected static function calculateScalingToHeight(BoxInterface $originalDimensions, int $requestedHeight, bool $allowUpScaling): BoxInterface + { + if ($allowUpScaling === false && $requestedHeight >= $originalDimensions->getHeight()) { + return $originalDimensions; + } + + $newDimensions = clone $originalDimensions; + $newDimensions = $newDimensions->heighten($requestedHeight); + + return $newDimensions; + } + + /** + * Calculates a resize dimension box that allows for outbound resize. + * The scaled image will be bigger than the requested dimensions in one dimension and then cropped. + * + * @param BoxInterface $imageSize + * @param BoxInterface $requestedDimensions + * @return BoxInterface + */ + public static function calculateOutboundScalingDimensions(BoxInterface $imageSize, BoxInterface $requestedDimensions, string $ratioMode = ImageInterface::RATIOMODE_INSET): BoxInterface + { + if ($ratioMode === ImageInterface::RATIOMODE_OUTBOUND) { + $ratios = [ + $requestedDimensions->getWidth() / $imageSize->getWidth(), + $requestedDimensions->getHeight() / $imageSize->getHeight() + ]; + + return $imageSize->scale(max($ratios)); + } + return $requestedDimensions; + } + + /** + * Calculate the informations for a preliminary crop to ensure that the given focal point stays inside the final image + * with the requested dimensions + * + * - The cropDimensions have the aspect of requested dimensions and have the maximal possible dimensions + * - The cropOffset will position the crop with the focal point as close to the center as possible + * - The returned focal point is the position of the focal point after the crop inside the requested dimensions + */ + public static function calculatePreliminaryCropSpecification( + BoxInterface $originalDimensions, + PointInterface $originalFocalPoint, + BoxInterface $targetDimensions, + ): PreliminaryCropSpecification { + $originalAspect = new AspectRatio($originalDimensions->getWidth(), $originalDimensions->getHeight()); + $targetAspect = new AspectRatio($targetDimensions->getWidth(), $targetDimensions->getHeight()); + + if ($originalAspect->getRatio() >= $targetAspect->getRatio()) { + // target-aspect is wider as original-aspect or same: use full height, width is cropped + $factor = $originalDimensions->getHeight() / $targetDimensions->getHeight(); + $cropDimensions = new \Imagine\Image\Box((int)($targetDimensions->getWidth() * $factor), $originalDimensions->getHeight()); + $cropOffsetX = $originalFocalPoint->getX() - (int)($cropDimensions->getWidth() / 2); + $cropOffsetXMax = $originalDimensions->getWidth() - $cropDimensions->getWidth(); + if ($cropOffsetX < 0) { + $cropOffsetX = 0; + } elseif ($cropOffsetX > $cropOffsetXMax) { + $cropOffsetX = $cropOffsetXMax; + } + $cropOffset = new Point($cropOffsetX, 0); + } else { + // target-aspect is higher than original-aspect: use full width, height is cropped + $factor = $originalDimensions->getWidth() / $targetDimensions->getWidth(); + $cropDimensions = new Box($originalDimensions->getWidth(), (int)($targetDimensions->getHeight() * $factor)); + $cropOffsetY = $originalFocalPoint->getY() - (int)($cropDimensions->getHeight() / 2); + $cropOffsetYMax = $originalDimensions->getHeight() - $cropDimensions->getHeight(); + if ($cropOffsetY < 0) { + $cropOffsetY = 0; + } elseif ($cropOffsetY > $cropOffsetYMax) { + $cropOffsetY = $cropOffsetYMax; + } + $cropOffset = new Point(0, $cropOffsetY); + } + + return new PreliminaryCropSpecification( + $cropOffset, + $cropDimensions, + new Point( + (int)round(($originalFocalPoint->getX() - $cropOffset->getX()) / $factor), + (int)round(($originalFocalPoint->getY() - $cropOffset->getY()) / $factor) + ) + ); + } +} diff --git a/Neos.Media/Classes/Domain/Model/Adjustment/ResizeImageAdjustment.php b/Neos.Media/Classes/Domain/Model/Adjustment/ResizeImageAdjustment.php index c94d5c43e89..425990eccbc 100644 --- a/Neos.Media/Classes/Domain/Model/Adjustment/ResizeImageAdjustment.php +++ b/Neos.Media/Classes/Domain/Model/Adjustment/ResizeImageAdjustment.php @@ -288,7 +288,15 @@ public function setAllowUpScaling(bool $allowUpScaling): void */ public function canBeApplied(ImagineImageInterface $image) { - $expectedDimensions = $this->calculateDimensions($image->getSize()); + $expectedDimensions = ResizeDimensionCalculator::calculateRequestedDimensions( + $image->getSize(), + $this->width, + $this->height, + $this->maximumWidth, + $this->maximumHeight, + $this->allowUpScaling ?? false, + $this->ratioMode ?? ImageInterface::RATIOMODE_INSET + ); return ((string)$expectedDimensions !== (string)$image->getSize()); } @@ -311,132 +319,19 @@ public function applyToImage(ImagineImageInterface $image) * * @param BoxInterface $originalDimensions Dimensions of the unadjusted image * @return BoxInterface + * @deprecated use ResizeDimensionCalculator::calculateRequestedDimensions instead */ protected function calculateDimensions(BoxInterface $originalDimensions): BoxInterface { - $newDimensions = clone $originalDimensions; - - switch (true) { - // height and width are set explicitly: - case ($this->width !== null && $this->height !== null): - $newDimensions = $this->calculateWithFixedDimensions($originalDimensions, $this->width, $this->height); - break; - // only width is set explicitly: - case ($this->width !== null): - $newDimensions = $this->calculateScalingToWidth($originalDimensions, $this->width); - break; - // only height is set explicitly: - case ($this->height !== null): - $newDimensions = $this->calculateScalingToHeight($originalDimensions, $this->height); - break; - } - - // We apply maximum dimensions and scale the new dimensions proportionally down to fit into the maximum. - if ($this->maximumWidth !== null && $newDimensions->getWidth() > $this->maximumWidth) { - $newDimensions = $newDimensions->widen($this->maximumWidth); - } - - if ($this->maximumHeight !== null && $newDimensions->getHeight() > $this->maximumHeight) { - $newDimensions = $newDimensions->heighten($this->maximumHeight); - } - - return $newDimensions; - } - - /** - * @param BoxInterface $originalDimensions - * @param integer $requestedWidth - * @param integer $requestedHeight - * @return BoxInterface - */ - protected function calculateWithFixedDimensions(BoxInterface $originalDimensions, int $requestedWidth, int $requestedHeight): BoxInterface - { - if ($this->ratioMode === ImageInterface::RATIOMODE_OUTBOUND) { - return $this->calculateOutboundBox($originalDimensions, $requestedWidth, $requestedHeight); - } - - $newDimensions = clone $originalDimensions; - - $ratios = [ - $requestedWidth / $originalDimensions->getWidth(), - $requestedHeight / $originalDimensions->getHeight() - ]; - - $ratio = min($ratios); - $newDimensions = $newDimensions->scale($ratio); - - if ($this->getAllowUpScaling() === false && $originalDimensions->contains($newDimensions) === false) { - return clone $originalDimensions; - } - - return $newDimensions; - } - - /** - * Calculate the final dimensions for an outbound box. usually exactly the requested width and height unless that - * would require upscaling and it is not allowed. - * - * @param BoxInterface $originalDimensions - * @param integer $requestedWidth - * @param integer $requestedHeight - * @return BoxInterface - */ - protected function calculateOutboundBox(BoxInterface $originalDimensions, int $requestedWidth, int $requestedHeight): BoxInterface - { - $newDimensions = new Box($requestedWidth, $requestedHeight); - - if ($this->getAllowUpScaling() === true || $originalDimensions->contains($newDimensions) === true) { - return $newDimensions; - } - - // We need to make sure that the new dimensions are such that no upscaling is needed. - $ratios = [ - $originalDimensions->getWidth() / $requestedWidth, - $originalDimensions->getHeight() / $requestedHeight - ]; - - $ratio = min($ratios); - $newDimensions = $newDimensions->scale($ratio); - - return $newDimensions; - } - - /** - * Calculates new dimensions with a requested width applied. Takes upscaling into consideration. - * - * @param BoxInterface $originalDimensions - * @param integer $requestedWidth - * @return BoxInterface - */ - protected function calculateScalingToWidth(BoxInterface $originalDimensions, int $requestedWidth): BoxInterface - { - if ($this->getAllowUpScaling() === false && $requestedWidth >= $originalDimensions->getWidth()) { - return $originalDimensions; - } - - $newDimensions = clone $originalDimensions; - $newDimensions = $newDimensions->widen($requestedWidth); - - return $newDimensions; - } - - /** - * Calculates new dimensions with a requested height applied. Takes upscaling into consideration. - * - * @param BoxInterface $originalDimensions - * @param integer $requestedHeight - * @return BoxInterface - */ - protected function calculateScalingToHeight(BoxInterface $originalDimensions, int $requestedHeight): BoxInterface - { - if ($this->getAllowUpScaling() === false && $requestedHeight >= $originalDimensions->getHeight()) { - return $originalDimensions; - } - - $newDimensions = clone $originalDimensions; - $newDimensions = $newDimensions->heighten($requestedHeight); - - return $newDimensions; + return ResizeDimensionCalculator::calculateRequestedDimensions( + $originalDimensions, + $this->width, + $this->height, + $this->maximumWidth, + $this->maximumHeight, + $this->allowUpScaling ?? false, + $this->ratioMode ?? ImageInterface::RATIOMODE_INSET + ); } /** @@ -456,22 +351,31 @@ protected function resize(ImagineImageInterface $image, string $mode = ImageInte throw new \InvalidArgumentException('Invalid mode specified', 1574686891); } - $imageSize = $image->getSize(); - $requestedDimensions = $this->calculateDimensions($imageSize); + $originalDimensions = $image->getSize(); - $image->strip(); + $requestedDimensions = ResizeDimensionCalculator::calculateRequestedDimensions( + $originalDimensions, + $this->width, + $this->height, + $this->maximumWidth, + $this->maximumHeight, + $this->allowUpScaling ?? false, + $this->ratioMode ?? ImageInterface::RATIOMODE_INSET + ); - $resizeDimensions = $requestedDimensions; - if ($mode === ImageInterface::RATIOMODE_OUTBOUND) { - $resizeDimensions = $this->calculateOutboundScalingDimensions($imageSize, $requestedDimensions); - } + $finalDimensions = ResizeDimensionCalculator::calculateOutboundScalingDimensions( + $originalDimensions, + $requestedDimensions, + $this->ratioMode + ); - $image->resize($resizeDimensions, $filter); + $image->strip(); + $image->resize($finalDimensions, $filter); if ($mode === ImageInterface::RATIOMODE_OUTBOUND) { $image->crop(new Point( - max(0, round(($resizeDimensions->getWidth() - $requestedDimensions->getWidth()) / 2)), - max(0, round(($resizeDimensions->getHeight() - $requestedDimensions->getHeight()) / 2)) + max(0, round(($finalDimensions->getWidth() - $requestedDimensions->getWidth()) / 2)), + max(0, round(($finalDimensions->getHeight() - $requestedDimensions->getHeight()) / 2)) ), $requestedDimensions); } @@ -485,14 +389,14 @@ protected function resize(ImagineImageInterface $image, string $mode = ImageInte * @param BoxInterface $imageSize * @param BoxInterface $requestedDimensions * @return BoxInterface + * @deprecated use ResizeDimensionCalculator::calculateOutboundScalingDimensions instead */ protected function calculateOutboundScalingDimensions(BoxInterface $imageSize, BoxInterface $requestedDimensions): BoxInterface { - $ratios = [ - $requestedDimensions->getWidth() / $imageSize->getWidth(), - $requestedDimensions->getHeight() / $imageSize->getHeight() - ]; - - return $imageSize->scale(max($ratios)); + return ResizeDimensionCalculator::calculateOutboundScalingDimensions( + $imageSize, + $requestedDimensions, + $this->ratioMode + ); } } diff --git a/Neos.Media/Classes/Domain/Model/Dto/PreliminaryCropSpecification.php b/Neos.Media/Classes/Domain/Model/Dto/PreliminaryCropSpecification.php new file mode 100644 index 00000000000..dd29da26a6e --- /dev/null +++ b/Neos.Media/Classes/Domain/Model/Dto/PreliminaryCropSpecification.php @@ -0,0 +1,35 @@ +focalPointX; + } + + public function setFocalPointX(?int $x): void + { + $this->focalPointX = $x; + } + + public function getFocalPointY(): ?int + { + return $this->focalPointY; + } + + public function setFocalPointY(?int $y): void + { + $this->focalPointY = $y; + } + + public function hasFocalPoint(): bool + { + if ($this->focalPointX !== null && $this->focalPointY !== null) { + return true; + } + return false; + } + + public function getFocalPoint(): ?PointInterface + { + if ($this->hasFocalPoint()) { + return new Point($this->focalPointX, $this->focalPointY); + } + return null; + } +} diff --git a/Neos.Media/Classes/Domain/Model/ImageVariant.php b/Neos.Media/Classes/Domain/Model/ImageVariant.php index 01a6b6476a3..6546fdd970b 100644 --- a/Neos.Media/Classes/Domain/Model/ImageVariant.php +++ b/Neos.Media/Classes/Domain/Model/ImageVariant.php @@ -32,9 +32,10 @@ * * @Flow\Entity */ -class ImageVariant extends Asset implements AssetVariantInterface, ImageInterface +class ImageVariant extends Asset implements AssetVariantInterface, ImageInterface, FocalPointSupportInterface { use DimensionsTrait; + use FocalPointTrait; /** * @var ImageService diff --git a/Neos.Media/Classes/Domain/Model/Thumbnail.php b/Neos.Media/Classes/Domain/Model/Thumbnail.php index 56d3cf87c82..873ccde897c 100644 --- a/Neos.Media/Classes/Domain/Model/Thumbnail.php +++ b/Neos.Media/Classes/Domain/Model/Thumbnail.php @@ -29,10 +29,11 @@ * } * ) */ -class Thumbnail implements ImageInterface +class Thumbnail implements ImageInterface, FocalPointSupportInterface { use DimensionsTrait; use QualityTrait; + use FocalPointTrait; /** * @var ThumbnailGeneratorStrategy diff --git a/Neos.Media/Classes/Domain/Model/ThumbnailConfiguration.php b/Neos.Media/Classes/Domain/Model/ThumbnailConfiguration.php index a24b015fba8..936df98e08e 100644 --- a/Neos.Media/Classes/Domain/Model/ThumbnailConfiguration.php +++ b/Neos.Media/Classes/Domain/Model/ThumbnailConfiguration.php @@ -123,7 +123,7 @@ public function getMaximumHeight() } /** - * @return boolean + * @return string */ public function getRatioMode() { diff --git a/Neos.Media/Classes/Domain/Model/ThumbnailGenerator/ImageThumbnailGenerator.php b/Neos.Media/Classes/Domain/Model/ThumbnailGenerator/ImageThumbnailGenerator.php index 8d8058cda3a..8be8f80a7f0 100644 --- a/Neos.Media/Classes/Domain/Model/ThumbnailGenerator/ImageThumbnailGenerator.php +++ b/Neos.Media/Classes/Domain/Model/ThumbnailGenerator/ImageThumbnailGenerator.php @@ -12,12 +12,18 @@ */ use Neos\Flow\Annotations as Flow; +use Neos\Media\Domain\Model\Adjustment\CropImageAdjustment; +use Neos\Media\Domain\Model\Adjustment\ResizeDimensionCalculator; +use Neos\Media\Domain\Model\Adjustment\MarkPointAdjustment; use Neos\Media\Domain\Model\Adjustment\QualityImageAdjustment; use Neos\Media\Domain\Model\Adjustment\ResizeImageAdjustment; +use Neos\Media\Domain\Model\Dto\PreliminaryCropSpecification; +use Neos\Media\Domain\Model\FocalPointSupportInterface; use Neos\Media\Domain\Model\ImageInterface; use Neos\Media\Domain\Model\Thumbnail; use Neos\Media\Domain\Service\ImageService; use Neos\Media\Exception; +use Neos\Media\Imagine\Box; /** * A system-generated preview version of an Image @@ -75,6 +81,59 @@ public function refresh(Thumbnail $thumbnail) ) ]; + $asset = $thumbnail->getOriginalAsset(); + $preliminaryCropSpecification = null; + if ($asset instanceof FocalPointSupportInterface && $asset->hasFocalPoint()) { + // in case we have a focal point we calculate the target dimension and add an + // preliminary crop to ensure that the focal point stays inside the final image + // while beeing as central as possible + + $originalFocalPoint = $asset->getFocalPoint(); + $originalDimensions = new Box($asset->getWidth(), $asset->getHeight()); + $requestedDimensions = ResizeDimensionCalculator::calculateRequestedDimensions( + originalDimensions: $originalDimensions, + width: $thumbnail->getConfigurationValue('width'), + maximumWidth: $thumbnail->getConfigurationValue('maximumWidth'), + height: $thumbnail->getConfigurationValue('height'), + maximumHeight: $thumbnail->getConfigurationValue('maximumHeight'), + ratioMode: $thumbnail->getConfigurationValue('ratioMode'), + allowUpScaling: $thumbnail->getConfigurationValue('allowUpScaling'), + ); + + $preliminaryCropSpecification = ResizeDimensionCalculator::calculatePreliminaryCropSpecification( + originalDimensions: $originalDimensions, + originalFocalPoint: $originalFocalPoint, + targetDimensions: $requestedDimensions, + ); + + $adjustments = array_merge( + [ + new CropImageAdjustment( + [ + 'x' => $preliminaryCropSpecification->cropOffset->getX(), + 'y' => $preliminaryCropSpecification->cropOffset->getY(), + 'width' => $preliminaryCropSpecification->cropDimensions->getWidth(), + 'height' => $preliminaryCropSpecification->cropDimensions->getHeight(), + ] + ) + ], + $adjustments, + [ + // this is for debugging purposes only + // @todo remove before merging + new MarkPointAdjustment( + [ + 'x' => $preliminaryCropSpecification->focalPoint->getX(), + 'y' => $preliminaryCropSpecification->focalPoint->getY(), + 'radius' => 5, + 'color' => '#0f0', + 'thickness' => 4 + ] + ), + ] + ); + } + $targetFormat = $thumbnail->getConfigurationValue('format'); $processedImageInfo = $this->imageService->processImage($thumbnail->getOriginalAsset()->getResource(), $adjustments, $targetFormat); @@ -82,6 +141,11 @@ public function refresh(Thumbnail $thumbnail) $thumbnail->setWidth($processedImageInfo['width']); $thumbnail->setHeight($processedImageInfo['height']); $thumbnail->setQuality($processedImageInfo['quality']); + + if ($preliminaryCropSpecification instanceof PreliminaryCropSpecification) { + $thumbnail->setFocalPointX($preliminaryCropSpecification->focalPoint->getX()); + $thumbnail->setFocalPointY($preliminaryCropSpecification->focalPoint->getY()); + } } catch (\Exception $exception) { $message = sprintf('Unable to generate thumbnail for the given image (filename: %s, SHA1: %s)', $thumbnail->getOriginalAsset()->getResource()->getFilename(), $thumbnail->getOriginalAsset()->getResource()->getSha1()); throw new Exception\NoThumbnailAvailableException($message, 1433109654, $exception); diff --git a/Neos.Media/Classes/Domain/Repository/ThumbnailRepository.php b/Neos.Media/Classes/Domain/Repository/ThumbnailRepository.php index 0fd34f67e0d..326b5292c67 100644 --- a/Neos.Media/Classes/Domain/Repository/ThumbnailRepository.php +++ b/Neos.Media/Classes/Domain/Repository/ThumbnailRepository.php @@ -152,7 +152,7 @@ public function persistThumbnailDirectly(Thumbnail $thumbnail, ThumbnailConfigur $assetIdentifier = $this->persistenceManager->getIdentifierByObject($thumbnail->getOriginalAsset()); $thumbnailResource = $thumbnail->getResource(); - $sql = 'INSERT INTO neos_media_domain_model_thumbnail (persistence_object_identifier, originalasset, resource, width, height, configuration, configurationhash, staticresource, quality) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)'; + $sql = 'INSERT INTO neos_media_domain_model_thumbnail (persistence_object_identifier, originalasset, resource, width, height, configuration, configurationhash, staticresource, quality, focalpointx, focalpointy) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)'; $params = [ $thumbnailIdentifier, $assetIdentifier, @@ -163,6 +163,8 @@ public function persistThumbnailDirectly(Thumbnail $thumbnail, ThumbnailConfigur $configuration->getHash(), $thumbnail->getStaticResource(), $thumbnail->getQuality(), + $thumbnail->getFocalPointX(), + $thumbnail->getFocalPointY(), ]; $connection = $this->entityManager->getConnection(); diff --git a/Neos.Media/Classes/Domain/Service/ThumbnailService.php b/Neos.Media/Classes/Domain/Service/ThumbnailService.php index 32843c971c3..b0ac9753ba2 100644 --- a/Neos.Media/Classes/Domain/Service/ThumbnailService.php +++ b/Neos.Media/Classes/Domain/Service/ThumbnailService.php @@ -18,12 +18,15 @@ use Neos\Flow\Persistence\Exception\IllegalObjectTypeException; use Neos\Flow\Persistence\PersistenceManagerInterface; use Neos\Flow\ResourceManagement\ResourceManager; +use Neos\Media\Domain\Model\Adjustment\ResizeDimensionCalculator; use Neos\Media\Domain\Model\AssetInterface; +use Neos\Media\Domain\Model\FocalPointSupportInterface; use Neos\Media\Domain\Model\ImageInterface; use Neos\Media\Domain\Model\Thumbnail; use Neos\Media\Domain\Model\ThumbnailConfiguration; use Neos\Media\Domain\Repository\ThumbnailRepository; use Neos\Media\Exception\ThumbnailServiceException; +use Neos\Media\Imagine\Box; use Neos\Utility\Arrays; use Neos\Utility\MediaTypes; use Psr\Log\LoggerInterface; @@ -84,6 +87,12 @@ class ThumbnailService */ protected $throwableStorage; + /** + * @var ResizeDimensionCalculator + * @Flow\Inject + */ + protected $imageDimensionCalculationHelperThingy; + /** * Returns a thumbnail of the given asset * @@ -147,6 +156,39 @@ public function getThumbnail(AssetInterface $asset, ThumbnailConfiguration $conf if ($thumbnail === null) { $thumbnail = new Thumbnail($asset, $configuration); + // predict dimensions async image thumbnails, this is not needed for immediately calculated images as those + // values are stored again after calculating + if ($async === true && $asset instanceof ImageInterface) { + $originalDimensions = new Box($asset->getWidth(), $asset->getHeight()); + + $requestedDimensions = ResizeDimensionCalculator::calculateRequestedDimensions( + originalDimensions: $originalDimensions, + width: $configuration->getWidth(), + maximumWidth: $configuration->getMaximumWidth(), + height: $configuration->getHeight(), + maximumHeight: $configuration->getMaximumHeight(), + ratioMode: $configuration->getRatioMode(), + allowUpScaling: $configuration->isUpScalingAllowed() + ); + + $thumbnail->setWidth($requestedDimensions->getWidth()); + $thumbnail->setHeight($requestedDimensions->getHeight()); + + // calculate focal point for new thumbnails + if ($asset instanceof FocalPointSupportInterface && $asset->hasFocalPoint()) { + $originalFocalPoint = $asset->getFocalPoint(); + + $preliminaryCropSpecification = ResizeDimensionCalculator::calculatePreliminaryCropSpecification( + originalDimensions: $originalDimensions, + originalFocalPoint: $originalFocalPoint, + targetDimensions: $requestedDimensions, + ); + + $thumbnail->setFocalPointX($preliminaryCropSpecification->focalPoint->getX()); + $thumbnail->setFocalPointY($preliminaryCropSpecification->focalPoint->getY()); + } + } + // If the thumbnail strategy failed to generate a valid thumbnail if ($async === false && $thumbnail->getResource() === null && $thumbnail->getStaticResource() === null) { // the thumbnail should not be persisted at this point, but remove is a no-op if the thumbnail diff --git a/Neos.Media/Migrations/Mysql/Version20240604184831.php b/Neos.Media/Migrations/Mysql/Version20240604184831.php new file mode 100644 index 00000000000..f7429b9014f --- /dev/null +++ b/Neos.Media/Migrations/Mysql/Version20240604184831.php @@ -0,0 +1,36 @@ +abortIf($this->connection->getDatabasePlatform()->getName() !== 'mysql', 'Migration can only be executed safely on "mysql".'); + + $this->addSql('ALTER TABLE neos_media_domain_model_imagevariant ADD focalpointx INT DEFAULT NULL, ADD focalpointy INT DEFAULT NULL'); + $this->addSql('ALTER TABLE neos_media_domain_model_thumbnail ADD focalpointx INT DEFAULT NULL, ADD focalpointy INT DEFAULT NULL'); + } + + public function down(Schema $schema): void + { + $this->abortIf($this->connection->getDatabasePlatform()->getName() !== 'mysql', 'Migration can only be executed safely on "mysql".'); + + $this->addSql('ALTER TABLE neos_media_domain_model_imagevariant DROP focalpointx, DROP focalpointy'); + $this->addSql('ALTER TABLE neos_media_domain_model_thumbnail DROP focalpointx, DROP focalpointy'); + + } +} diff --git a/Neos.Media/Tests/Unit/Domain/Model/Adjustment/ResizeDimensionCalculatorTest.php b/Neos.Media/Tests/Unit/Domain/Model/Adjustment/ResizeDimensionCalculatorTest.php new file mode 100644 index 00000000000..495c356eac5 --- /dev/null +++ b/Neos.Media/Tests/Unit/Domain/Model/Adjustment/ResizeDimensionCalculatorTest.php @@ -0,0 +1,291 @@ + [ + new \Imagine\Image\Box(400, 400), + new Point(200, 200), + new Box(200, 200), + + new Point(0, 0), + new Box(400, 400), + new Point(100, 100), + ]; + + yield 'portrait to portrait' => [ + new Box(800, 400), + new Point(400, 200), + new Box(400, 200), + + new Point(0, 0), + new Box(800, 400), + new Point(200, 100), + ]; + + yield 'portrait to square fp left' => [ + new Box(800, 400), + new Point(50, 200), + new Box(400, 400), + + new Point(0, 0), + new Box(400, 400), + new Point(50, 200), + ]; + + yield 'portrait to square fp center' => [ + new Box(800, 400), + new Point(400, 200), + new Box(400, 400), + + new Point(200, 0), + new Box(400, 400), + new Point(200, 200), + ]; + + yield 'portrait to square fp right' => [ + new Box(800, 400), + new Point(700, 100), + new Box(400, 400), + + new Point(400, 0), + new Box(400, 400), + new Point(300, 100), + ]; + + yield 'landscape to square fp center' => [ + new Box(400, 800), + new Point(200, 400), + new Box(400, 400), + + new Point(0, 200), + new Box(400, 400), + new Point(200, 200), + ]; + + yield 'landscape to square fp top' => [ + new Box(400, 800), + new Point(350, 50), + new Box(400, 400), + + new Point(0, 0), + new Box(400, 400), + new Point(350, 50), + ]; + + yield 'landscape to square fp bottom' => [ + new Box(400, 800), + new Point(300, 750), + new Box(200, 200), + + new Point(0, 400), + new Box(400, 400), + new Point(150, 175), + ]; + } + + /** + * @dataProvider calculateCropConfigurationCentersFocalPointDataProvider + * @test + */ + public function calculateCropConfigurationCentersFocalPoint( + BoxInterface $originalDimensions, + PointInterface $originalFocalPoint, + BoxInterface $requestedDimensions, + PointInterface $expectedCropOffset, + BoxInterface $expectedCropDimensions, + PointInterface $expectedCroppedFocalPoint + ): void { + $preliminaryCropSpecification = ResizeDimensionCalculator::calculatePreliminaryCropSpecification( + $originalDimensions, + $originalFocalPoint, + $requestedDimensions + ); + + $this->assertEquals($expectedCropDimensions->getWidth(), $preliminaryCropSpecification->cropDimensions->getWidth()); + $this->assertEquals($expectedCropDimensions->getHeight(), $preliminaryCropSpecification->cropDimensions->getHeight()); + + $this->assertEquals($expectedCropOffset->getX(), $preliminaryCropSpecification->cropOffset->getX()); + $this->assertEquals($expectedCropOffset->getY(), $preliminaryCropSpecification->cropOffset->getY()); + + $this->assertEquals($expectedCroppedFocalPoint->getX(), $preliminaryCropSpecification->focalPoint->getX()); + $this->assertEquals($expectedCroppedFocalPoint->getY(), $preliminaryCropSpecification->focalPoint->getY()); + } +}