2021-12-14 18:32:47 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
2012-08-18 21:10:35 +01:00
|
|
|
|
2023-01-10 22:44:09 +00:00
|
|
|
namespace Shimmie2;
|
|
|
|
|
2024-02-10 18:35:55 +00:00
|
|
|
use function MicroHTML\{INPUT};
|
|
|
|
|
2020-06-14 11:05:55 -05:00
|
|
|
// TODO Add warning that rotate doesn't support lossless webp output
|
|
|
|
|
2012-08-18 21:10:35 +01:00
|
|
|
/**
|
|
|
|
* This class is just a wrapper around SCoreException.
|
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
class ImageRotateException extends SCoreException
|
|
|
|
{
|
2012-08-18 21:10:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This class handles image rotate requests.
|
|
|
|
*/
|
2019-05-28 17:59:38 +01:00
|
|
|
class RotateImage extends Extension
|
|
|
|
{
|
2021-12-14 18:32:47 +00:00
|
|
|
public const SUPPORTED_MIME = [MimeType::JPEG, MimeType::PNG, MimeType::GIF, MimeType::WEBP];
|
2019-06-18 08:06:05 -05:00
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onInitExt(InitExtEvent $event): void
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
|
|
|
global $config;
|
|
|
|
$config->set_default_bool('rotate_enabled', true);
|
|
|
|
$config->set_default_int('rotate_default_deg', 180);
|
|
|
|
}
|
2012-08-18 21:10:35 +01:00
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onImageAdminBlockBuilding(ImageAdminBlockBuildingEvent $event): void
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
|
|
|
global $user, $config;
|
2019-09-29 19:00:51 +01:00
|
|
|
if ($user->can(Permissions::EDIT_FILES) && $config->get_bool("rotate_enabled")
|
2020-06-14 11:05:55 -05:00
|
|
|
&& MimeType::matches_array($event->image->get_mime(), self::SUPPORTED_MIME)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
/* Add a link to rotate the image */
|
2024-02-10 18:35:55 +00:00
|
|
|
$event->add_part(SHM_SIMPLE_FORM(
|
|
|
|
'rotate/'.$event->image->id,
|
|
|
|
INPUT(["type" => 'number', "name" => 'rotate_deg', "id" => "rotate_deg", "placeholder" => "Rotation degrees"]),
|
|
|
|
INPUT(["type" => 'submit', "value" => 'Rotate', "id" => "rotatebutton"]),
|
|
|
|
));
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onSetupBuilding(SetupBuildingEvent $event): void
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2020-10-26 10:13:28 -05:00
|
|
|
$sb = $event->panel->create_new_block("Image Rotate");
|
2019-05-28 17:59:38 +01:00
|
|
|
$sb->add_bool_option("rotate_enabled", "Allow rotating images: ");
|
|
|
|
$sb->add_label("<br>Default Orientation: ");
|
|
|
|
$sb->add_int_option("rotate_default_deg");
|
|
|
|
$sb->add_label(" deg");
|
|
|
|
}
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPageRequest(PageRequestEvent $event): void
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
|
|
|
global $page, $user;
|
2012-08-18 21:10:35 +01:00
|
|
|
|
2024-02-11 11:34:09 +00:00
|
|
|
if ($event->page_matches("rotate/{image_id}", method: "POST", permission: Permissions::EDIT_FILES)) {
|
2019-05-28 17:59:38 +01:00
|
|
|
// Try to get the image ID
|
2024-02-11 11:34:09 +00:00
|
|
|
$image_id = $event->get_iarg('image_id');
|
2024-02-20 00:22:25 +00:00
|
|
|
$image = Image::by_id_ex($image_id);
|
|
|
|
/* Check if options were given to rotate an image. */
|
|
|
|
$deg = int_escape($event->req_POST('rotate_deg'));
|
|
|
|
|
|
|
|
/* Attempt to rotate the image */
|
|
|
|
$this->rotate_image($image_id, $deg);
|
|
|
|
$page->set_mode(PageMode::REDIRECT);
|
|
|
|
$page->set_redirect(make_link("post/view/".$image_id));
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
2019-08-07 14:53:59 -05:00
|
|
|
|
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
// Private functions
|
|
|
|
/* ----------------------------- */
|
2024-01-20 14:10:59 +00:00
|
|
|
private function rotate_image(int $image_id, int $deg): void
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
|
|
|
if (($deg <= -360) || ($deg >= 360)) {
|
|
|
|
throw new ImageRotateException("Invalid options for rotation angle. ($deg)");
|
|
|
|
}
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2024-02-20 00:22:25 +00:00
|
|
|
$image_obj = Image::by_id_ex($image_id);
|
2019-05-28 17:59:38 +01:00
|
|
|
$hash = $image_obj->hash;
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2019-06-15 11:18:52 -05:00
|
|
|
$image_filename = warehouse_path(Image::IMAGE_DIR, $hash);
|
2023-11-11 21:49:12 +00:00
|
|
|
if (file_exists($image_filename) === false) {
|
2019-05-28 17:59:38 +01:00
|
|
|
throw new ImageRotateException("$image_filename does not exist.");
|
|
|
|
}
|
2019-06-09 13:22:48 -05:00
|
|
|
|
2024-02-20 00:22:25 +00:00
|
|
|
$info = \Safe\getimagesize($image_filename);
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2024-02-20 00:22:25 +00:00
|
|
|
// we need to fully-enable phpstan-safe-rules to get the
|
|
|
|
// full type hint
|
|
|
|
// @phpstan-ignore-next-line
|
2019-06-24 10:05:16 -05:00
|
|
|
$memory_use = Media::calc_memory_use($info);
|
2019-05-28 17:59:38 +01:00
|
|
|
$memory_limit = get_memory_limit();
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
if ($memory_use > $memory_limit) {
|
|
|
|
throw new ImageRotateException("The image is too large to rotate given the memory limits. ($memory_use > $memory_limit)");
|
|
|
|
}
|
2019-08-07 14:53:59 -05:00
|
|
|
|
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
/* Attempt to load the image */
|
2024-02-20 00:22:25 +00:00
|
|
|
$image = imagecreatefromstring(\Safe\file_get_contents($image_filename));
|
2019-06-14 13:16:58 +01:00
|
|
|
if ($image == false) {
|
|
|
|
throw new ImageRotateException("Could not load image: ".$image_filename);
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2019-06-09 13:22:48 -05:00
|
|
|
$background_color = 0;
|
2019-06-14 13:47:50 +01:00
|
|
|
switch ($info[2]) {
|
2019-06-09 13:22:48 -05:00
|
|
|
case IMAGETYPE_PNG:
|
|
|
|
case IMAGETYPE_WEBP:
|
|
|
|
$background_color = imagecolorallocatealpha($image, 0, 0, 0, 127);
|
|
|
|
break;
|
|
|
|
}
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($background_color === false) {
|
2019-06-12 17:40:43 -05:00
|
|
|
throw new ImageRotateException("Unable to allocate transparent color");
|
|
|
|
}
|
|
|
|
|
2019-06-09 13:22:48 -05:00
|
|
|
$image_rotated = imagerotate($image, $deg, $background_color);
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($image_rotated === false) {
|
2019-06-12 17:40:43 -05:00
|
|
|
throw new ImageRotateException("Image rotate failed");
|
|
|
|
}
|
2019-06-09 13:22:48 -05:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
/* Temp storage while we rotate */
|
2024-01-20 20:48:47 +00:00
|
|
|
$tmp_filename = shm_tempnam('rotate');
|
2019-05-28 17:59:38 +01:00
|
|
|
if (empty($tmp_filename)) {
|
|
|
|
throw new ImageRotateException("Unable to save temporary image file.");
|
|
|
|
}
|
2019-08-07 14:53:59 -05:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
/* Output to the same format as the original image */
|
|
|
|
switch ($info[2]) {
|
2022-10-27 17:29:24 +01:00
|
|
|
case IMAGETYPE_GIF:
|
|
|
|
$result = imagegif($image_rotated, $tmp_filename);
|
|
|
|
break;
|
|
|
|
case IMAGETYPE_JPEG:
|
|
|
|
$result = imagejpeg($image_rotated, $tmp_filename);
|
|
|
|
break;
|
|
|
|
case IMAGETYPE_PNG:
|
|
|
|
$result = imagepng($image_rotated, $tmp_filename, 9);
|
|
|
|
break;
|
|
|
|
case IMAGETYPE_WEBP:
|
|
|
|
$result = imagewebp($image_rotated, $tmp_filename);
|
|
|
|
break;
|
|
|
|
case IMAGETYPE_BMP:
|
|
|
|
$result = imagebmp($image_rotated, $tmp_filename, true);
|
|
|
|
break;
|
2022-10-27 17:21:46 +01:00
|
|
|
default:
|
|
|
|
throw new ImageRotateException("Unsupported image type.");
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2019-06-09 13:22:48 -05:00
|
|
|
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($result === false) {
|
2019-06-09 13:22:48 -05:00
|
|
|
throw new ImageRotateException("Could not save image: ".$tmp_filename);
|
|
|
|
}
|
|
|
|
|
2024-02-20 00:22:25 +00:00
|
|
|
$new_hash = \Safe\md5_file($tmp_filename);
|
2019-05-28 17:59:38 +01:00
|
|
|
/* Move the new image into the main storage location */
|
2024-01-09 02:33:14 +00:00
|
|
|
$target = warehouse_path(Image::IMAGE_DIR, $new_hash);
|
2019-05-28 17:59:38 +01:00
|
|
|
if (!@copy($tmp_filename, $target)) {
|
|
|
|
throw new ImageRotateException("Failed to copy new image file from temporary location ({$tmp_filename}) to archive ($target)");
|
|
|
|
}
|
|
|
|
|
2024-01-09 02:33:14 +00:00
|
|
|
send_event(new ImageReplaceEvent($image_obj, $tmp_filename));
|
2019-06-09 13:22:48 -05:00
|
|
|
|
2024-01-09 02:33:14 +00:00
|
|
|
log_info("rotate", "Rotated >>{$image_id} - New hash: {$new_hash}");
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
2012-08-18 21:10:35 +01:00
|
|
|
}
|