Skip to content

Commit

Permalink
attempt to automatically select an intervention/image driver
Browse files Browse the repository at this point in the history
  • Loading branch information
frasmage committed Apr 9, 2024
1 parent e1222e3 commit a9c3351
Show file tree
Hide file tree
Showing 5 changed files with 86 additions and 11 deletions.
17 changes: 17 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,22 @@
# Changelog

## 6.1.0

* Attempt to automatically select an `intervention/image` driver based on the available extensions.
* Fix an error in package discovery if attempting to install both `plank/laravel-mediable` and `intervention/image-laravel` at the same time

## 6.0.5

* move ImageManipulator singleton to lazy instantiation

## 6.0.4

* Fix alt migration default value for the mysql dialect. Default value assigned from the Media model

## 6.0.3

* fix service provider database migration bindings by @frasmage in #349

## 6.0.2
- Added `intervention/image-laravel` package to the composer suggests list
- Updated documentation with configuration instructions for intervention/image
Expand Down
29 changes: 22 additions & 7 deletions docs/source/variants.rst
Original file line number Diff line number Diff line change
Expand Up @@ -10,30 +10,44 @@ Laravel-Mediable integrates the `intervention/image <http://image.intervention.i
Configure Intervention/image ImageManager
-----------------------------------------

Before you can use the ImageManipulation features of this package, you will need to make sure that the intervention/image package is properly configured. Intervention/image is capable of using either the `GD <https://www.php.net/manual/en/book.image.php>`_ or `ImageMagick <https://www.php.net/manual/en/book.imagick.php>`_ libraries as the underlying driver.
Before you can use the ImageManipulation features of this package, you will need to make sure that the intervention/image package is properly configured. Intervention/image is capable of using either the `GD <https://www.php.net/manual/en/book.image.php>`_ or `ImageMagick <https://www.php.net/manual/en/book.imagick.php>`_ PHP extensions as the underlying driver.

If intervention/image is not configured in the Laravel service container, this package will attempt to automatically select an intervention/image driver based on the extensions available, preferring `imagick` if available, and falling back to `gd`. If neither are available, attempting to use the ImageManipulator will result in an exception.

To configure Intervention/image yourself, refer to the following guides based on your version of the package.

Intervention/image >=3.0
^^^^^^^^^^^^^^^^^^^^^^^

RECOMMENDED: install the `intervention/image-laravel <https://image.intervention.io/v3/introduction/frameworks#laravel>`_ package to configure the container bindings automatically.

Alternatively, you can add the necessary bindings to the service container by adding the following to one of the service providers of your application.
Alternatively, you can add the necessary bindings to the service container manually by adding the following bindings to one of the service providers of your application.

::

<?php
// if using GD
$app->bind(Intervention\Image\Interfaces\DriverInterface::class, \Intervention\Image\Drivers\Gd\Driver::class);
class AppServiceProvider extends ServiceProvider
{
public function register()
{
// if using GD
$app->bind(Intervention\Image\Interfaces\DriverInterface::class,
\Intervention\Image\Drivers\Gd\Driver::class
);

// if using Imagick
$app->bind(Intervention\Image\Interfaces\DriverInterface::class, \Intervention\Image\Drivers\Imagick\Driver::class);
// if using Imagick
$app->bind(Intervention\Image\Interfaces\DriverInterface::class,
\Intervention\Image\Drivers\Imagick\Driver::class
);
}
}

Intervention/image <3.0
^^^^^^^^^^^^^^^^^^^^^^^

RECOMMENDED: `follow the steps <https://image.intervention.io/v2/introduction/installation#integration-in-laravel>`_ to enable the intervention/image Laravel service provider.

Otherwise, by default, intervention/image will use the `GD <https://www.php.net/manual/en/book.image.php>`_ library driver. If you intend to use the additional features of the `ImageMagick <https://www.php.net/manual/en/book.imagick.php>`_ driver, you should make sure that the PHP extension is installed and the correct configuration is bound to the Laravel service container.
Alternatively, you can add the necessary bindings to the service container manually by adding the following bindings to one of the service providers of your application.

::

