started working on 0.2 overhaul
This commit is contained in:
parent
a92ab71dfd
commit
37ab60d81b
23 changed files with 169 additions and 495 deletions
30
.atoum.php
30
.atoum.php
|
@ -1,30 +0,0 @@
|
||||||
<?php
|
|
||||||
use mageekguy\atoum\reports;
|
|
||||||
|
|
||||||
$runner
|
|
||||||
->addTestsFromDirectory(__DIR__ . '/tests/units');
|
|
||||||
|
|
||||||
$runner->getScore()->getCoverage()->excludeDirectory(__DIR__ . '/tests/units/mock');
|
|
||||||
|
|
||||||
$travis = getenv('TRAVIS');
|
|
||||||
if ($travis) {
|
|
||||||
$script->addDefaultReport();
|
|
||||||
$coverallsToken = getenv('COVERALLS_REPO_TOKEN');
|
|
||||||
if ($coverallsToken) {
|
|
||||||
$coverallsReport = new reports\asynchronous\coveralls('classes', $coverallsToken);
|
|
||||||
$defaultFinder = $coverallsReport->getBranchFinder();
|
|
||||||
$coverallsReport
|
|
||||||
->setBranchFinder(function () use ($defaultFinder) {
|
|
||||||
if (($branch = getenv('TRAVIS_BRANCH')) === false) {
|
|
||||||
$branch = $defaultFinder();
|
|
||||||
}
|
|
||||||
return $branch;
|
|
||||||
}
|
|
||||||
)
|
|
||||||
->setServiceName('travis-ci')
|
|
||||||
->setServiceJobId(getenv('TRAVIS_JOB_ID'))
|
|
||||||
->addDefaultWriter()
|
|
||||||
;
|
|
||||||
$runner->addReport($coverallsReport);
|
|
||||||
}
|
|
||||||
}
|
|
19
.devcontainer/Dockerfile
Normal file
19
.devcontainer/Dockerfile
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
FROM ubuntu:22.04
|
||||||
|
|
||||||
|
# prepare to install php 8.2
|
||||||
|
RUN apt update && apt install -y software-properties-common
|
||||||
|
RUN add-apt-repository ppa:ondrej/php
|
||||||
|
RUN apt update
|
||||||
|
|
||||||
|
# install php 8.2 and other fundamental packages
|
||||||
|
RUN export DEBIAN_FRONTEND=noninteractive; apt install -y --no-install-recommends php8.2 php-curl git openssl unzip
|
||||||
|
|
||||||
|
# install composer and its CA certificates
|
||||||
|
COPY --from=composer:latest /usr/bin/composer /usr/local/bin/composer
|
||||||
|
COPY --from=composer:latest /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/ca-certificates.crt
|
||||||
|
|
||||||
|
# install the PHP extensions that basically all PHP projects should need
|
||||||
|
RUN export DEBIAN_FRONTEND=noninteractive; apt install -y php8.2-opcache php-xdebug php-mbstring php-zip php-gd php-xml
|
||||||
|
|
||||||
|
# install extensions that are more project-specific
|
||||||
|
RUN export DEBIAN_FRONTEND=noninteractive; apt install -y php-gmagick
|
37
.devcontainer/devcontainer.json
Normal file
37
.devcontainer/devcontainer.json
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
{
|
||||||
|
// build from dockerfile
|
||||||
|
"build": {
|
||||||
|
"dockerfile": "Dockerfile",
|
||||||
|
"context": ".."
|
||||||
|
},
|
||||||
|
// specify run arguments
|
||||||
|
"runArgs": [
|
||||||
|
"--dns=8.8.8.8" // for some reason DNS doesn't work right unless we explicitly name a DNS server
|
||||||
|
],
|
||||||
|
// mount entire sites_v2 directory, so we can access global config and shared DB
|
||||||
|
"workspaceMount": "source=${localWorkspaceFolder},target=/workspace/${localWorkspaceFolderBasename},type=bind,consistency=cached",
|
||||||
|
"workspaceFolder": "/workspace/${localWorkspaceFolderBasename}",
|
||||||
|
// specify extensions that we want
|
||||||
|
"customizations": {
|
||||||
|
"vscode": {
|
||||||
|
"extensions": [
|
||||||
|
"DEVSENSE.intelli-php-vscode",
|
||||||
|
"DEVSENSE.phptools-vscode",
|
||||||
|
"DEVSENSE.profiler-php-vscode",
|
||||||
|
"DEVSENSE.composer-php-vscode",
|
||||||
|
"SanderRonde.phpstan-vscode",
|
||||||
|
"sibiraj-s.vscode-scss-formatter",
|
||||||
|
"mrmlnc.vscode-scss",
|
||||||
|
"Gruntfuggly.todo-tree",
|
||||||
|
"redhat.vscode-yaml",
|
||||||
|
"oliversturm.fix-json",
|
||||||
|
"ecmel.vscode-html-css",
|
||||||
|
"yzhang.markdown-all-in-one",
|
||||||
|
"DavidAnson.vscode-markdownlint",
|
||||||
|
"helixquar.randomeverything",
|
||||||
|
"neilbrayfield.php-docblocker",
|
||||||
|
"ms-vscode.test-adapter-converter"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,9 +0,0 @@
|
||||||
language: php
|
|
||||||
php:
|
|
||||||
- 7.1
|
|
||||||
- 7.2
|
|
||||||
- 7.3
|
|
||||||
- 7.4
|
|
||||||
install:
|
|
||||||
- composer install
|
|
||||||
script: composer test
|
|
19
README.md
19
README.md
|
@ -1,9 +1,6 @@
|
||||||
# image-transform
|
# image-transform
|
||||||
|
|
||||||
[![Build Status](https://travis-ci.org/jobyone/image-transform.svg?branch=main)](https://travis-ci.org/jobyone/image-transform)
|
A tightly-focused library for performing a very limited set of simple image transformations. This library's purpose is to eschew the standard kitchen sink approach to PHP image libraries in favor of high performance, wider driver support, and a dead simple API.
|
||||||
[![Coverage Status](https://coveralls.io/repos/github/jobyone/image-transform/badge.svg?branch=main)](https://coveralls.io/github/jobyone/image-transform?branch=main)
|
|
||||||
|
|
||||||
A tightly-focused library for performing a very limited set of simple image transformations. This library's purpose is to eschew the standard kitchen sink approach to PHP image libraries in favor of high performance, wide driver support, and a dead simple API.
|
|
||||||
|
|
||||||
## Current state
|
## Current state
|
||||||
|
|
||||||
|
@ -11,12 +8,12 @@ This library is under active development, and until a 1.0 release is made you sh
|
||||||
|
|
||||||
### Current progress
|
### Current progress
|
||||||
|
|
||||||
| Driver | Rotate | Mirror | Resize | Crop | Overlay | Grayscale | Colorize |
|
| Driver | Rotate | Mirror | Resize | Crop |
|
||||||
| :--------- | :----: | :----: | :----: | :--: | :-----: | :-------: | :------: |
|
| :--------- | :----: | :----: | :----: | :--: |
|
||||||
| GD | X | X | X | X | | | |
|
| GD | X | X | X | X |
|
||||||
| Imagick | | | | | | | |
|
| Imagick | | | | |
|
||||||
| Gmagick | | | | | | | |
|
| Gmagick | | | | |
|
||||||
| ImagickCLI | X | X | X | X | | | |
|
| ImagickCLI | X | X | X | X |
|
||||||
|
|
||||||
## Roadmap
|
## Roadmap
|
||||||
|
|
||||||
|
@ -57,7 +54,7 @@ More complex, and also lesser used effects/stages that may or may not make it in
|
||||||
|
|
||||||
#### Order of operations
|
#### Order of operations
|
||||||
|
|
||||||
In the name of simplicity and ease of use, the effective order of operations will always be as reflected above:
|
In the name of simplicity, ease of use, and performance, the effective order of operations will always and only be:
|
||||||
|
|
||||||
1. Orientation
|
1. Orientation
|
||||||
2. Resizing and cropping
|
2. Resizing and cropping
|
||||||
|
|
|
@ -15,16 +15,16 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"suggest": {
|
"suggest": {
|
||||||
"ext-imagick": "to use the Imagick implementation",
|
"ext-imagick": "to use the ImageMagick implementation",
|
||||||
"ext-gmagick": "to use the Gmagick implementation"
|
"ext-gmagick": "to use the Gmagick implementation"
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
"php": ">=7.1",
|
"php": ">=8.1",
|
||||||
"ext-gd": "*"
|
"ext-gd": "*"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"atoum/atoum": "^3.4",
|
"ext-gmagick": "*",
|
||||||
"atoum/stubs": "^2.6"
|
"ext-gd": "*"
|
||||||
},
|
},
|
||||||
"autoload-dev": {
|
"autoload-dev": {
|
||||||
"psr-4": {
|
"psr-4": {
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
use ByJoby\ImageTransform\Drivers\ImagickCLIDriver;
|
use ByJoby\ImageTransform\Drivers\MagickCliDriver;
|
||||||
use ByJoby\ImageTransform\Sizers\Cover;
|
use ByJoby\ImageTransform\Sizers\Cover;
|
||||||
|
|
||||||
include __DIR__ . '/../vendor/autoload.php';
|
include __DIR__ . '/../vendor/autoload.php';
|
||||||
|
|
||||||
// first step is instantiate a Driver, in this case ImagickCLI
|
// first step is instantiate a Driver, in this case ImagickCLI
|
||||||
$driver = new ImagickCLIDriver();
|
$driver = new MagickCliDriver();
|
||||||
|
|
||||||
// instantiate an Image object using a source file and Sizer
|
// instantiate an Image object using a source file and Sizer
|
||||||
// in this example we're covering a 200x500 box
|
// in this example we're covering a 200x500 box
|
||||||
|
|
25
src/DefaultDriver.php
Normal file
25
src/DefaultDriver.php
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
<?php
|
||||||
|
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
||||||
|
namespace ByJoby\ImageTransform;
|
||||||
|
|
||||||
|
use ByJoby\ImageTransform\Drivers\GDDriver;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stores a default driver to be used when a driver is not set for an Image.
|
||||||
|
* Defaults to a GD driver.
|
||||||
|
*/
|
||||||
|
class DefaultDriver
|
||||||
|
{
|
||||||
|
protected $driver;
|
||||||
|
|
||||||
|
public static function get(): DriverInterface
|
||||||
|
{
|
||||||
|
return static::$driver
|
||||||
|
?? static::$driver = new GDDriver();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function set(DriverInterface $driver): void
|
||||||
|
{
|
||||||
|
static::$driver = $driver;
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,6 +6,24 @@ use ByJoby\ImageTransform\Sizers\AbstractSizer;
|
||||||
|
|
||||||
interface DriverInterface
|
interface DriverInterface
|
||||||
{
|
{
|
||||||
public function image(string $src, AbstractSizer $sizer): Image;
|
/**
|
||||||
|
* Save an image with its current settings. If a filename is specified the
|
||||||
|
* image will be saved to it and null returned, otherwise the image will be
|
||||||
|
* returned as a string.
|
||||||
|
*
|
||||||
|
* @param Image $image
|
||||||
|
* @param string|null $filename
|
||||||
|
* @return string|null
|
||||||
|
*/
|
||||||
public function save(Image $image, ?string $filename = null): ?string;
|
public function save(Image $image, ?string $filename = null): ?string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the temp directory in which files should be created if necessary for
|
||||||
|
* processing images. A directory in the system temp folder will be used by
|
||||||
|
* default.
|
||||||
|
*
|
||||||
|
* @param string $dir
|
||||||
|
* @return static
|
||||||
|
*/
|
||||||
|
public function setTempDir(string $dir): static;
|
||||||
}
|
}
|
|
@ -2,7 +2,7 @@
|
||||||
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
||||||
namespace ByJoby\ImageTransform\Drivers;
|
namespace ByJoby\ImageTransform\Drivers;
|
||||||
|
|
||||||
abstract class AbstractCLIDriver extends AbstractDriver
|
abstract class AbstractCliDriver extends AbstractDriver
|
||||||
{
|
{
|
||||||
public function __construct()
|
public function __construct()
|
||||||
{
|
{
|
|
@ -9,20 +9,11 @@ use ByJoby\ImageTransform\Sizers\AbstractSizer;
|
||||||
abstract class AbstractDriver implements DriverInterface
|
abstract class AbstractDriver implements DriverInterface
|
||||||
{
|
{
|
||||||
protected $tempDir = null;
|
protected $tempDir = null;
|
||||||
protected $chmod = 0775;
|
protected $chmod_dir = 0775;
|
||||||
|
protected $chmod_file = 0665;
|
||||||
|
|
||||||
abstract protected function doSave(Image $image, string $filename);
|
abstract protected function doSave(Image $image, string $filename);
|
||||||
|
|
||||||
public function __clone()
|
|
||||||
{
|
|
||||||
$this->tempDir = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function image(string $src, AbstractSizer $sizer): Image
|
|
||||||
{
|
|
||||||
return new Image($src, $this, $sizer);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function tempDir(): string
|
public function tempDir(): string
|
||||||
{
|
{
|
||||||
if (!$this->tempDir) {
|
if (!$this->tempDir) {
|
||||||
|
@ -31,12 +22,13 @@ abstract class AbstractDriver implements DriverInterface
|
||||||
return $this->tempDir;
|
return $this->tempDir;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function setTempDir(string $dir)
|
public function setTempDir(string $dir): static
|
||||||
{
|
{
|
||||||
if (!$this->mkdir($dir)) {
|
if (!$this->mkdir($dir)) {
|
||||||
throw new \Exception("Temp directory " . htmlentities($dir) . " doesn't exist or isn't writeable, and couldn't be created.");
|
throw new \Exception("Temp directory " . htmlentities($dir) . " doesn't exist or isn't writeable, and couldn't be created.");
|
||||||
}
|
}
|
||||||
$this->tempDir = $dir;
|
$this->tempDir = $dir;
|
||||||
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function mkdir(string $dir)
|
protected function mkdir(string $dir)
|
||||||
|
@ -52,7 +44,7 @@ abstract class AbstractDriver implements DriverInterface
|
||||||
if (is_dir($parent)) {
|
if (is_dir($parent)) {
|
||||||
// check parent permissions
|
// check parent permissions
|
||||||
if (!is_writeable($parent)) {
|
if (!is_writeable($parent)) {
|
||||||
chmod($parent, $this->chmod);
|
chmod($parent, $this->chmod_dir);
|
||||||
}
|
}
|
||||||
if (!is_writeable($parent)) {
|
if (!is_writeable($parent)) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -61,7 +53,7 @@ abstract class AbstractDriver implements DriverInterface
|
||||||
if (!mkdir($dir)) {
|
if (!mkdir($dir)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
chmod($dir, $this->chmod);
|
chmod($dir, $this->chmod_dir);
|
||||||
return is_writeable($dir);
|
return is_writeable($dir);
|
||||||
} else {
|
} else {
|
||||||
// parent doesn't exist, so recursive call failed
|
// parent doesn't exist, so recursive call failed
|
||||||
|
@ -83,11 +75,14 @@ abstract class AbstractDriver implements DriverInterface
|
||||||
touch($filename);
|
touch($filename);
|
||||||
}
|
}
|
||||||
$this->doSave($image, realpath($filename));
|
$this->doSave($image, realpath($filename));
|
||||||
|
chmod($filename, $this->chmod_file);
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
$filename = $this->tempDir() . '/save.jpg';
|
$filename = $this->tempDir() . '/' . uniqid() . '.jpg';
|
||||||
$this->doSave($image, $filename);
|
$this->doSave($image, $filename);
|
||||||
return file_get_contents($filename);
|
$output = file_get_contents($filename);
|
||||||
|
unlink($filename);
|
||||||
|
return $output;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -5,8 +5,9 @@ namespace ByJoby\ImageTransform\Drivers;
|
||||||
use ByJoby\ImageTransform\Image;
|
use ByJoby\ImageTransform\Image;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This driver uses PHP's built-in GD libary. This is by far the slowest
|
* This driver uses PHP's built-in GD libary. This is by far the slowest driver,
|
||||||
* driver, but support is basically universal.
|
* and its memory use is absolutely atrocious for large images, but support is
|
||||||
|
* basically universal.
|
||||||
*/
|
*/
|
||||||
class GDDriver extends AbstractExtensionDriver
|
class GDDriver extends AbstractExtensionDriver
|
||||||
{
|
{
|
||||||
|
|
|
@ -5,13 +5,13 @@ namespace ByJoby\ImageTransform\Drivers;
|
||||||
use ByJoby\ImageTransform\Image;
|
use ByJoby\ImageTransform\Image;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This driver uses exec() and command-line ImageMagick utilities to
|
* This driver uses exec() and command-line ImageMagick (or Gmagick) utilities
|
||||||
* transform images. It is likely approaching the limits of how fast this
|
* to transform images. It is likely approaching the limits of how fast this
|
||||||
* library can possibly be. The downside is that it will only run if
|
* library can possibly be. The downside is that it will only run if you have
|
||||||
* you have exec() enabled, and your server allows it to execute the
|
* exec() enabled, and your server allows it to execute the ImageMagick/Gmagick
|
||||||
* ImageMagick binaries.
|
* binaries and use them to read and write image files.
|
||||||
*/
|
*/
|
||||||
class ImagickCLIDriver extends AbstractCLIDriver
|
class MagickCliDriver extends AbstractCliDriver
|
||||||
{
|
{
|
||||||
protected $mogrify_executable;
|
protected $mogrify_executable;
|
||||||
|
|
|
@ -3,21 +3,23 @@
|
||||||
namespace ByJoby\ImageTransform;
|
namespace ByJoby\ImageTransform;
|
||||||
|
|
||||||
use ByJoby\ImageTransform\Sizers\AbstractSizer;
|
use ByJoby\ImageTransform\Sizers\AbstractSizer;
|
||||||
|
use ByJoby\ImageTransform\Sizers\Original;
|
||||||
|
|
||||||
class Image
|
class Image
|
||||||
{
|
{
|
||||||
protected $source, $driver;
|
protected $source;
|
||||||
protected $originalWidth, $originalHeight;
|
protected $driver;
|
||||||
|
protected $originalWidth;
|
||||||
|
protected $originalHeight;
|
||||||
protected $rotation = 0;
|
protected $rotation = 0;
|
||||||
protected $flipH = false;
|
protected $flipH = false;
|
||||||
protected $flipV = false;
|
protected $flipV = false;
|
||||||
protected $sizer = null;
|
protected $sizer = null;
|
||||||
|
|
||||||
public function __construct(string $source, DriverInterface $driver, AbstractSizer $sizer)
|
public function __construct(string $source, AbstractSizer|null $sizer = null)
|
||||||
{
|
{
|
||||||
$this->setSource($source);
|
$this->setSource($source);
|
||||||
$this->setSizer($sizer);
|
$this->setSizer($sizer ?? new Original());
|
||||||
$this->driver = clone $driver;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function source(): string
|
public function source(): string
|
||||||
|
@ -25,7 +27,7 @@ class Image
|
||||||
return $this->source;
|
return $this->source;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setSource(string $source)
|
public function setSource(string $source): static
|
||||||
{
|
{
|
||||||
// set source
|
// set source
|
||||||
$this->source = realpath($source);
|
$this->source = realpath($source);
|
||||||
|
@ -41,6 +43,8 @@ class Image
|
||||||
}
|
}
|
||||||
// get height/width
|
// get height/width
|
||||||
list($this->originalWidth, $this->originalHeight) = getimagesize($this->source);
|
list($this->originalWidth, $this->originalHeight) = getimagesize($this->source);
|
||||||
|
// return self
|
||||||
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function sizer(): AbstractSizer
|
public function sizer(): AbstractSizer
|
||||||
|
@ -48,15 +52,17 @@ class Image
|
||||||
return $this->sizer;
|
return $this->sizer;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setSizer(AbstractSizer $sizer)
|
public function setSizer(AbstractSizer $sizer): static
|
||||||
{
|
{
|
||||||
$this->sizer = clone $sizer;
|
$this->sizer = clone $sizer;
|
||||||
$this->sizer->image($this);
|
$this->sizer->image($this);
|
||||||
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function rotate(int $steps = 1)
|
public function rotate(int $steps = 1): static
|
||||||
{
|
{
|
||||||
$this->rotation = ($this->rotation + $steps) % 4;
|
$this->rotation = ($this->rotation + $steps) % 4;
|
||||||
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function rotation(): int
|
public function rotation(): int
|
||||||
|
@ -64,22 +70,24 @@ class Image
|
||||||
return $this->rotation;
|
return $this->rotation;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function flipH()
|
public function flipH(): static
|
||||||
{
|
{
|
||||||
$this->flipH = !$this->flipH;
|
$this->flipH = !$this->flipH;
|
||||||
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function flipV()
|
public function flipV(): static
|
||||||
{
|
{
|
||||||
$this->flipV = !$this->flipV;
|
$this->flipV = !$this->flipV;
|
||||||
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getFlipH()
|
public function getFlipH(): bool
|
||||||
{
|
{
|
||||||
return $this->flipH;
|
return $this->flipH;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getFlipV()
|
public function getFlipV(): bool
|
||||||
{
|
{
|
||||||
return $this->flipV;
|
return $this->flipV;
|
||||||
}
|
}
|
||||||
|
@ -114,8 +122,14 @@ class Image
|
||||||
return $this->originalHeight;
|
return $this->originalHeight;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function save(string $file)
|
public function driver(): DriverInterface
|
||||||
{
|
{
|
||||||
$this->driver->save($this, $file);
|
return $this->driver ?? DefaultDriver::get();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function save(string $file): static
|
||||||
|
{
|
||||||
|
$this->driver()->save($this, $file);
|
||||||
|
return $this;
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -2,6 +2,9 @@
|
||||||
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
||||||
namespace ByJoby\ImageTransform\Sizers;
|
namespace ByJoby\ImageTransform\Sizers;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This sizer does no manipulations and keeps the image the size it originally was.
|
||||||
|
*/
|
||||||
class Original extends AbstractSizer
|
class Original extends AbstractSizer
|
||||||
{
|
{
|
||||||
public function resizeToWidth(): ?int
|
public function resizeToWidth(): ?int
|
||||||
|
|
Before Width: | Height: | Size: 4.8 KiB After Width: | Height: | Size: 4.8 KiB |
Before Width: | Height: | Size: 4 KiB After Width: | Height: | Size: 4 KiB |
Before Width: | Height: | Size: 5.6 KiB After Width: | Height: | Size: 5.6 KiB |
|
@ -1,173 +0,0 @@
|
||||||
<?php
|
|
||||||
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
|
||||||
namespace ByJoby\ImageTransform\tests\units\Sizers;
|
|
||||||
|
|
||||||
use atoum;
|
|
||||||
use ByJoby\ImageTransform\Sizers\Cover as SizerUnderTest;
|
|
||||||
use ByJoby\ImageTransform\tests\units\mock\MockDriver;
|
|
||||||
|
|
||||||
class Cover extends atoum
|
|
||||||
{
|
|
||||||
public function testPortraitWhenImageIsTaller()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',100,150))
|
|
||||||
->integer($image->width())->isEqualTo(100)
|
|
||||||
->integer($image->height())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(200)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(150)
|
|
||||||
->given($image->rotate())
|
|
||||||
->integer($image->width())->isEqualTo(100)
|
|
||||||
->integer($image->height())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(150)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testPortraitWhenImageIsShorter()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',50,300))
|
|
||||||
->integer($image->width())->isEqualTo(50)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(300)
|
|
||||||
->given($image->rotate())
|
|
||||||
->integer($image->width())->isEqualTo(50)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(600)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(300)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testPortraitWithSquareImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x200.jpg',75,300))
|
|
||||||
->integer($image->width())->isEqualTo(75)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(300)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testPortraitWithLandscapeImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',100,200))
|
|
||||||
->integer($image->width())->isEqualTo(100)
|
|
||||||
->integer($image->height())->isEqualTo(200)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(400)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(200)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(200)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWhenImageIsTaller()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',150,100))
|
|
||||||
->integer($image->width())->isEqualTo(150)
|
|
||||||
->integer($image->height())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(200)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(100)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWhenImageIsShorter()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',300,75))
|
|
||||||
->integer($image->width())->isEqualTo(300)
|
|
||||||
->integer($image->height())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(75)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWithSquareImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x200.jpg',300,75))
|
|
||||||
->integer($image->width())->isEqualTo(300)
|
|
||||||
->integer($image->height())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(75)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWithPortraitImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',200,100))
|
|
||||||
->integer($image->width())->isEqualTo(200)
|
|
||||||
->integer($image->height())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(200)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(400)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(200)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(100)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testWithSmallSquareImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x200.jpg',300,300))
|
|
||||||
->integer($image->width())->isEqualTo(300)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(300)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testWithSmallLandscapeImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',300,300))
|
|
||||||
->integer($image->width())->isEqualTo(300)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(600)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(300)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testWithSmallPortraitImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',300,300))
|
|
||||||
->integer($image->width())->isEqualTo(300)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(600)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToHeight())->isEqualTo(300)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function image(string $path, int $width, int $height)
|
|
||||||
{
|
|
||||||
$driver = new MockDriver();
|
|
||||||
return $driver->image($path, new SizerUnderTest($width, $height));
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,173 +0,0 @@
|
||||||
<?php
|
|
||||||
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
|
||||||
namespace ByJoby\ImageTransform\tests\units\Sizers;
|
|
||||||
|
|
||||||
use atoum;
|
|
||||||
use ByJoby\ImageTransform\Sizers\Fit as SizerUnderTest;
|
|
||||||
use ByJoby\ImageTransform\tests\units\mock\MockDriver;
|
|
||||||
|
|
||||||
class Fit extends atoum
|
|
||||||
{
|
|
||||||
public function testPortraitWhenImageIsTaller()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',100,150))
|
|
||||||
->integer($image->width())->isEqualTo(75)
|
|
||||||
->integer($image->height())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
->given($image->rotate())
|
|
||||||
->integer($image->width())->isEqualTo(100)
|
|
||||||
->integer($image->height())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testPortraitWhenImageIsShorter()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',50,300))
|
|
||||||
->integer($image->width())->isEqualTo(50)
|
|
||||||
->integer($image->height())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
->given($image->rotate())
|
|
||||||
->integer($image->width())->isEqualTo(50)
|
|
||||||
->integer($image->height())->isEqualTo(25)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(25)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testPortraitWithSquareImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x200.jpg',75,300))
|
|
||||||
->integer($image->width())->isEqualTo(75)
|
|
||||||
->integer($image->height())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testPortraitWithLandscapeImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',100,200))
|
|
||||||
->integer($image->width())->isEqualTo(100)
|
|
||||||
->integer($image->height())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWhenImageIsTaller()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',150,100))
|
|
||||||
->integer($image->width())->isEqualTo(150)
|
|
||||||
->integer($image->height())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWhenImageIsShorter()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',300,75))
|
|
||||||
->integer($image->width())->isEqualTo(150)
|
|
||||||
->integer($image->height())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWithSquareImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x200.jpg',300,75))
|
|
||||||
->integer($image->width())->isEqualTo(75)
|
|
||||||
->integer($image->height())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(75)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLandscapeWithPortraitImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',200,100))
|
|
||||||
->integer($image->width())->isEqualTo(50)
|
|
||||||
->integer($image->height())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(50)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testWithSmallSquareImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x200.jpg',300,300))
|
|
||||||
->integer($image->width())->isEqualTo(300)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testWithSmallLandscapeImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../200x100.jpg',300,300))
|
|
||||||
->integer($image->width())->isEqualTo(300)
|
|
||||||
->integer($image->height())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testWithSmallPortraitImage()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg',300,300))
|
|
||||||
->integer($image->width())->isEqualTo(150)
|
|
||||||
->integer($image->height())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isEqualTo(150)
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isEqualTo(300)
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function image(string $path, int $width, int $height)
|
|
||||||
{
|
|
||||||
$driver = new MockDriver();
|
|
||||||
return $driver->image($path, new SizerUnderTest($width, $height));
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,36 +0,0 @@
|
||||||
<?php
|
|
||||||
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
|
||||||
namespace ByJoby\ImageTransform\tests\units\Sizers;
|
|
||||||
|
|
||||||
use atoum;
|
|
||||||
use ByJoby\ImageTransform\Sizers\Original as SizerUnderTest;
|
|
||||||
use ByJoby\ImageTransform\tests\units\mock\MockDriver;
|
|
||||||
|
|
||||||
class Original extends atoum
|
|
||||||
{
|
|
||||||
public function testRotation()
|
|
||||||
{
|
|
||||||
$this
|
|
||||||
->given($image = $this->image(__DIR__.'/../100x200.jpg'))
|
|
||||||
->integer($image->width())->isEqualTo(100)
|
|
||||||
->integer($image->height())->isEqualTo(200)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isNull()
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
->given($image->rotate())
|
|
||||||
->integer($image->width())->isEqualTo(200)
|
|
||||||
->integer($image->height())->isEqualTo(100)
|
|
||||||
->variable($image->sizer()->resizeToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->resizeToHeight())->isNull()
|
|
||||||
->variable($image->sizer()->cropToWidth())->isNull()
|
|
||||||
->variable($image->sizer()->cropToHeight())->isNull()
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function image(string $path)
|
|
||||||
{
|
|
||||||
$driver = new MockDriver();
|
|
||||||
return $driver->image($path, new SizerUnderTest());
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,14 +0,0 @@
|
||||||
<?php
|
|
||||||
/* image-transform | https://github.com/jobyone/image-transform | MIT License */
|
|
||||||
namespace ByJoby\ImageTransform\tests\units\mock;
|
|
||||||
|
|
||||||
use ByJoby\ImageTransform\Drivers\AbstractDriver;
|
|
||||||
use ByJoby\ImageTransform\Image;
|
|
||||||
|
|
||||||
class MockDriver extends AbstractDriver
|
|
||||||
{
|
|
||||||
protected function doSave(Image $image, string $filename)
|
|
||||||
{
|
|
||||||
//does nothing
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in a new issue