18ce3bd73SGreg Roach<?php 28ce3bd73SGreg Roach 38ce3bd73SGreg Roach/** 48ce3bd73SGreg Roach * webtrees: online genealogy 589f7189bSGreg Roach * Copyright (C) 2021 webtrees development team 68ce3bd73SGreg Roach * This program is free software: you can redistribute it and/or modify 78ce3bd73SGreg Roach * it under the terms of the GNU General Public License as published by 88ce3bd73SGreg Roach * the Free Software Foundation, either version 3 of the License, or 98ce3bd73SGreg Roach * (at your option) any later version. 108ce3bd73SGreg Roach * This program is distributed in the hope that it will be useful, 118ce3bd73SGreg Roach * but WITHOUT ANY WARRANTY; without even the implied warranty of 128ce3bd73SGreg Roach * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 138ce3bd73SGreg Roach * GNU General Public License for more details. 148ce3bd73SGreg Roach * You should have received a copy of the GNU General Public License 1589f7189bSGreg Roach * along with this program. If not, see <https://www.gnu.org/licenses/>. 168ce3bd73SGreg Roach */ 178ce3bd73SGreg Roach 188ce3bd73SGreg Roachdeclare(strict_types=1); 198ce3bd73SGreg Roach 208ce3bd73SGreg Roachnamespace Fisharebest\Webtrees\Http\RequestHandlers; 218ce3bd73SGreg Roach 2281b729d3SGreg Roachuse Fisharebest\Webtrees\Http\Exceptions\HttpNotFoundException; 238ce3bd73SGreg Roachuse Fisharebest\Webtrees\I18N; 248ce3bd73SGreg Roachuse Fisharebest\Webtrees\Media; 258ce3bd73SGreg Roachuse Fisharebest\Webtrees\Mime; 268ce3bd73SGreg Roachuse Fisharebest\Webtrees\Registry; 278ce3bd73SGreg Roachuse Fisharebest\Webtrees\Services\DatatablesService; 288ce3bd73SGreg Roachuse Fisharebest\Webtrees\Services\MediaFileService; 298ce3bd73SGreg Roachuse Fisharebest\Webtrees\Services\TreeService; 308ce3bd73SGreg Roachuse Illuminate\Database\Capsule\Manager as DB; 318ce3bd73SGreg Roachuse Illuminate\Database\Query\Builder; 328ce3bd73SGreg Roachuse Illuminate\Database\Query\Expression; 338ce3bd73SGreg Roachuse Illuminate\Database\Query\JoinClause; 34f0448b68SGreg Roachuse League\Flysystem\FilesystemException; 35f7cf8a15SGreg Roachuse League\Flysystem\FilesystemOperator; 36f0448b68SGreg Roachuse League\Flysystem\UnableToCheckFileExistence; 37f32d77e6SGreg Roachuse League\Flysystem\UnableToReadFile; 38f7cf8a15SGreg Roachuse League\Flysystem\UnableToRetrieveMetadata; 398ce3bd73SGreg Roachuse Psr\Http\Message\ResponseInterface; 408ce3bd73SGreg Roachuse Psr\Http\Message\ServerRequestInterface; 418ce3bd73SGreg Roachuse Psr\Http\Server\RequestHandlerInterface; 428ce3bd73SGreg Roachuse Throwable; 438ce3bd73SGreg Roach 448ce3bd73SGreg Roachuse function assert; 458ce3bd73SGreg Roachuse function e; 46f7cf8a15SGreg Roachuse function getimagesizefromstring; 478ce3bd73SGreg Roachuse function intdiv; 488ce3bd73SGreg Roachuse function route; 498ce3bd73SGreg Roachuse function str_starts_with; 508ce3bd73SGreg Roachuse function strlen; 518ce3bd73SGreg Roachuse function substr; 528ce3bd73SGreg Roachuse function view; 538ce3bd73SGreg Roach 548ce3bd73SGreg Roach/** 558ce3bd73SGreg Roach * Manage media from the control panel. 568ce3bd73SGreg Roach */ 578ce3bd73SGreg Roachclass ManageMediaData implements RequestHandlerInterface 588ce3bd73SGreg Roach{ 59c4943cffSGreg Roach private DatatablesService $datatables_service; 608ce3bd73SGreg Roach 61c4943cffSGreg Roach private MediaFileService $media_file_service; 628ce3bd73SGreg Roach 63c4943cffSGreg Roach private TreeService $tree_service; 648ce3bd73SGreg Roach 658ce3bd73SGreg Roach /** 668ce3bd73SGreg Roach * MediaController constructor. 678ce3bd73SGreg Roach * 688ce3bd73SGreg Roach * @param DatatablesService $datatables_service 698ce3bd73SGreg Roach * @param MediaFileService $media_file_service 708ce3bd73SGreg Roach * @param TreeService $tree_service 718ce3bd73SGreg Roach */ 728ce3bd73SGreg Roach public function __construct( 738ce3bd73SGreg Roach DatatablesService $datatables_service, 748ce3bd73SGreg Roach MediaFileService $media_file_service, 758ce3bd73SGreg Roach TreeService $tree_service 768ce3bd73SGreg Roach ) { 778ce3bd73SGreg Roach $this->datatables_service = $datatables_service; 788ce3bd73SGreg Roach $this->media_file_service = $media_file_service; 798ce3bd73SGreg Roach $this->tree_service = $tree_service; 808ce3bd73SGreg Roach } 818ce3bd73SGreg Roach 828ce3bd73SGreg Roach /** 838ce3bd73SGreg Roach * @param ServerRequestInterface $request 848ce3bd73SGreg Roach * 858ce3bd73SGreg Roach * @return ResponseInterface 868ce3bd73SGreg Roach */ 878ce3bd73SGreg Roach public function handle(ServerRequestInterface $request): ResponseInterface 888ce3bd73SGreg Roach { 898ce3bd73SGreg Roach $data_filesystem = Registry::filesystem()->data(); 908ce3bd73SGreg Roach 918ce3bd73SGreg Roach $files = $request->getQueryParams()['files']; // local|external|unused 928ce3bd73SGreg Roach 938ce3bd73SGreg Roach // Files within this folder 948ce3bd73SGreg Roach $media_folder = $request->getQueryParams()['media_folder']; 958ce3bd73SGreg Roach 968ce3bd73SGreg Roach // Show sub-folders within $media_folder 978ce3bd73SGreg Roach $subfolders = $request->getQueryParams()['subfolders']; // include|exclude 988ce3bd73SGreg Roach 998ce3bd73SGreg Roach $search_columns = ['multimedia_file_refn', 'descriptive_title']; 1008ce3bd73SGreg Roach 1018ce3bd73SGreg Roach $sort_columns = [ 1028ce3bd73SGreg Roach 0 => 'multimedia_file_refn', 10396c9e653SGreg Roach 2 => new Expression('descriptive_title || multimedia_file_refn'), 1048ce3bd73SGreg Roach ]; 1058ce3bd73SGreg Roach 1068ce3bd73SGreg Roach // Convert a row from the database into a row for datatables 107f70bcff5SGreg Roach $callback = function (object $row): array { 1088ce3bd73SGreg Roach $tree = $this->tree_service->find((int) $row->m_file); 1098ce3bd73SGreg Roach $media = Registry::mediaFactory()->make($row->m_id, $tree, $row->m_gedcom); 1108ce3bd73SGreg Roach assert($media instanceof Media); 1118ce3bd73SGreg Roach 112f41d9852SGreg Roach $is_http = str_starts_with($row->multimedia_file_refn, 'http://'); 113f41d9852SGreg Roach $is_https = str_starts_with($row->multimedia_file_refn, 'https://'); 114f41d9852SGreg Roach 115f41d9852SGreg Roach if ($is_http || $is_https) { 1162fad1159SGreg Roach return [ 1172fad1159SGreg Roach '<a href="' . e($row->multimedia_file_refn) . '">' . e($row->multimedia_file_refn) . '</a>', 1182fad1159SGreg Roach view('icons/mime', ['type' => Mime::DEFAULT_TYPE]), 1192fad1159SGreg Roach $this->mediaObjectInfo($media), 1202fad1159SGreg Roach ]; 1212fad1159SGreg Roach } 1222fad1159SGreg Roach 123f41d9852SGreg Roach try { 1248ce3bd73SGreg Roach $path = $row->media_folder . $row->multimedia_file_refn; 1258ce3bd73SGreg Roach 1268ce3bd73SGreg Roach try { 127f7cf8a15SGreg Roach $mime_type = Registry::filesystem()->data()->mimeType($path); 128f7cf8a15SGreg Roach } catch (UnableToRetrieveMetadata $ex) { 129f7cf8a15SGreg Roach $mime_type = Mime::DEFAULT_TYPE; 130f7cf8a15SGreg Roach } 1318ce3bd73SGreg Roach 1328ce3bd73SGreg Roach if (str_starts_with($mime_type, 'image/')) { 1338ce3bd73SGreg Roach $url = route(AdminMediaFileThumbnail::class, ['path' => $path]); 1348ce3bd73SGreg Roach $img = '<img src="' . e($url) . '">'; 1358ce3bd73SGreg Roach } else { 1368ce3bd73SGreg Roach $img = view('icons/mime', ['type' => $mime_type]); 1378ce3bd73SGreg Roach } 1388ce3bd73SGreg Roach 1398ce3bd73SGreg Roach $url = route(AdminMediaFileDownload::class, ['path' => $path]); 1408ce3bd73SGreg Roach $img = '<a href="' . e($url) . '" type="' . $mime_type . '" class="gallery">' . $img . '</a>'; 141f32d77e6SGreg Roach } catch (UnableToReadFile $ex) { 1428ce3bd73SGreg Roach $url = route(AdminMediaFileThumbnail::class, ['path' => $path]); 1438ce3bd73SGreg Roach $img = '<img src="' . e($url) . '">'; 1448ce3bd73SGreg Roach } 1458ce3bd73SGreg Roach 1468ce3bd73SGreg Roach return [ 1472fad1159SGreg Roach e($row->multimedia_file_refn), 1488ce3bd73SGreg Roach $img, 1498ce3bd73SGreg Roach $this->mediaObjectInfo($media), 1508ce3bd73SGreg Roach ]; 1518ce3bd73SGreg Roach }; 1528ce3bd73SGreg Roach 1538ce3bd73SGreg Roach switch ($files) { 1548ce3bd73SGreg Roach case 'local': 1558ce3bd73SGreg Roach $query = DB::table('media_file') 1568ce3bd73SGreg Roach ->join('media', static function (JoinClause $join): void { 1578ce3bd73SGreg Roach $join 1588ce3bd73SGreg Roach ->on('media.m_file', '=', 'media_file.m_file') 1598ce3bd73SGreg Roach ->on('media.m_id', '=', 'media_file.m_id'); 1608ce3bd73SGreg Roach }) 1619ba2a180SGreg Roach ->leftJoin('gedcom_setting', static function (JoinClause $join): void { 1629ba2a180SGreg Roach $join 1639ba2a180SGreg Roach ->on('gedcom_setting.gedcom_id', '=', 'media.m_file') 1649ba2a180SGreg Roach ->where('setting_name', '=', 'MEDIA_DIRECTORY'); 1659ba2a180SGreg Roach }) 1668ce3bd73SGreg Roach ->where('multimedia_file_refn', 'NOT LIKE', 'http://%') 1678ce3bd73SGreg Roach ->where('multimedia_file_refn', 'NOT LIKE', 'https://%') 1688ce3bd73SGreg Roach ->select([ 1698ce3bd73SGreg Roach 'media.*', 1708ce3bd73SGreg Roach 'multimedia_file_refn', 1718ce3bd73SGreg Roach 'descriptive_title', 1729ba2a180SGreg Roach new Expression("COALESCE(setting_value, 'media/') AS media_folder"), 1738ce3bd73SGreg Roach ]); 1748ce3bd73SGreg Roach 1758ce3bd73SGreg Roach $query->where(new Expression('setting_value || multimedia_file_refn'), 'LIKE', $media_folder . '%'); 1768ce3bd73SGreg Roach 1778ce3bd73SGreg Roach if ($subfolders === 'exclude') { 1788ce3bd73SGreg Roach $query->where(new Expression('setting_value || multimedia_file_refn'), 'NOT LIKE', $media_folder . '%/%'); 1798ce3bd73SGreg Roach } 1808ce3bd73SGreg Roach 1818ce3bd73SGreg Roach return $this->datatables_service->handleQuery($request, $query, $search_columns, $sort_columns, $callback); 1828ce3bd73SGreg Roach 1838ce3bd73SGreg Roach case 'external': 1848ce3bd73SGreg Roach $query = DB::table('media_file') 1858ce3bd73SGreg Roach ->join('media', static function (JoinClause $join): void { 1868ce3bd73SGreg Roach $join 1878ce3bd73SGreg Roach ->on('media.m_file', '=', 'media_file.m_file') 1888ce3bd73SGreg Roach ->on('media.m_id', '=', 'media_file.m_id'); 1898ce3bd73SGreg Roach }) 1908ce3bd73SGreg Roach ->where(static function (Builder $query): void { 1918ce3bd73SGreg Roach $query 1928ce3bd73SGreg Roach ->where('multimedia_file_refn', 'LIKE', 'http://%') 1938ce3bd73SGreg Roach ->orWhere('multimedia_file_refn', 'LIKE', 'https://%'); 1948ce3bd73SGreg Roach }) 1958ce3bd73SGreg Roach ->select([ 1968ce3bd73SGreg Roach 'media.*', 1978ce3bd73SGreg Roach 'multimedia_file_refn', 1988ce3bd73SGreg Roach 'descriptive_title', 19996c9e653SGreg Roach new Expression("'' AS media_folder"), 2008ce3bd73SGreg Roach ]); 2018ce3bd73SGreg Roach 2028ce3bd73SGreg Roach return $this->datatables_service->handleQuery($request, $query, $search_columns, $sort_columns, $callback); 2038ce3bd73SGreg Roach 2048ce3bd73SGreg Roach case 'unused': 2058ce3bd73SGreg Roach // Which trees use which media folder? 2068ce3bd73SGreg Roach $media_trees = DB::table('gedcom') 2078ce3bd73SGreg Roach ->join('gedcom_setting', 'gedcom_setting.gedcom_id', '=', 'gedcom.gedcom_id') 2088ce3bd73SGreg Roach ->where('setting_name', '=', 'MEDIA_DIRECTORY') 2098ce3bd73SGreg Roach ->where('gedcom.gedcom_id', '>', 0) 2108ce3bd73SGreg Roach ->pluck('setting_value', 'gedcom_name'); 2118ce3bd73SGreg Roach 2128ce3bd73SGreg Roach $disk_files = $this->media_file_service->allFilesOnDisk($data_filesystem, $media_folder, $subfolders === 'include'); 2138ce3bd73SGreg Roach $db_files = $this->media_file_service->allFilesInDatabase($media_folder, $subfolders === 'include'); 2148ce3bd73SGreg Roach 2158ce3bd73SGreg Roach // All unused files 2168ce3bd73SGreg Roach $unused_files = $disk_files->diff($db_files) 2178ce3bd73SGreg Roach ->map(static function (string $file): array { 2188ce3bd73SGreg Roach return (array) $file; 2198ce3bd73SGreg Roach }); 2208ce3bd73SGreg Roach 2218ce3bd73SGreg Roach $search_columns = [0]; 2228ce3bd73SGreg Roach $sort_columns = [0 => 0]; 2238ce3bd73SGreg Roach 2248ce3bd73SGreg Roach $callback = function (array $row) use ($data_filesystem, $media_trees): array { 225f7cf8a15SGreg Roach try { 226f0448b68SGreg Roach $mime_type = $data_filesystem->mimeType($row[0]) ?: Mime::DEFAULT_TYPE; 227f0448b68SGreg Roach } catch (FileSystemException | UnableToRetrieveMetadata $ex) { 228f7cf8a15SGreg Roach $mime_type = Mime::DEFAULT_TYPE; 229f7cf8a15SGreg Roach } 230f7cf8a15SGreg Roach 2318ce3bd73SGreg Roach 2328ce3bd73SGreg Roach if (str_starts_with($mime_type, 'image/')) { 2338ce3bd73SGreg Roach $url = route(AdminMediaFileThumbnail::class, ['path' => $row[0]]); 2348ce3bd73SGreg Roach $img = '<img src="' . e($url) . '">'; 2358ce3bd73SGreg Roach } else { 2368ce3bd73SGreg Roach $img = view('icons/mime', ['type' => $mime_type]); 2378ce3bd73SGreg Roach } 2388ce3bd73SGreg Roach 2398ce3bd73SGreg Roach $url = route(AdminMediaFileDownload::class, ['path' => $row[0]]); 2408ce3bd73SGreg Roach $img = '<a href="' . e($url) . '">' . $img . '</a>'; 2418ce3bd73SGreg Roach 2428ce3bd73SGreg Roach // Form to create new media object in each tree 2438ce3bd73SGreg Roach $create_form = ''; 2448ce3bd73SGreg Roach foreach ($media_trees as $media_tree => $media_directory) { 2458ce3bd73SGreg Roach if (str_starts_with($row[0], $media_directory)) { 2468ce3bd73SGreg Roach $tmp = substr($row[0], strlen($media_directory)); 2478ce3bd73SGreg Roach $create_form .= 248315eb316SGreg Roach '<p><a href="#" data-bs-toggle="modal" data-bs-backdrop="static" data-bs-target="#modal-create-media-from-file" data-file="' . e($tmp) . '" data-url="' . e(route(CreateMediaObjectFromFile::class, ['tree' => $media_tree])) . '" onclick="document.getElementById(\'modal-create-media-from-file-form\').action=this.dataset.url; document.getElementById(\'file\').value=this.dataset.file;">' . I18N::translate('Create') . '</a> — ' . e($media_tree) . '<p>'; 2498ce3bd73SGreg Roach } 2508ce3bd73SGreg Roach } 2518ce3bd73SGreg Roach 252d4786c66SGreg Roach $delete_link = '<p><a data-wt-confirm="' . I18N::translate('Are you sure you want to delete “%s”?', e($row[0])) . '" data-wt-post-url="' . e(route(DeletePath::class, [ 2538ce3bd73SGreg Roach 'path' => $row[0], 2548ce3bd73SGreg Roach ])) . '" href="#">' . I18N::translate('Delete') . '</a></p>'; 2558ce3bd73SGreg Roach 2568ce3bd73SGreg Roach return [ 2578ce3bd73SGreg Roach $this->mediaFileInfo($data_filesystem, $row[0]) . $delete_link, 2588ce3bd73SGreg Roach $img, 2598ce3bd73SGreg Roach $create_form, 2608ce3bd73SGreg Roach ]; 2618ce3bd73SGreg Roach }; 2628ce3bd73SGreg Roach 2638ce3bd73SGreg Roach return $this->datatables_service->handleCollection($request, $unused_files, $search_columns, $sort_columns, $callback); 2648ce3bd73SGreg Roach 2658ce3bd73SGreg Roach default: 2668ce3bd73SGreg Roach throw new HttpNotFoundException(); 2678ce3bd73SGreg Roach } 2688ce3bd73SGreg Roach } 2698ce3bd73SGreg Roach 2708ce3bd73SGreg Roach /** 2718ce3bd73SGreg Roach * Generate some useful information and links about a media object. 2728ce3bd73SGreg Roach * 2738ce3bd73SGreg Roach * @param Media $media 2748ce3bd73SGreg Roach * 2758ce3bd73SGreg Roach * @return string HTML 2768ce3bd73SGreg Roach */ 2778ce3bd73SGreg Roach private function mediaObjectInfo(Media $media): string 2788ce3bd73SGreg Roach { 279*d72643a1SDavid Drury $element = Registry::elementFactory()->make('NOTE:CONC'); 280*d72643a1SDavid Drury $html = '<b><a href="' . e($media->url()) . '">' . $media->fullName() . '</a></b>' . $element->value($media->getNote(), $media->tree()); 2818ce3bd73SGreg Roach 2828ce3bd73SGreg Roach $linked = []; 2838ce3bd73SGreg Roach foreach ($media->linkedIndividuals('OBJE') as $link) { 2848ce3bd73SGreg Roach $linked[] = '<a href="' . e($link->url()) . '">' . $link->fullName() . '</a>'; 2858ce3bd73SGreg Roach } 2868ce3bd73SGreg Roach foreach ($media->linkedFamilies('OBJE') as $link) { 2878ce3bd73SGreg Roach $linked[] = '<a href="' . e($link->url()) . '">' . $link->fullName() . '</a>'; 2888ce3bd73SGreg Roach } 2898ce3bd73SGreg Roach foreach ($media->linkedSources('OBJE') as $link) { 2908ce3bd73SGreg Roach $linked[] = '<a href="' . e($link->url()) . '">' . $link->fullName() . '</a>'; 2918ce3bd73SGreg Roach } 2928ce3bd73SGreg Roach foreach ($media->linkedNotes('OBJE') as $link) { 2938ce3bd73SGreg Roach // Invalid GEDCOM - you cannot link a NOTE to an OBJE 2948ce3bd73SGreg Roach $linked[] = '<a href="' . e($link->url()) . '">' . $link->fullName() . '</a>'; 2958ce3bd73SGreg Roach } 2968ce3bd73SGreg Roach foreach ($media->linkedRepositories('OBJE') as $link) { 2978ce3bd73SGreg Roach // Invalid GEDCOM - you cannot link a REPO to an OBJE 2988ce3bd73SGreg Roach $linked[] = '<a href="' . e($link->url()) . '">' . $link->fullName() . '</a>'; 2998ce3bd73SGreg Roach } 3008ce3bd73SGreg Roach foreach ($media->linkedLocations('OBJE') as $link) { 3018ce3bd73SGreg Roach $linked[] = '<a href="' . e($link->url()) . '">' . $link->fullName() . '</a>'; 3028ce3bd73SGreg Roach } 3038ce3bd73SGreg Roach if ($linked !== []) { 3048ce3bd73SGreg Roach $html .= '<ul>'; 3058ce3bd73SGreg Roach foreach ($linked as $link) { 3068ce3bd73SGreg Roach $html .= '<li>' . $link . '</li>'; 3078ce3bd73SGreg Roach } 3088ce3bd73SGreg Roach $html .= '</ul>'; 3098ce3bd73SGreg Roach } else { 3108ce3bd73SGreg Roach $html .= '<div class="alert alert-danger">' . I18N::translate('There are no links to this media object.') . '</div>'; 3118ce3bd73SGreg Roach } 3128ce3bd73SGreg Roach 3138ce3bd73SGreg Roach return $html; 3148ce3bd73SGreg Roach } 3158ce3bd73SGreg Roach 3168ce3bd73SGreg Roach /** 3178ce3bd73SGreg Roach * Generate some useful information and links about a media file. 3188ce3bd73SGreg Roach * 319f7cf8a15SGreg Roach * @param FilesystemOperator $data_filesystem 3208ce3bd73SGreg Roach * @param string $file 3218ce3bd73SGreg Roach * 3228ce3bd73SGreg Roach * @return string 3238ce3bd73SGreg Roach */ 324f7cf8a15SGreg Roach private function mediaFileInfo(FilesystemOperator $data_filesystem, string $file): string 3258ce3bd73SGreg Roach { 3268ce3bd73SGreg Roach $html = '<dl>'; 3278ce3bd73SGreg Roach $html .= '<dt>' . I18N::translate('Filename') . '</dt>'; 3288ce3bd73SGreg Roach $html .= '<dd>' . e($file) . '</dd>'; 3298ce3bd73SGreg Roach 330f0448b68SGreg Roach try { 331f0448b68SGreg Roach $file_exists = $data_filesystem->fileExists($file); 332f0448b68SGreg Roach } catch (FilesystemException | UnableToCheckFileExistence $ex) { 333f0448b68SGreg Roach $file_exists = false; 334f0448b68SGreg Roach } 335f0448b68SGreg Roach 336f0448b68SGreg Roach if ($file_exists) { 337f0448b68SGreg Roach try { 338f7cf8a15SGreg Roach $size = $data_filesystem->fileSize($file); 339f0448b68SGreg Roach } catch (FilesystemException | UnableToRetrieveMetadata $ex) { 340f0448b68SGreg Roach $size = 0; 341f0448b68SGreg Roach } 3428ce3bd73SGreg Roach $size = intdiv($size + 1023, 1024); // Round up to next KB 3438ce3bd73SGreg Roach /* I18N: size of file in KB */ 3448ce3bd73SGreg Roach $size = I18N::translate('%s KB', I18N::number($size)); 3458ce3bd73SGreg Roach $html .= '<dt>' . I18N::translate('File size') . '</dt>'; 3468ce3bd73SGreg Roach $html .= '<dd>' . $size . '</dd>'; 3478ce3bd73SGreg Roach 3488ce3bd73SGreg Roach try { 3498ce3bd73SGreg Roach // This will work for local filesystems. For remote filesystems, we will 3508ce3bd73SGreg Roach // need to copy the file locally to work out the image size. 351f7cf8a15SGreg Roach $imgsize = getimagesizefromstring($data_filesystem->read($file)); 3528ce3bd73SGreg Roach $html .= '<dt>' . I18N::translate('Image dimensions') . '</dt>'; 3538ce3bd73SGreg Roach /* I18N: image dimensions, width × height */ 3548ce3bd73SGreg Roach $html .= '<dd>' . I18N::translate('%1$s × %2$s pixels', I18N::number($imgsize['0']), I18N::number($imgsize['1'])) . '</dd>'; 355f0448b68SGreg Roach } catch (FilesystemException | UnableToReadFile | Throwable $ex) { 3568ce3bd73SGreg Roach // Not an image, or not a valid image? 3578ce3bd73SGreg Roach } 3588ce3bd73SGreg Roach } 3598ce3bd73SGreg Roach 3608ce3bd73SGreg Roach $html .= '</dl>'; 3618ce3bd73SGreg Roach 3628ce3bd73SGreg Roach return $html; 3638ce3bd73SGreg Roach } 3648ce3bd73SGreg Roach} 365