Expand All @@ -49,6 +63,7 @@ Otherwise, by default, intervention/image will use the `GD <https://www.php.net/
ImageManager::class,
function() {
return new ImageManager(['driver' => 'imagick']);
// return new ImageManager(['driver' => 'gd']);
}
);
}
Expand Down
5 changes: 5 additions & 0 deletions src/Exceptions/MediaUpload/ConfigurationException.php
Original file line number Diff line number Diff line change
Expand Up @@ -52,4 +52,9 @@ public static function invalidOptimizer(string $optimizerClass): self
{
return new self("Invalid optimizer class `{$optimizerClass}`. Must implement `\Spatie\ImageOptimizer\Optimizer`.");
}

public static function interventionImageNotConfigured(): self
{
return new self("Before variants can be created, the intervention/image package must be configured in the Laravel container.");
}
}
16 changes: 14 additions & 2 deletions src/ImageManipulator.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,15 @@
use Intervention\Image\Image;
use Intervention\Image\ImageManager;
use Plank\Mediable\Exceptions\ImageManipulationException;
use Plank\Mediable\Exceptions\MediaUpload\ConfigurationException;
use Plank\Mediable\SourceAdapters\SourceAdapterInterface;
use Plank\Mediable\SourceAdapters\StreamAdapter;
use Psr\Http\Message\StreamInterface;
use Spatie\ImageOptimizer\OptimizerChain;

class ImageManipulator
{
private ImageManager $imageManager;
private ?ImageManager $imageManager;

/**
* @var ImageManipulation[]
Expand All @@ -33,7 +34,7 @@ class ImageManipulator
private ImageOptimizer $imageOptimizer;

public function __construct(
ImageManager $imageManager,
?ImageManager $imageManager,
FilesystemManager $filesystem,
ImageOptimizer $imageOptimizer
) {
Expand All @@ -47,6 +48,9 @@ public function defineVariant(
ImageManipulation $manipulation,
?array $tags = []
) {
if (!$this->imageManager) {
throw ConfigurationException::interventionImageNotConfigured();
}
$this->variantDefinitions[$variantName] = $manipulation;
foreach ($tags as $tag) {
$this->variantDefinitionGroups[$tag][] = $variantName;
Expand Down Expand Up @@ -105,6 +109,10 @@ public function createImageVariant(
string $variantName,
bool $forceRecreate = false
): Media {
if (!$this->imageManager) {
throw ConfigurationException::interventionImageNotConfigured();
}

$this->validateMedia($media);

$modelClass = config('mediable.model');
Expand Down Expand Up @@ -217,6 +225,10 @@ public function manipulateUpload(
SourceAdapterInterface $source,
ImageManipulation $manipulation
): StreamAdapter {
if (!$this->imageManager) {
throw ConfigurationException::interventionImageNotConfigured();
}

$outputFormat = $this->determineOutputFormat($manipulation, $media);
if (method_exists($this->imageManager, 'read')) {
// Intervention Image >=3.0
Expand Down
30 changes: 28 additions & 2 deletions src/MediableServiceProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,9 @@
use Illuminate\Filesystem\Filesystem;
use Illuminate\Filesystem\FilesystemManager;
use Illuminate\Support\ServiceProvider;
use Intervention\Image\Drivers\Gd\Driver;
use Intervention\Image\ImageManager;
use Intervention\Image\Interfaces\DriverInterface;
use Mimey\MimeTypes;
use Plank\Mediable\Commands\ImportMediaCommand;
use Plank\Mediable\Commands\PruneMediaCommand;
Expand Down Expand Up @@ -108,11 +110,12 @@ public function register(): void
);

$this->registerSourceAdapterFactory();
$this->registerImageManipulator();
$this->registerUploader();
$this->registerMover();
$this->registerUrlGeneratorFactory();
$this->registerConsoleCommands();
$this->registerImageManipulator();

}

/**
Expand Down Expand Up @@ -192,8 +195,31 @@ public function registerUrlGeneratorFactory(): void
public function registerImageManipulator(): void
{
$this->app->singleton(ImageManipulator::class, function (Container $app) {
$imageManager = null;
if ($app->has(ImageManager::class) || $app->has(DriverInterface::class)) {
$imageManager = $app->get(ImageManager::class);
} elseif (extension_loaded('imagick')) {
if (class_exists(\Intervention\Image\Drivers\Imagick\Driver::class)) {
// intervention/image >=3.0
$imageManager = ImageManager::imagick();
} else {
// intervention/image <3.0
/** @phpstan-ignore-next-line */
$imageManager = new ImageManager(['driver' => 'imagick']);

Check failure on line 208 in src/MediableServiceProvider.php

View workflow job for this annotation

GitHub Actions / PHP 8.3

No error to ignore is reported on line 208.
}
} elseif (extension_loaded('gd')) {
if (class_exists(Driver::class)) {
// intervention/image >=3.0
$imageManager = ImageManager::gd();
} else {
// intervention/image <3.0
/** @phpstan-ignore-next-line */
$imageManager = new ImageManager(['driver' => 'gd']);

Check failure on line 217 in src/MediableServiceProvider.php

View workflow job for this annotation

GitHub Actions / PHP 8.3

No error to ignore is reported on line 217.
}
}

return new ImageManipulator(
$app->get(ImageManager::class),
$imageManager,
$app->get(FilesystemManager::class),
$app->get(ImageOptimizer::class)
);
Expand Down

0 comments on commit a9c3351

Please sign in to comment.