2020-03-30 03:33:16 +09:00
< ? php
2020-03-30 04:56:35 +09:00
// {{{ License
2021-04-16 07:28:28 +09:00
2020-05-21 01:53:53 +09:00
// This file is part of GNU social - https://www.gnu.org/software/social
2020-03-30 04:56:35 +09:00
//
// GNU social is free software: you can redistribute it and/or modify
2020-05-11 05:43:15 +09:00
// it under the terms of the GNU Affero General Public License as published by
2020-03-30 04:56:35 +09: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-11 05:43:15 +09:00
// You should have received a copy of the GNU Affero General Public License
2020-03-30 04:56:35 +09:00
// along with GNU social. If not, see <http://www.gnu.org/licenses/>.
2021-04-16 07:28:28 +09:00
2020-03-30 04:56:35 +09:00
// }}}
2020-03-30 03:33:16 +09:00
namespace App\Entity ;
2021-04-17 01:11:34 +09:00
use App\Core\Cache ;
2021-04-16 19:46:53 +09:00
use App\Core\DB\DB ;
2020-08-15 15:18:23 +09:00
use App\Core\Entity ;
2021-04-16 19:46:53 +09:00
use App\Core\Event ;
2021-04-18 10:17:57 +09:00
use App\Core\GSFile ;
2021-07-23 04:56:29 +09:00
use function App\Core\I18n\_m ;
2021-04-16 19:46:53 +09:00
use App\Core\Log ;
2021-04-17 00:57:25 +09:00
use App\Util\Common ;
2021-07-23 04:56:29 +09:00
use App\Util\Exception\ClientException ;
2021-04-16 19:46:53 +09:00
use App\Util\Exception\NotFoundException ;
use App\Util\Exception\ServerException ;
2020-05-11 05:43:15 +09:00
use DateTimeInterface ;
2021-07-23 04:56:29 +09:00
use Symfony\Component\Mime\MimeTypes ;
2020-05-11 05:43:15 +09:00
2020-03-30 03:33:16 +09:00
/**
2021-04-16 07:28:28 +09:00
* Entity for Attachment thumbnails
2020-03-30 03:33:16 +09: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-20 08:29:43 +09:00
* @ author Hugo Sales < hugo @ hsal . es >
2021-07-21 05:17:53 +09:00
* @ author Diogo Peralta Cordeiro < mail @ diogo . site >
2021-02-20 08:29:43 +09:00
* @ copyright 2020 - 2021 Free Software Foundation , Inc http :// www . fsf . org
2020-03-30 03:33:16 +09:00
* @ license https :// www . gnu . org / licenses / agpl . html GNU AGPL v3 or later
*/
2021-04-17 04:27:33 +09:00
class AttachmentThumbnail extends Entity
2020-03-30 03:33:16 +09:00
{
2020-03-30 23:00:13 +09:00
// {{{ Autocode
2021-05-06 01:03:03 +09:00
// @codeCoverageIgnoreStart
2021-04-16 07:28:28 +09:00
private int $attachment_id ;
2021-07-23 04:56:29 +09:00
private ? string $mimetype ;
2020-03-31 00:13:51 +09:00
private int $width ;
private int $height ;
2021-05-02 06:48:44 +09:00
private string $filename ;
2021-04-28 06:24:48 +09:00
private \DateTimeInterface $modified ;
2020-03-31 00:13:51 +09:00
2021-04-16 07:28:28 +09:00
public function setAttachmentId ( int $attachment_id ) : self
2020-03-31 00:13:51 +09:00
{
2021-04-16 07:28:28 +09:00
$this -> attachment_id = $attachment_id ;
2020-03-31 00:13:51 +09:00
return $this ;
}
2020-08-09 01:11:18 +09:00
2021-04-16 07:28:28 +09:00
public function getAttachmentId () : int
2020-03-31 00:13:51 +09:00
{
2021-04-16 07:28:28 +09:00
return $this -> attachment_id ;
2020-03-31 00:13:51 +09:00
}
2021-07-23 04:56:29 +09:00
public function setMimetype ( ? string $mimetype ) : self
{
$this -> mimetype = $mimetype ;
return $this ;
}
public function getMimetype () : ? string
{
return $this -> mimetype ;
}
2020-03-31 00:13:51 +09:00
public function setWidth ( int $width ) : self
{
$this -> width = $width ;
return $this ;
}
2020-08-09 01:11:18 +09:00
2020-03-31 00:13:51 +09:00
public function getWidth () : int
{
return $this -> width ;
}
public function setHeight ( int $height ) : self
{
$this -> height = $height ;
return $this ;
}
2020-08-09 01:11:18 +09:00
2020-03-31 00:13:51 +09:00
public function getHeight () : int
{
return $this -> height ;
}
2021-05-02 06:48:44 +09:00
public function setFilename ( string $filename ) : self
{
$this -> filename = $filename ;
return $this ;
}
public function getFilename () : string
{
return $this -> filename ;
}
2021-05-03 00:02:26 +09:00
public function setModified ( DateTimeInterface $modified ) : self
2020-03-31 00:13:51 +09:00
{
$this -> modified = $modified ;
return $this ;
}
2020-08-09 01:11:18 +09:00
2021-05-03 00:02:26 +09:00
public function getModified () : DateTimeInterface
2020-03-31 00:13:51 +09:00
{
return $this -> modified ;
}
2021-05-06 01:03:03 +09:00
// @codeCoverageIgnoreEnd
2020-03-30 23:00:13 +09:00
// }}} Autocode
2020-03-30 03:33:16 +09:00
2021-04-17 00:57:25 +09: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-21 05:17:53 +09:00
/**
* @ param Attachment $attachment
* @ param int $width
* @ param int $height
* @ param bool $crop
*
* @ throws ServerException
* @ throws \App\Util\Exception\TemporaryFileException
*
* @ return mixed
*/
2021-05-02 06:48:44 +09:00
public static function getOrCreate ( Attachment $attachment , int $width , int $height , bool $crop )
2021-04-16 19:46:53 +09:00
{
2021-07-21 07:31:53 +09:00
// We need to keep these in mind for DB indexing
$predicted_width = null ;
$predicted_height = null ;
2021-04-16 19:46:53 +09:00
try {
2021-04-17 01:11:34 +09:00
return Cache :: get ( 'thumb-' . $attachment -> getId () . " - { $width } x { $height } " ,
2021-07-21 07:31:53 +09:00
function () use ( $crop , $attachment , $width , $height , & $predicted_width , & $predicted_height ) {
2021-07-23 04:56:29 +09:00
[ $predicted_width , $predicted_height ] = self :: predictScalingValues ( $attachment -> getWidth (), $attachment -> getHeight (), $width , $height , $crop );
2021-05-02 06:48:44 +09:00
return DB :: findOneBy ( 'attachment_thumbnail' , [ 'attachment_id' => $attachment -> getId (), 'width' => $predicted_width , 'height' => $predicted_height ]);
});
2021-04-16 19:46:53 +09:00
} catch ( NotFoundException $e ) {
2021-07-23 04:56:29 +09:00
$thumbnail = self :: create ([ 'attachment_id' => $attachment -> getId ()]);
Event :: handle ( 'ResizerAvailable' , [ & $event_map ]);
$mimetype = $attachment -> getMimetype ();
$major_mime = GSFile :: mimetypeMajor ( $mimetype );
if ( in_array ( $major_mime , array_keys ( $event_map ))) {
$temp = null ; // Let the EncoderPlugin create a temporary file for us
if ( ! Event :: handle (
$event_map [ $major_mime ],
[ $attachment -> getPath (), & $temp , & $width , & $height , $crop , & $mimetype ]
)
) {
$thumbnail -> setWidth ( $predicted_width );
$thumbnail -> setHeight ( $predicted_height );
$ext = '.' . MimeTypes :: getDefault () -> getExtensions ( $temp -> getMimeType ())[ 0 ];
$filename = " { $predicted_width } x { $predicted_height } { $ext } - " . $attachment -> getFileHash ();
$thumbnail -> setFilename ( $filename );
DB :: persist ( $thumbnail );
DB :: flush ();
$temp -> move ( Common :: config ( 'thumbnail' , 'dir' ), $filename );
return $thumbnail ;
} else {
Log :: debug ( $m = ( 'Somehow the EncoderPlugin didn\'t handle ' . $attachment -> getMimetype ()));
throw new ServerException ( $m );
}
2021-04-16 19:46:53 +09:00
} else {
2021-07-23 04:56:29 +09:00
throw new ClientException ( _m ( 'Can not generate thumbnail for attachment with id={id}' , [ 'id' => $attachment -> getId ()]));
2021-04-16 19:46:53 +09:00
}
}
}
2021-04-17 00:57:25 +09:00
public function getPath ()
{
return Common :: config ( 'thumbnail' , 'dir' ) . $this -> getFilename ();
}
2021-04-26 06:26:53 +09: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-03 00:02:26 +09:00
'width' => $this -> getWidth (),
'src' => $this -> getUrl (),
2021-04-26 06:26:53 +09:00
];
return $overwrite ? array_merge ( $orig , $attrs ) : array_merge ( $attrs , $orig );
}
2020-08-07 11:03:55 +09:00
/**
2021-04-30 03:12:32 +09:00
* Delete an attachment thumbnail
2020-08-07 11:03:55 +09:00
*/
2021-04-30 03:12:32 +09:00
public function delete ( bool $flush = true ) : void
2020-08-07 11:03:55 +09:00
{
2021-04-30 03:12:32 +09: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 11:03:55 +09:00
}
2021-05-02 06:48:44 +09: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-03 00:02:26 +09:00
) : array {
2021-05-02 06:48:44 +09: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-03 00:02:26 +09:00
$t_ar = $maxW / $maxH ;
2021-05-02 06:48:44 +09: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-03 00:02:26 +09:00
return [( int ) $rw , ( int ) $rh ];
2021-05-02 06:48:44 +09:00
}
2020-03-30 03:33:16 +09:00
public static function schemaDef () : array
{
return [
2021-05-03 00:02:26 +09:00
'name' => 'attachment_thumbnail' ,
2020-03-30 03:33:16 +09:00
'fields' => [
2021-04-16 07:28:28 +09:00
'attachment_id' => [ 'type' => 'int' , 'foreign key' => true , 'target' => 'Attachment.id' , 'multiplicity' => 'one to one' , 'not null' => true , 'description' => 'thumbnail for what attachment' ],
2021-07-23 04:56:29 +09:00
// 'mimetype' => ['type' => 'varchar', 'length' => 50, 'description' => 'mime type of resource'],
'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' => 'thumbnail filename' ],
'modified' => [ 'type' => 'timestamp' , 'not null' => true , 'default' => 'CURRENT_TIMESTAMP' , 'description' => 'date this record was modified' ],
2020-03-30 03:33:16 +09:00
],
2021-04-16 07:28:28 +09:00
'primary key' => [ 'attachment_id' , 'width' , 'height' ],
2021-05-03 00:02:26 +09:00
'indexes' => [
2021-04-16 07:28:28 +09:00
'attachment_thumbnail_attachment_id_idx' => [ 'attachment_id' ],
2020-07-01 01:26:40 +09:00
],
2020-03-30 03:33:16 +09:00
];
}
2020-07-01 01:26:40 +09:00
}