Skip to content

Commit

Permalink
feat: add ImageFileInfo::thumbHash() and ThumbHash
Browse files Browse the repository at this point in the history
  • Loading branch information
kbond committed Mar 21, 2024
1 parent e42e29a commit ab22abb
Show file tree
Hide file tree
Showing 5 changed files with 228 additions and 2 deletions.
52 changes: 51 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,8 @@
[![CI Status](https://github.com/zenstruck/image/workflows/CI/badge.svg)](https://github.com/zenstruck/image/actions?query=workflow%3ACI)
[![codecov](https://codecov.io/gh/zenstruck/image/branch/1.x/graph/badge.svg?token=MBKSCPO6U5)](https://codecov.io/gh/zenstruck/image)

Image file wrapper to provide image-specific metadata and transformations.
Image file wrapper to provide image-specific [metadata](#usage), generic [transformations](#transformations),
and [ThumbHash generator](#thumbhash).

## Installation

Expand Down Expand Up @@ -177,3 +178,52 @@ use Imagine\Image\ImageInterface;
$image->as(ImageInterface::class); // ImageInterface object for this image
$image->as(\Imagick::class); // \Imagick object for this image
```

### ThumbHash

> A very compact representation of an image placeholder. Store it inline with your data and show
> it while the real image is loading for a smoother loading experience.
>
> **-- [evanw.github.io/thumbhash](https://evanw.github.io/thumbhash/)**
> [!NOTE]
> [`srwiez/thumbhash`](https://github.com/SRWieZ/thumbhash) is required for this feature
> (install with `composer require srwiez/thumbhash`).
> [!NOTE]
> [`Imagick`](https://www.php.net/manual/en/book.imagick.php) is required for this feature.
#### Generate from Image

```php
use Zenstruck\Image\Hash\ThumbHash;

/** @var Zenstruck\ImageFileInfo $image */

$thumbHash = $image->thumbHash(); // ThumbHash

$thumbHash->dataUri(); // string - the ThumbHash as a data-uri
$thumbHash->approximateAspectRatio(); // float - the approximate aspect ratio
$thumbHash->key(); // string - small string representation that can be cached/stored in a database
```

> [!CAUTION]
> Generating from an image can be slow depending on the size of the source image. It is recommended
> to cache the data-uri and/or key for subsequent requests of the same ThumbHash image.
#### Generate from Key

When generating from an image, the `ThumbHash::key()` method returns a small string that
can be stored for later use. This key can be used to generate the ThumbHash without
needing to re-process the image.

```php
use Zenstruck\Image\Hash\ThumbHash;

/** @var string $key */

$thumbHash = ThumbHash::fromKey($key); // ThumbHash

$thumbHash->dataUri(); // string - the ThumbHash as a data-uri
$thumbHash->approximateAspectRatio(); // float - the approximate aspect ratio
```
4 changes: 3 additions & 1 deletion composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
"phpunit/phpunit": "^9.5.0",
"psr/container": "^1.0|^2.0",
"spatie/image": "^2.0|^3.2",
"srwiez/thumbhash": "^1.2",
"symfony/phpunit-bridge": "^6.1|^7.0",
"symfony/var-dumper": "^5.4|^6.0|^7.0"
},
Expand All @@ -38,6 +39,7 @@
},
"suggest": {
"imagine/imagine": "To use the Imagine image transformer.",
"intervention/image": "To use the Intervention image transformer."
"intervention/image": "To use the Intervention image transformer.",
"srwiez/thumbhash": "To generate ThumbHashes."
}
}
111 changes: 111 additions & 0 deletions src/Image/Hash/ThumbHash.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
<?php

/*
* This file is part of the zenstruck/image package.
*
* (c) Kevin Bond <[email protected]>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Zenstruck\Image\Hash;

use Zenstruck\ImageFileInfo;

/**
* @author Kevin Bond <[email protected]>
*/
final class ThumbHash
{
/** @var list<int> */
private array $hash;
private string $dataUri;

private function __construct(private \SplFileInfo|string $source)
{
if (!\class_exists(\Thumbhash\Thumbhash::class)) {
throw new \LogicException(\sprintf('"%s" requires the "srwiez/thumbhash" package to be installed. Run "composer require srwiez/thumbhash".', self::class));
}

if (!\class_exists(\Imagick::class)) {
throw new \LogicException(\sprintf('"%s" requires the "imagick" extension to be installed.', self::class));
}
}

/**
* Create from either an \SplFileInfo or a "key" string.
*/
public static function from(\SplFileInfo|string $source): self
{
return new self($source);
}

public function dataUri(): string
{
return $this->dataUri ??= \Thumbhash\Thumbhash::toDataURL($this->hash());
}

public function key(): string
{
if (\is_string($this->source)) {
return $this->source;
}

return $this->source = \Thumbhash\Thumbhash::convertHashToString($this->hash());
}

/**
* @return list<int>
*/
public function hash(): array
{
if (isset($this->hash)) {
return $this->hash;
}

if (\is_string($this->source)) {
return $this->hash = \Thumbhash\Thumbhash::convertStringToHash($this->source);
}

[$width, $height, $pixels] = self::extractSizeAndPixels($this->source);

return $this->hash = \Thumbhash\Thumbhash::RGBAToHash($width, $height, $pixels);
}

public function approximateAspectRatio(): float
{
return \Thumbhash\Thumbhash::toApproximateAspectRatio($this->hash());
}

/**
* @see \Thumbhash\extract_size_and_pixels_with_imagick()
*
* @return array{int, int, array}
*/
private static function extractSizeAndPixels(\SplFileInfo $file): array
{
$image = ImageFileInfo::wrap($file)->as(\Imagick::class);

if ($image->getImageWidth() > 100 || $image->getImageHeight() > 100) {
$image->scaleImage(100, 100, bestfit: true);
}

$width = $image->getImageWidth();
$height = $image->getImageHeight();
$pixels = [];

for ($y = 0; $y < $height; ++$y) {
for ($x = 0; $x < $width; ++$x) {
$pixel = $image->getImagePixelColor($x, $y);
$colors = $pixel->getColor(2);
$pixels[] = $colors['r'];
$pixels[] = $colors['g'];
$pixels[] = $colors['b'];
$pixels[] = $colors['a'];
}
}

return [$width, $height, $pixels];
}
}
6 changes: 6 additions & 0 deletions src/ImageFileInfo.php
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
namespace Zenstruck;

use Zenstruck\Image\Dimensions;
use Zenstruck\Image\Hash\ThumbHash;
use Zenstruck\Image\Transformer\MultiTransformer;

/**
Expand Down Expand Up @@ -202,6 +203,11 @@ public function delete(): void
}
}

public function thumbHash(): ThumbHash
{
return ThumbHash::from($this);
}

private static function transformer(): MultiTransformer
{
return self::$transformer ??= new MultiTransformer();
Expand Down
57 changes: 57 additions & 0 deletions tests/Hash/ThumbHashTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
<?php

/*
* This file is part of the zenstruck/image package.
*
* (c) Kevin Bond <[email protected]>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Zenstruck\Image\Tests\Hash;

use PHPUnit\Framework\TestCase;
use Zenstruck\Image\Hash\ThumbHash;
use Zenstruck\ImageFileInfo;

/**
* @author Kevin Bond <[email protected]>
*/
final class ThumbHashTest extends TestCase
{
/**
* @test
*/
public function generate_from_image(): void
{
$fixtureDir = __DIR__.'/../Fixture/files';

$thumbHash = ImageFileInfo::wrap($fixtureDir.'/symfony.jpg')->thumbHash();

$this->assertStringStartsWith('data:image/png;base64,', $thumbHash->dataUri());
$this->assertGreaterThan(20, \mb_strlen($thumbHash->key()));
$this->assertGreaterThan(20, \count($thumbHash->hash()));
$this->assertGreaterThan(0.79, $thumbHash->approximateAspectRatio());
$this->assertLessThan(0.9, $thumbHash->approximateAspectRatio());
}

/**
* @test
*/
public function generate_from_key_string(): void
{
$thumbHash = ThumbHash::from('JAgSBgD3xhinqMd3WXuLhZmoAAAAAAA');

$this->assertStringStartsWith('data:image/png;base64,', $thumbHash->dataUri());
$this->assertGreaterThan(20, \mb_strlen($thumbHash->key()));
$this->assertGreaterThan(20, \count($thumbHash->hash()));
$this->assertGreaterThan(0.79, $thumbHash->approximateAspectRatio());
$this->assertLessThan(0.9, $thumbHash->approximateAspectRatio());

$this->assertSame('', $thumbHash->dataUri());
$this->assertEqualsWithDelta(0.86, $thumbHash->approximateAspectRatio(), 0.01);
$this->assertSame('JAgSBgD3xhinqMd3WXuLhZmoAAAAAAA', $thumbHash->key());
$this->assertCount(23, $thumbHash->hash());
}
}

0 comments on commit ab22abb

Please sign in to comment.