1 | <?php |
||||||
2 | |||||||
3 | namespace Jackal\ImageMerge\Model; |
||||||
4 | |||||||
5 | use Exception; |
||||||
6 | use Jackal\ImageMerge\Builder\ImageBuilder; |
||||||
7 | |||||||
8 | use Jackal\ImageMerge\Command\Options\SingleCoordinateFileObjectCommandOption; |
||||||
9 | use Jackal\ImageMerge\Command\Asset\ImageAssetCommand; |
||||||
10 | use Jackal\ImageMerge\Exception\InvalidColorException; |
||||||
11 | use Jackal\ImageMerge\Http\Response\ImageResponse; |
||||||
12 | use Jackal\ImageMerge\Metadata\Metadata; |
||||||
13 | use Jackal\ImageMerge\Model\File\FileObjectInterface; |
||||||
14 | use Jackal\ImageMerge\Model\File\FileTempObject; |
||||||
15 | use Jackal\ImageMerge\Model\Format\ImageReader; |
||||||
16 | use Jackal\ImageMerge\Model\Format\ImageWriter; |
||||||
17 | use Jackal\ImageMerge\Utils\ColorUtils; |
||||||
18 | use Jackal\ImageMerge\ValueObject\Coordinate; |
||||||
19 | |||||||
20 | /** |
||||||
21 | * Class Image |
||||||
22 | * @package Jackal\ImageMerge\Model |
||||||
23 | */ |
||||||
24 | class Image |
||||||
25 | { |
||||||
26 | /** |
||||||
27 | * @var integer |
||||||
28 | */ |
||||||
29 | private $width; |
||||||
30 | |||||||
31 | /** |
||||||
32 | * @var integer |
||||||
33 | */ |
||||||
34 | private $height; |
||||||
35 | |||||||
36 | /** |
||||||
37 | * @var resource |
||||||
38 | */ |
||||||
39 | private $resource; |
||||||
40 | |||||||
41 | /** |
||||||
42 | * @var Metadata |
||||||
43 | */ |
||||||
44 | private $metadata; |
||||||
45 | |||||||
46 | /** |
||||||
47 | * Image constructor. |
||||||
48 | * @param $width |
||||||
49 | * @param $height |
||||||
50 | * @param bool $transparent |
||||||
51 | * @throws InvalidColorException |
||||||
52 | */ |
||||||
53 | public function __construct($width, $height, $transparent = true) |
||||||
54 | { |
||||||
55 | $this->width = $width; |
||||||
56 | $this->height = $height; |
||||||
57 | |||||||
58 | $resource = imagecreatetruecolor($this->width, $this->height); |
||||||
59 | imagecolortransparent($resource); |
||||||
0 ignored issues
–
show
Bug
introduced
by
![]() |
|||||||
60 | |||||||
61 | if ($transparent) { |
||||||
62 | imagesavealpha($resource, true); |
||||||
0 ignored issues
–
show
It seems like
$resource can also be of type false ; however, parameter $image of imagesavealpha() does only seem to accept resource , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
63 | $color = ColorUtils::colorIdentifier($resource, new Color(Color::BLACK), true); |
||||||
64 | imagefill($resource, 0, 0, $color); |
||||||
0 ignored issues
–
show
It seems like
$resource can also be of type false ; however, parameter $image of imagefill() does only seem to accept resource , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
65 | } |
||||||
66 | |||||||
67 | $this->resource = $resource; |
||||||
0 ignored issues
–
show
It seems like
$resource can also be of type false . However, the property $resource is declared as type resource . Maybe add an additional type check?
Our type inference engine has found a suspicous assignment of a value to a property. This check raises an issue when a value that can be of a mixed type is assigned to a property that is type hinted more strictly. For example, imagine you have a variable Either this assignment is in error or a type check should be added for that assignment. class Id
{
public $id;
public function __construct($id)
{
$this->id = $id;
}
}
class Account
{
/** @var Id $id */
public $id;
}
$account_id = false;
if (starsAreRight()) {
$account_id = new Id(42);
}
$account = new Account();
if ($account instanceof Id)
{
$account->id = $account_id;
}
![]() |
|||||||
68 | } |
||||||
69 | |||||||
70 | /** |
||||||
71 | * @param FileObjectInterface $filePathName |
||||||
72 | * @return Image |
||||||
73 | * @throws Exception |
||||||
74 | */ |
||||||
75 | public static function fromFile(FileObjectInterface $filePathName) |
||||||
76 | { |
||||||
77 | $resource = ImageReader::fromPathname($filePathName); |
||||||
78 | $imageResource = $resource->getResource(); |
||||||
79 | |||||||
80 | $image = new self(imagesx($imageResource), imagesy($imageResource)); |
||||||
81 | $command = new ImageAssetCommand(new SingleCoordinateFileObjectCommandOption($filePathName, new Coordinate(0, 0))); |
||||||
82 | $command->execute($image); |
||||||
83 | |||||||
84 | return $image; |
||||||
85 | } |
||||||
86 | |||||||
87 | /** |
||||||
88 | * @param $contentString |
||||||
89 | * @return Image |
||||||
90 | * @throws Exception |
||||||
91 | */ |
||||||
92 | public static function fromString($contentString) |
||||||
93 | { |
||||||
94 | $file = FileTempObject::fromString($contentString); |
||||||
95 | $resource = ImageReader::fromPathname($file); |
||||||
96 | |||||||
97 | $image = new self(imagesx($resource->getResource()), imagesy($resource->getResource())); |
||||||
98 | $command = new ImageAssetCommand(new SingleCoordinateFileObjectCommandOption($file, new Coordinate(0, 0))); |
||||||
99 | $command->execute($image); |
||||||
100 | |||||||
101 | return $image; |
||||||
102 | } |
||||||
103 | |||||||
104 | /** |
||||||
105 | * @param $resource |
||||||
106 | * @return Image |
||||||
107 | */ |
||||||
108 | public function assignResource($resource) |
||||||
109 | { |
||||||
110 | $this->resource = $resource; |
||||||
111 | |||||||
112 | return $this; |
||||||
113 | } |
||||||
114 | |||||||
115 | /** |
||||||
116 | * @param null $fromX |
||||||
0 ignored issues
–
show
|
|||||||
117 | * @param null $fromY |
||||||
0 ignored issues
–
show
|
|||||||
118 | * @param null $width |
||||||
0 ignored issues
–
show
|
|||||||
119 | * @param null $height |
||||||
0 ignored issues
–
show
|
|||||||
120 | * @return bool |
||||||
121 | * @throws Exception |
||||||
122 | */ |
||||||
123 | public function isDark($fromX = null, $fromY = null, $width = null, $height = null) |
||||||
124 | { |
||||||
125 | $samples = 10; |
||||||
126 | $threshold = 60; |
||||||
127 | |||||||
128 | if (!is_null($fromY) and !is_null($fromY) and !is_null($width) and !is_null($height)) { |
||||||
0 ignored issues
–
show
|
|||||||
129 | $builder = new ImageBuilder(clone $this); |
||||||
130 | $builder->crop($fromX, $fromY, $width, $height); |
||||||
131 | $portion = $builder->getImage(); |
||||||
132 | } else { |
||||||
133 | $portion = $this; |
||||||
134 | } |
||||||
135 | |||||||
136 | $luminance = 0; |
||||||
137 | for ($x = 1;$x <= $samples;$x++) { |
||||||
138 | for ($y = 1;$y <= $samples;$y++) { |
||||||
139 | $coordX = round($portion->getWidth() / $samples * $x) - ($portion->getWidth() / $samples / 2); |
||||||
140 | $cooordY = round($portion->getHeight() / $samples * $y) - ($portion->getHeight() / $samples / 2); |
||||||
141 | $rgb = imagecolorat($portion->getResource(), $coordX, $cooordY); |
||||||
0 ignored issues
–
show
$cooordY of type double is incompatible with the type integer expected by parameter $y of imagecolorat() .
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() $coordX of type double is incompatible with the type integer expected by parameter $x of imagecolorat() .
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
142 | $r = ($rgb >> 16) & 0xFF; |
||||||
143 | $g = ($rgb >> 8) & 0xFF; |
||||||
144 | $b = $rgb & 0xFF; |
||||||
145 | |||||||
146 | // choose a simple luminance formula from here |
||||||
147 | // http://stackoverflow.com/questions/596216/formula-to-determine-brightness-of-rgb-color |
||||||
148 | $luminance += ($r + $r + $b + $g + $g + $g) / 6; |
||||||
149 | } |
||||||
150 | } |
||||||
151 | |||||||
152 | return $luminance / ($samples * $samples) <= $threshold; |
||||||
153 | } |
||||||
154 | |||||||
155 | /** |
||||||
156 | * @return resource |
||||||
157 | */ |
||||||
158 | public function getResource() |
||||||
159 | { |
||||||
160 | return $this->resource; |
||||||
161 | } |
||||||
162 | |||||||
163 | public function getResourceClone() |
||||||
164 | { |
||||||
165 | $original = $this->resource; |
||||||
166 | $copy = imagecreatetruecolor($this->width, $this->height); |
||||||
167 | |||||||
168 | imagecopy($copy, $original, 0, 0, 0, 0, $this->width, $this->height); |
||||||
0 ignored issues
–
show
It seems like
$copy can also be of type false ; however, parameter $dst_im of imagecopy() does only seem to accept resource , maybe add an additional type check?
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
![]() |
|||||||
169 | |||||||
170 | return $copy; |
||||||
171 | } |
||||||
172 | |||||||
173 | /** |
||||||
174 | * @param null $filePathName |
||||||
0 ignored issues
–
show
|
|||||||
175 | * @return bool|ImageResponse |
||||||
176 | * @throws Exception |
||||||
177 | */ |
||||||
178 | public function toPNG($filePathName = null) |
||||||
179 | { |
||||||
180 | return ImageWriter::toPNG($this->getResource(), $filePathName); |
||||||
181 | } |
||||||
182 | |||||||
183 | /** |
||||||
184 | * @param null $filePathName |
||||||
0 ignored issues
–
show
|
|||||||
185 | * @return bool|ImageResponse |
||||||
186 | * @throws Exception |
||||||
187 | */ |
||||||
188 | public function toJPG($filePathName = null) |
||||||
189 | { |
||||||
190 | return ImageWriter::toJPG($this->getResource(), $filePathName); |
||||||
191 | } |
||||||
192 | |||||||
193 | /** |
||||||
194 | * @param null $filePathName |
||||||
0 ignored issues
–
show
|
|||||||
195 | * @return bool|ImageResponse |
||||||
196 | * @throws Exception |
||||||
197 | */ |
||||||
198 | public function toGIF($filePathName = null) |
||||||
199 | { |
||||||
200 | return ImageWriter::toGIF($this->getResource(), $filePathName); |
||||||
201 | } |
||||||
202 | |||||||
203 | /** |
||||||
204 | * @param null $filePathName |
||||||
0 ignored issues
–
show
|
|||||||
205 | * @return bool|ImageResponse |
||||||
206 | * @throws Exception |
||||||
207 | */ |
||||||
208 | public function toWebP($filePathName = null){ |
||||||
209 | return ImageWriter::toWebP($this->getResource(), $filePathName); |
||||||
210 | } |
||||||
211 | |||||||
212 | /** |
||||||
213 | * @return mixed |
||||||
214 | */ |
||||||
215 | public function getWidth() |
||||||
216 | { |
||||||
217 | return imagesx($this->getResource()); |
||||||
218 | } |
||||||
219 | |||||||
220 | /** |
||||||
221 | * @return mixed |
||||||
222 | */ |
||||||
223 | public function getHeight() |
||||||
224 | { |
||||||
225 | return imagesy($this->getResource()); |
||||||
226 | } |
||||||
227 | |||||||
228 | /** |
||||||
229 | * @return float |
||||||
230 | */ |
||||||
231 | public function getAspectRatio() |
||||||
232 | { |
||||||
233 | return $this->getWidth() / $this->getHeight(); |
||||||
234 | } |
||||||
235 | |||||||
236 | /** |
||||||
237 | * @return bool |
||||||
238 | */ |
||||||
239 | public function isVertical() |
||||||
240 | { |
||||||
241 | return $this->getAspectRatio() < 1; |
||||||
242 | } |
||||||
243 | |||||||
244 | /** |
||||||
245 | * @return bool |
||||||
246 | */ |
||||||
247 | public function isHorizontal() |
||||||
248 | { |
||||||
249 | return $this->getAspectRatio() > 1; |
||||||
250 | } |
||||||
251 | |||||||
252 | /** |
||||||
253 | * @return bool |
||||||
254 | */ |
||||||
255 | public function isSquare() |
||||||
256 | { |
||||||
257 | return $this->getAspectRatio() == 1; |
||||||
258 | } |
||||||
259 | |||||||
260 | public function addMetadata(Metadata $metadata) |
||||||
261 | { |
||||||
262 | $this->metadata = $metadata; |
||||||
263 | } |
||||||
264 | |||||||
265 | /** |
||||||
266 | * @return Metadata |
||||||
267 | */ |
||||||
268 | public function getMetadata() |
||||||
269 | { |
||||||
270 | return $this->metadata; |
||||||
271 | } |
||||||
272 | } |
||||||
273 |