2020-03-29 19:33:16 +01:00
< ? php
2020-03-29 20:56:35 +01:00
// {{{ License
2021-04-15 23:28:28 +01:00
2020-05-20 17:53:53 +01:00
// This file is part of GNU social - https://www.gnu.org/software/social
2020-03-29 20:56:35 +01:00
//
// GNU social is free software: you can redistribute it and/or modify
2020-05-10 21:43:15 +01:00
// it under the terms of the GNU Affero General Public License as published by
2020-03-29 20:56:35 +01:00
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// GNU social is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
2020-05-10 21:43:15 +01:00
// You should have received a copy of the GNU Affero General Public License
2020-03-29 20:56:35 +01:00
// along with GNU social. If not, see <http://www.gnu.org/licenses/>.
2021-04-15 23:28:28 +01:00
2020-03-29 20:56:35 +01:00
// }}}
2020-03-29 19:33:16 +01:00
namespace App\Entity ;
2021-04-16 17:11:34 +01:00
use App\Core\Cache ;
2021-04-16 11:46:53 +01:00
use App\Core\DB\DB ;
2020-08-15 07:18:23 +01:00
use App\Core\Entity ;
2021-04-16 11:46:53 +01:00
use App\Core\Event ;
2021-04-18 02:17:57 +01:00
use App\Core\GSFile ;
2021-04-16 11:46:53 +01:00
use App\Core\Log ;
2021-04-16 16:57:25 +01:00
use App\Util\Common ;
2021-04-16 11:46:53 +01:00
use App\Util\Exception\NotFoundException ;
use App\Util\Exception\ServerException ;
2021-05-01 22:48:44 +01:00
use App\Util\TemporaryFile ;
2020-05-10 21:43:15 +01:00
use DateTimeInterface ;
2020-03-29 19:33:16 +01:00
/**
2021-04-15 23:28:28 +01:00
* Entity for Attachment thumbnails
2020-03-29 19:33:16 +01:00
*
* @ category DB
* @ package GNUsocial
*
* @ author Zach Copley < zach @ status . net >
* @ copyright 2010 StatusNet Inc .
* @ author Mikael Nordfeldth < mmn @ hethane . se >
* @ copyright 2009 - 2014 Free Software Foundation , Inc http :// www . fsf . org
2021-02-19 23:29:43 +00:00
* @ author Hugo Sales < hugo @ hsal . es >
2021-07-20 21:17:53 +01:00
* @ author Diogo Peralta Cordeiro < mail @ diogo . site >
2021-02-19 23:29:43 +00:00
* @ copyright 2020 - 2021 Free Software Foundation , Inc http :// www . fsf . org
2020-03-29 19:33:16 +01:00
* @ license https :// www . gnu . org / licenses / agpl . html GNU AGPL v3 or later
*/
2021-04-16 20:27:33 +01:00
class AttachmentThumbnail extends Entity
2020-03-29 19:33:16 +01:00
{
2020-03-30 15:00:13 +01:00
// {{{ Autocode
2021-05-05 17:03:03 +01:00
// @codeCoverageIgnoreStart
2021-04-15 23:28:28 +01:00
private int $attachment_id ;
2020-03-30 16:13:51 +01:00
private int $width ;
private int $height ;
2021-05-01 22:48:44 +01:00
private string $filename ;
2021-04-27 22:24:48 +01:00
private \DateTimeInterface $modified ;
2020-03-30 16:13:51 +01:00
2021-04-15 23:28:28 +01:00
public function setAttachmentId ( int $attachment_id ) : self
2020-03-30 16:13:51 +01:00
{
2021-04-15 23:28:28 +01:00
$this -> attachment_id = $attachment_id ;
2020-03-30 16:13:51 +01:00
return $this ;
}
2020-08-08 17:11:18 +01:00
2021-04-15 23:28:28 +01:00
public function getAttachmentId () : int
2020-03-30 16:13:51 +01:00
{
2021-04-15 23:28:28 +01:00
return $this -> attachment_id ;
2020-03-30 16:13:51 +01:00
}
public function setWidth ( int $width ) : self
{
$this -> width = $width ;
return $this ;
}
2020-08-08 17:11:18 +01:00
2020-03-30 16:13:51 +01:00
public function getWidth () : int
{
return $this -> width ;
}
public function setHeight ( int $height ) : self
{
$this -> height = $height ;
return $this ;
}
2020-08-08 17:11:18 +01:00
2020-03-30 16:13:51 +01:00
public function getHeight () : int
{
return $this -> height ;
}
2021-05-01 22:48:44 +01:00
public function setFilename ( string $filename ) : self
{
$this -> filename = $filename ;
return $this ;
}
public function getFilename () : string
{
return $this -> filename ;
}
2021-05-02 16:02:26 +01:00
public function setModified ( DateTimeInterface $modified ) : self
2020-03-30 16:13:51 +01:00
{
$this -> modified = $modified ;
return $this ;
}
2020-08-08 17:11:18 +01:00
2021-05-02 16:02:26 +01:00
public function getModified () : DateTimeInterface
2020-03-30 16:13:51 +01:00
{
return $this -> modified ;
}
2021-05-05 17:03:03 +01:00
// @codeCoverageIgnoreEnd
2020-03-30 15:00:13 +01:00
// }}} Autocode
2020-03-29 19:33:16 +01:00
2021-04-16 16:57:25 +01:00
private Attachment $attachment ;
public function setAttachment ( Attachment $attachment )
{
$this -> attachment = $attachment ;
}
public function getAttachment ()
{
if ( isset ( $this -> attachment )) {
return $this -> attachment ;
} else {
return $this -> attachment = DB :: findOneBy ( 'attachment' , [ 'id' => $this -> attachment_id ]);
}
}
2021-07-20 21:17:53 +01:00
/**
* @ param Attachment $attachment
* @ param int $width
* @ param int $height
* @ param bool $crop
*
* @ throws ServerException
* @ throws \App\Util\Exception\TemporaryFileException
*
* @ return mixed
*/
2021-05-01 22:48:44 +01:00
public static function getOrCreate ( Attachment $attachment , int $width , int $height , bool $crop )
2021-04-16 11:46:53 +01:00
{
try {
2021-04-16 17:11:34 +01:00
return Cache :: get ( 'thumb-' . $attachment -> getId () . " - { $width } x { $height } " ,
2021-05-01 22:48:44 +01:00
function () use ( $crop , $attachment , $width , $height ) {
[ $predicted_width , $predicted_height ] = self :: predictScalingValues ( $attachment -> getWidth (), $attachment -> getHeight (), $width , $height , $crop );
return DB :: findOneBy ( 'attachment_thumbnail' , [ 'attachment_id' => $attachment -> getId (), 'width' => $predicted_width , 'height' => $predicted_height ]);
});
2021-04-16 11:46:53 +01:00
} catch ( NotFoundException $e ) {
2021-05-02 16:02:26 +01:00
$ext = image_type_to_extension ( IMAGETYPE_WEBP , include_dot : true );
2021-07-20 21:17:53 +01:00
$temp = new TemporaryFile ([ 'prefix' => 'gs-thumbnail' , 'suffix' => $ext ]);
2021-05-02 16:02:26 +01:00
$thumbnail = self :: create ([ 'attachment_id' => $attachment -> getId ()]);
$event_map = [ 'image' => 'ResizeImagePath' , 'video' => 'ResizeVideoPath' ];
2021-04-18 02:17:57 +01:00
$major_mime = GSFile :: mimetypeMajor ( $attachment -> getMimetype ());
2021-07-20 21:17:53 +01:00
if ( in_array ( $major_mime , array_keys ( $event_map )) && ! Event :: handle ( $event_map [ $major_mime ], [ $attachment -> getPath (), $temp -> getRealPath (), & $width , & $height , $crop , & $mimetype ])) {
2021-05-01 22:48:44 +01:00
$thumbnail -> setWidth ( $width );
$thumbnail -> setHeight ( $height );
$filename = " { $width } x { $height } { $ext } - " . $attachment -> getFileHash ();
2021-07-20 21:17:53 +01:00
$temp -> move ( Common :: config ( 'thumbnail' , 'dir' ), $filename );
2021-05-01 22:48:44 +01:00
$thumbnail -> setFilename ( $filename );
2021-04-16 23:55:00 +01:00
DB :: persist ( $thumbnail );
DB :: flush ();
2021-04-16 16:57:25 +01:00
return $thumbnail ;
2021-04-16 11:46:53 +01:00
} else {
Log :: debug ( $m = ( 'Cannot resize attachment with mimetype ' . $attachment -> getMimetype ()));
throw new ServerException ( $m );
}
}
}
2021-04-16 16:57:25 +01:00
public function getPath ()
{
return Common :: config ( 'thumbnail' , 'dir' ) . $this -> getFilename ();
}
2021-04-25 22:26:53 +01:00
public function getUrl ()
{
return Router :: url ( 'attachment_thumbnail' , [ 'id' => $this -> getAttachmentId (), 'w' => $this -> getWidth (), 'h' => $this -> getHeight ()]);
}
/**
* Get the HTML attributes for this thumbnail
*/
public function getHTMLAttributes ( array $orig = [], bool $overwrite = true )
{
$attrs = [
'height' => $this -> getHeight (),
2021-05-02 16:02:26 +01:00
'width' => $this -> getWidth (),
'src' => $this -> getUrl (),
2021-04-25 22:26:53 +01:00
];
return $overwrite ? array_merge ( $orig , $attrs ) : array_merge ( $attrs , $orig );
}
2020-08-07 03:03:55 +01:00
/**
2021-04-29 19:12:32 +01:00
* Delete an attachment thumbnail
2020-08-07 03:03:55 +01:00
*/
2021-04-29 19:12:32 +01:00
public function delete ( bool $flush = true ) : void
2020-08-07 03:03:55 +01:00
{
2021-04-29 19:12:32 +01:00
$filepath = $this -> getPath ();
if ( file_exists ( $filepath )) {
if ( @ unlink ( $filepath ) === false ) {
Log :: warning ( " Failed deleting file for attachment thumbnail with id= { $this -> attachment_id } , width= { $this -> width } , height= { $this -> height } at { $filepath } " );
}
}
DB :: remove ( $this );
if ( $flush ) {
DB :: flush ();
}
2020-08-07 03:03:55 +01:00
}
2021-05-01 22:48:44 +01:00
/**
* Gets scaling values for images of various types . Cropping can be enabled .
*
* Values will scale _up_ to fit max values if cropping is enabled !
* With cropping disabled , the max value of each axis will be respected .
*
* @ param $width int Original width
* @ param $height int Original height
* @ param $maxW int Resulting max width
* @ param $maxH int Resulting max height
* @ param $crop bool Crop to the size ( not preserving aspect ratio )
*
* @ return array [ predicted width , predicted height ]
*/
public static function predictScalingValues (
int $width ,
int $height ,
int $maxW ,
int $maxH ,
bool $crop
2021-05-02 16:02:26 +01:00
) : array {
2021-05-01 22:48:44 +01:00
// Cropping data (for original image size). Default values, 0 and null,
// imply no cropping and with preserved aspect ratio (per axis).
$cx = 0 ; // crop x
$cy = 0 ; // crop y
$cw = null ; // crop area width
$ch = null ; // crop area height
if ( $crop ) {
$s_ar = $width / $height ;
2021-05-02 16:02:26 +01:00
$t_ar = $maxW / $maxH ;
2021-05-01 22:48:44 +01:00
$rw = $maxW ;
$rh = $maxH ;
// Source aspect ratio differs from target, recalculate crop points!
if ( $s_ar > $t_ar ) {
$cx = floor ( $width / 2 - $height * $t_ar / 2 );
$cw = ceil ( $height * $t_ar );
} elseif ( $s_ar < $t_ar ) {
$cy = floor ( $height / 2 - $width / $t_ar / 2 );
$ch = ceil ( $width / $t_ar );
}
} else {
$rw = $maxW ;
$rh = ceil ( $height * $rw / $width );
// Scaling caused too large height, decrease to max accepted value
if ( $rh > $maxH ) {
$rh = $maxH ;
$rw = ceil ( $width * $rh / $height );
}
}
2021-05-02 16:02:26 +01:00
return [( int ) $rw , ( int ) $rh ];
2021-05-01 22:48:44 +01:00
}
2020-03-29 19:33:16 +01:00
public static function schemaDef () : array
{
return [
2021-05-02 16:02:26 +01:00
'name' => 'attachment_thumbnail' ,
2020-03-29 19:33:16 +01:00
'fields' => [
2021-04-15 23:28:28 +01:00
'attachment_id' => [ 'type' => 'int' , 'foreign key' => true , 'target' => 'Attachment.id' , 'multiplicity' => 'one to one' , 'not null' => true , 'description' => 'thumbnail for what attachment' ],
2021-05-02 16:02:26 +01:00
'width' => [ 'type' => 'int' , 'not null' => true , 'description' => 'width of thumbnail' ],
'height' => [ 'type' => 'int' , 'not null' => true , 'description' => 'height of thumbnail' ],
'filename' => [ 'type' => 'varchar' , 'length' => 191 , 'not null' => true , 'description' => 'thubmnail filename' ],
'modified' => [ 'type' => 'timestamp' , 'not null' => true , 'default' => 'CURRENT_TIMESTAMP' , 'description' => 'date this record was modified' ],
2020-03-29 19:33:16 +01:00
],
2021-04-15 23:28:28 +01:00
'primary key' => [ 'attachment_id' , 'width' , 'height' ],
2021-05-02 16:02:26 +01:00
'indexes' => [
2021-04-15 23:28:28 +01:00
'attachment_thumbnail_attachment_id_idx' => [ 'attachment_id' ],
2020-06-30 17:26:40 +01:00
],
2020-03-29 19:33:16 +01:00
];
}
2020-06-30 17:26:40 +01:00
}