2 namespace TYPO3\CMS\Filelist
;
5 * This file is part of the TYPO3 CMS project.
7 * It is free software; you can redistribute it and/or modify it under
8 * the terms of the GNU General Public License, either version 2
9 * of the License, or any later version.
11 * For the full copyright and license information, please read the
12 * LICENSE.txt file that was distributed with this source code.
14 * The TYPO3 project - inspiring people to share!
17 use TYPO3\CMS\Backend\Clipboard\Clipboard
;
18 use TYPO3\CMS\Backend\Configuration\TranslationConfigurationProvider
;
19 use TYPO3\CMS\Backend\RecordList\AbstractRecordList
;
20 use TYPO3\CMS\Backend\Utility\BackendUtility
;
21 use TYPO3\CMS\Backend\Utility\IconUtility
;
22 use TYPO3\CMS\Core\Database\DatabaseConnection
;
23 use TYPO3\CMS\Core\Imaging\Icon
;
24 use TYPO3\CMS\Core\Imaging\IconFactory
;
25 use TYPO3\CMS\Core\Messaging\FlashMessage
;
26 use TYPO3\CMS\Core\
Resource\File
;
27 use TYPO3\CMS\Core\
Resource\Folder
;
28 use TYPO3\CMS\Core\
Resource\InaccessibleFolder
;
29 use TYPO3\CMS\Core\
Resource\ProcessedFile
;
30 use TYPO3\CMS\Core\
Resource\ResourceFactory
;
31 use TYPO3\CMS\Core\
Resource\Utility\ListUtility
;
32 use TYPO3\CMS\Core\Type\Bitmask\JsConfirmation
;
33 use TYPO3\CMS\Core\Utility\GeneralUtility
;
34 use TYPO3\CMS\Core\
Resource\FolderInterface
;
37 * Class for rendering of File>Filelist
39 class FileList
extends AbstractRecordList
{
42 * Default Max items shown
49 * Thumbnails on records containing files (pictures)
53 public $thumbs = FALSE;
56 * Space icon used for alignment when no button is available
63 * Max length of strings
75 * If TRUE click menus are generated on files and folders
79 public $clickMenus = 1;
82 * The field to sort by
89 * Reverse sorting flag
98 public $firstElementNumber = 0;
103 public $clipBoard = 0;
108 public $bigControlPanel = 0;
118 public $HTMLcode = '';
123 public $totalbytes = 0;
128 public $dirs = array();
133 public $files = array();
143 protected $folderObject;
146 * Counting the elements no matter what
150 public $eCounter = 0;
155 public $totalItems = '';
160 public $CBnames = array();
163 * @var Clipboard $clipObj
168 * @var ResourceFactory
170 protected $resourceFactory;
173 * @param ResourceFactory $resourceFactory
175 public function injectResourceFactory(ResourceFactory
$resourceFactory) {
176 $this->resourceFactory
= $resourceFactory;
182 protected $iconFactory;
187 public function __construct() {
188 $this->iconFactory
= GeneralUtility
::makeInstance(IconFactory
::class);
192 * Initialization of class
194 * @param Folder $folderObject The folder to work on
195 * @param int $pointer Pointer
196 * @param bool $sort Sorting column
197 * @param bool $sortRev Sorting direction
198 * @param bool $clipBoard
199 * @param bool $bigControlPanel Show clipboard flag
202 public function start(Folder
$folderObject, $pointer, $sort, $sortRev, $clipBoard = FALSE, $bigControlPanel = FALSE) {
203 $this->script
= BackendUtility
::getModuleUrl('file_list');
204 $this->folderObject
= $folderObject;
206 $this->totalbytes
= 0;
208 $this->HTMLcode
= '';
209 $this->path
= $folderObject->getReadablePath();
211 $this->sortRev
= $sortRev;
212 $this->firstElementNumber
= $pointer;
213 $this->clipBoard
= $clipBoard;
214 $this->bigControlPanel
= $bigControlPanel;
215 // Setting the maximum length of the filenames to the user's settings or minimum 30 (= $this->fixedL)
216 $this->fixedL
= max($this->fixedL
, $this->getBackendUser()->uc
['titleLen']);
217 $this->getLanguageService()->includeLLFile('EXT:lang/locallang_common.xlf');
218 $this->resourceFactory
= ResourceFactory
::getInstance();
222 * Reading files and directories, counting elements and generating the list in ->HTMLcode
226 public function generateList() {
227 $this->HTMLcode
.= $this->getTable('fileext,tstamp,size,rw,_REF_');
231 * Return the buttons used by the filelist to include in the top header
233 * @param Folder $folderObject
236 public function getButtonsAndOtherMarkers(Folder
$folderObject) {
237 $otherMarkers = array(
242 'level_up' => $this->getLinkToParentFolder($folderObject),
248 // Makes the code for the folder icon in the top
250 $title = htmlspecialchars($folderObject->getReadablePath());
251 // Start compiling the HTML
252 // If this is some subFolder under the mount root....
253 if ($folderObject->getStorage()->isWithinFileMountBoundaries($folderObject)) {
254 // The icon with link
255 $otherMarkers['PAGE_ICON'] = IconUtility
::getSpriteIconForResource($folderObject, array('title' => $title));
256 // No HTML specialchars here - HTML like <strong> </strong> is allowed
257 $otherMarkers['TITLE'] .= GeneralUtility
::removeXSS(GeneralUtility
::fixed_lgd_cs($title, -($this->fixedL +
20)));
259 // This is the root folder
260 $otherMarkers['PAGE_ICON'] = IconUtility
::getSpriteIconForResource($folderObject, array('title' => $title, 'mount-root' => TRUE));
261 $otherMarkers['TITLE'] .= htmlspecialchars(GeneralUtility
::fixed_lgd_cs($title, -($this->fixedL +
20)));
263 if ($this->clickMenus
) {
264 $otherMarkers['PAGE_ICON'] = $GLOBALS['SOBE']->doc
->wrapClickMenuOnIcon($otherMarkers['PAGE_ICON'], $folderObject->getCombinedIdentifier());
266 // Add paste button if clipboard is initialized
267 if ($this->clipObj
instanceof Clipboard
&& $folderObject->checkActionPermission('write')) {
268 $elFromTable = $this->clipObj
->elFromTable('_FILE');
269 if (!empty($elFromTable)) {
270 $addPasteButton = TRUE;
271 $elToConfirm = array();
272 foreach ($elFromTable as $key => $element) {
273 $clipBoardElement = $this->resourceFactory
->retrieveFileOrFolderObject($element);
274 if ($clipBoardElement instanceof Folder
&& $clipBoardElement->getStorage()->isWithinFolder($clipBoardElement, $folderObject)) {
275 $addPasteButton = FALSE;
277 $elToConfirm[$key] = $clipBoardElement->getName();
279 if ($addPasteButton) {
280 $buttons['PASTE'] = '<a href="' . htmlspecialchars($this->clipObj
->pasteUrl('_FILE', $folderObject->getCombinedIdentifier())) . '" onclick="return ' . htmlspecialchars($this->clipObj
->confirmMsg('_FILE', $this->path
, 'into', $elToConfirm)) . '" title="' . $this->getLanguageService()->getLL('clip_paste', TRUE) . '">' . $this->iconFactory
->getIcon('actions-document-paste-after', Icon
::SIZE_SMALL
) . '</a>';
286 $buttons['refresh'] = '<a href="' . htmlspecialchars($this->listURL()) . '" title="' . $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:labels.reload', TRUE) . '">' . IconUtility
::getSpriteIcon('actions-system-refresh') . '</a>';
287 return array($buttons, $otherMarkers);
291 * Wrapping input string in a link with clipboard command.
293 * @param string $string String to be linked - must be htmlspecialchar'ed / prepared before.
294 * @param string $table table - NOT USED
295 * @param string $cmd "cmd" value
296 * @param string $warning Warning for JS confirm message
297 * @return string Linked string
299 public function linkClipboardHeaderIcon($string, $table, $cmd, $warning = '') {
300 $onClickEvent = 'document.dblistForm.cmd.value=\'' . $cmd . '\';document.dblistForm.submit();';
302 $onClickEvent = 'if (confirm(' . GeneralUtility
::quoteJSvalue($warning) . ')){' . $onClickEvent . '}';
304 return '<a href="#" class="btn btn-default" onclick="' . htmlspecialchars($onClickEvent) . 'return false;">' . $string . '</a>';
308 * Returns a table with directories and files listed.
310 * @param array $rowlist Array of files from path
311 * @return string HTML-table
313 public function getTable($rowlist) {
314 // prepare space icon
315 $this->spaceIcon
= '<span class="btn btn-default disabled">' . IconUtility
::getSpriteIcon('empty-empty') . '</span>';
317 // @todo use folder methods directly when they support filters
318 $storage = $this->folderObject
->getStorage();
319 $storage->resetFileAndFolderNameFiltersToDefault();
321 // Only render the contents of a browsable storage
322 if ($this->folderObject
->getStorage()->isBrowsable()) {
324 $foldersCount = $storage->countFoldersInFolder($this->folderObject
);
325 $filesCount = $storage->countFilesInFolder($this->folderObject
);
326 } catch (\TYPO3\CMS\Core\
Resource\Exception\InsufficientFolderAccessPermissionsException
$e) {
331 if ($foldersCount <= $this->firstElementNumber
) {
332 $foldersFrom = FALSE;
335 $foldersFrom = $this->firstElementNumber
;
336 if ($this->firstElementNumber +
$this->iLimit
> $foldersCount) {
337 $foldersNum = $foldersCount - $this->firstElementNumber
;
339 $foldersNum = $this->iLimit
;
342 if ($foldersCount >= $this->firstElementNumber +
$this->iLimit
) {
346 if ($this->firstElementNumber
<= $foldersCount) {
348 $filesNum = $this->iLimit
- $foldersNum;
350 $filesFrom = $this->firstElementNumber
- $foldersCount;
351 if ($filesFrom +
$this->iLimit
> $filesCount) {
352 $filesNum = $filesCount - $filesFrom;
354 $filesNum = $this->iLimit
;
358 $folders = $storage->getFoldersInFolder($this->folderObject
, $foldersFrom, $foldersNum, TRUE, FALSE, trim($this->sort
), (bool)$this->sortRev
);
359 $files = $this->folderObject
->getFiles($filesFrom, $filesNum, Folder
::FILTER_MODE_USE_OWN_AND_STORAGE_FILTERS
, FALSE, trim($this->sort
), (bool)$this->sortRev
);
360 $this->totalItems
= $foldersCount +
$filesCount;
361 // Adds the code of files/dirs
364 // Cleaning rowlist for duplicates and place the $titleCol as the first column always!
365 $rowlist = '_LOCALIZATION_,' . $rowlist;
366 $rowlist = GeneralUtility
::rmFromList($titleCol, $rowlist);
367 $rowlist = GeneralUtility
::uniqueList($rowlist);
368 $rowlist = $rowlist ?
$titleCol . ',' . $rowlist : $titleCol;
369 if ($this->clipBoard
) {
370 $rowlist = str_replace('_LOCALIZATION_,', '_LOCALIZATION_,_CLIPBOARD_,', $rowlist);
371 $this->addElement_tdCssClass
['_CLIPBOARD_'] = 'col-clipboard';
373 if ($this->bigControlPanel
) {
374 $rowlist = str_replace('_LOCALIZATION_,', '_LOCALIZATION_,_CONTROL_,', $rowlist);
375 $this->addElement_tdCssClass
['_CONTROL_'] = 'col-control';
377 $this->fieldArray
= explode(',', $rowlist);
379 // Add classes to table cells
380 $this->addElement_tdCssClass
[$titleCol] = 'col-title';
381 $this->addElement_tdCssClass
['_LOCALIZATION_'] = 'col-localizationa';
383 $folders = ListUtility
::resolveSpecialFolderNames($folders);
386 // Directories are added
387 $this->eCounter
= $this->firstElementNumber
;
388 list($flag, $code) = $this->fwd_rwd_nav();
391 $iOut .= $this->formatDirList($folders);
393 $iOut .= $this->formatFileList($files);
395 $this->eCounter
= $this->firstElementNumber +
$this->iLimit
<= $this->totalItems
396 ?
$this->firstElementNumber +
$this->iLimit
398 list($flag, $code) = $this->fwd_rwd_nav();
401 // Header line is drawn
403 foreach ($this->fieldArray
as $v) {
404 if ($v == '_CLIPBOARD_' && $this->clipBoard
) {
407 $elFromTable = $this->clipObj
->elFromTable($table);
408 if (!empty($elFromTable) && $this->folderObject
->checkActionPermission('write')) {
409 $addPasteButton = TRUE;
410 $elToConfirm = array();
411 foreach ($elFromTable as $key => $element) {
412 $clipBoardElement = $this->resourceFactory
->retrieveFileOrFolderObject($element);
413 if ($clipBoardElement instanceof Folder
&& $clipBoardElement->getStorage()->isWithinFolder($clipBoardElement, $this->folderObject
)) {
414 $addPasteButton = FALSE;
416 $elToConfirm[$key] = $clipBoardElement->getName();
418 if ($addPasteButton) {
419 $cells[] = '<a class="btn btn-default" href="' . htmlspecialchars($this->clipObj
->pasteUrl('_FILE', $this->folderObject
->getCombinedIdentifier())) . '" onclick="return ' . htmlspecialchars($this->clipObj
->confirmMsg('_FILE', $this->path
, 'into', $elToConfirm)) . '" title="' . $this->getLanguageService()->getLL('clip_paste', 1) . '">' . $this->iconFactory
->getIcon('actions-document-paste-after', Icon
::SIZE_SMALL
) . '</a>';
422 if ($this->clipObj
->current
!== 'normal' && $iOut) {
423 $cells[] = $this->linkClipboardHeaderIcon(IconUtility
::getSpriteIcon('actions-edit-copy', array('title' => $this->getLanguageService()->getLL('clip_selectMarked', TRUE))), $table, 'setCB');
424 $cells[] = $this->linkClipboardHeaderIcon(IconUtility
::getSpriteIcon('actions-edit-delete', array('title' => $this->getLanguageService()->getLL('clip_deleteMarked'))), $table, 'delete', $this->getLanguageService()->getLL('clip_deleteMarkedWarning'));
425 $onClick = 'checkOffCB(\'' . implode(',', $this->CBnames
) . '\', this); return false;';
426 $cells[] = '<a class="btn btn-default" rel="" href="#" onclick="' . htmlspecialchars($onClick) . '" title="' . $this->getLanguageService()->getLL('clip_markRecords', TRUE) . '">' . IconUtility
::getSpriteIcon('actions-document-select') . '</a>';
428 $theData[$v] = implode('', $cells);
431 $theT = $this->linkWrapSort($this->getLanguageService()->getLL('c_' . $v, TRUE), $this->folderObject
->getCombinedIdentifier(), $v);
432 $theData[$v] = $theT;
436 $out .= '<thead>' . $this->addelement(1, '', $theData, '', '', '', 'th') . '</thead>';
437 $out .= '<tbody>' . $iOut . '</tbody>';
438 // half line is drawn
444 <div class="table-fit">
445 <table class="table table-striped table-hover" id="typo3-filelist">
451 /** @var $flashMessage FlashMessage */
452 $flashMessage = GeneralUtility
::makeInstance(FlashMessage
::class, $this->getLanguageService()->getLL('storageNotBrowsableMessage'), $this->getLanguageService()->getLL('storageNotBrowsableTitle'), FlashMessage
::INFO
);
453 $out = $flashMessage->render();
459 * If there is a parent folder and user has access to it, return an icon
460 * which is linked to the filelist of the parent folder.
462 * @param Folder $currentFolder
465 protected function getLinkToParentFolder(Folder
$currentFolder) {
468 $currentStorage = $currentFolder->getStorage();
469 $parentFolder = $currentFolder->getParentFolder();
470 if ($parentFolder->getIdentifier() !== $currentFolder->getIdentifier() && $currentStorage->isWithinFileMountBoundaries($parentFolder)) {
471 $levelUp = $this->linkWrapDir(
472 IconUtility
::getSpriteIcon(
473 'actions-view-go-up',
474 array('title' => $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.php:labels.upOneLevel', TRUE))
479 } catch (\Exception
$e) {}
484 * Gets the number of files and total size of a folder
488 public function getFolderInfo() {
489 if ($this->counter
== 1) {
490 $fileLabel = $this->getLanguageService()->getLL('file', TRUE);
492 $fileLabel = $this->getLanguageService()->getLL('files', TRUE);
494 return $this->counter
. ' ' . $fileLabel . ', ' . GeneralUtility
::formatSize($this->totalbytes
, $this->getLanguageService()->getLL('byteSizeUnits', TRUE));
498 * This returns tablerows for the directories in the array $items['sorting'].
500 * @param Folder[] $folders Folders of \TYPO3\CMS\Core\Resource\Folder
501 * @return string HTML table rows.
503 public function formatDirList(array $folders) {
505 foreach ($folders as $folderName => $folderObject) {
506 $role = $folderObject->getRole();
507 if ($role === FolderInterface
::ROLE_PROCESSING
) {
508 // don't show processing-folder
511 if ($role !== FolderInterface
::ROLE_DEFAULT
) {
512 $displayName = '<strong>' . htmlspecialchars($folderName) . '</strong>';
514 $displayName = htmlspecialchars($folderName);
517 $isLocked = $folderObject instanceof InaccessibleFolder
;
518 $isWritable = $folderObject->checkActionPermission('write');
523 // The icon with link
524 $theIcon = IconUtility
::getSpriteIconForResource($folderObject, array('title' => $folderName));
525 if (!$isLocked && $this->clickMenus
) {
526 $theIcon = $GLOBALS['SOBE']->doc
->wrapClickMenuOnIcon($theIcon, $folderObject->getCombinedIdentifier());
529 // Preparing and getting the data-array
532 foreach ($this->fieldArray
as $field) {
533 $theData[$field] = '';
535 $theData['file'] = $displayName;
537 foreach ($this->fieldArray
as $field) {
541 $numFiles = $folderObject->getFileCount();
542 } catch (\TYPO3\CMS\Core\
Resource\Exception\InsufficientFolderAccessPermissionsException
$e) {
545 $theData[$field] = $numFiles . ' ' . $this->getLanguageService()->getLL(($numFiles === 1 ?
'file' : 'files'), TRUE);
548 $theData[$field] = '<strong class="text-danger">' . $this->getLanguageService()->getLL('read', TRUE) . '</strong>' . (!$isWritable ?
'' : '<strong class="text-danger">' . $this->getLanguageService()->getLL('write', TRUE) . '</strong>');
551 $theData[$field] = $this->getLanguageService()->getLL('folder', TRUE);
554 // @todo: FAL: how to get the mtime info -- $theData[$field] = \TYPO3\CMS\Backend\Utility\BackendUtility::date($theFile['tstamp']);
555 $theData[$field] = '-';
558 $theData[$field] = $this->linkWrapDir($displayName, $folderObject);
561 $theData[$field] = $this->makeEdit($folderObject);
564 $theData[$field] = $this->makeClip($folderObject);
567 $theData[$field] = $this->makeRef($folderObject);
570 $theData[$field] = GeneralUtility
::fixed_lgd_cs($theData[$field], $this->fixedL
);
574 $out .= $this->addelement(1, $theIcon, $theData);
580 * Wraps the directory-titles
582 * @param string $title String to be wrapped in links
583 * @param Folder $folderObject Folder to work on
584 * @return string HTML
586 public function linkWrapDir($title, Folder
$folderObject) {
587 $href = $this->script
. '&id=' . rawurlencode($folderObject->getCombinedIdentifier());
588 $onclick = ' onclick="' . htmlspecialchars(('top.document.getElementsByName("navigation")[0].contentWindow.Tree.highlightActiveItem("file","folder' . GeneralUtility
::md5int($folderObject->getCombinedIdentifier()) . '_"+top.fsMod.currentBank)')) . '"';
589 // Sometimes $code contains plain HTML tags. In such a case the string should not be modified!
590 if ((string)$title === strip_tags($title)) {
591 return '<a href="' . htmlspecialchars($href) . '"' . $onclick . ' title="' . htmlspecialchars($title) . '">' . GeneralUtility
::fixed_lgd_cs($title, $this->fixedL
) . '</a>';
593 return '<a href="' . htmlspecialchars($href) . '"' . $onclick . '>' . $title . '</a>';
598 * Wraps filenames in links which opens the metadata editor.
600 * @param string $code String to be wrapped in links
601 * @param File $fileObject File to be linked
602 * @return string HTML
604 public function linkWrapFile($code, File
$fileObject) {
606 if ($fileObject instanceof File
&& $fileObject->isIndexed() && $fileObject->checkActionPermission('write') && $this->getBackendUser()->check('tables_modify', 'sys_file_metadata')) {
607 $metaData = $fileObject->_getMetaData();
609 'sys_file_metadata' => array($metaData['uid'] => 'edit')
611 $editOnClick = BackendUtility
::editOnClick(GeneralUtility
::implodeArrayForUrl('edit', $data), '', $this->listUrl());
612 $title = htmlspecialchars($this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.editMetadata'));
613 $code = '<a href="#" title="' . $title . '" onclick="' . htmlspecialchars($editOnClick) . '">' . GeneralUtility
::fixed_lgd_cs($code, $this->fixedL
) . '</a>';
615 } catch (\Exception
$e) {
616 // intentional fall-through
622 * Returns list URL; This is the URL of the current script with id and imagemode parameters, that's all.
623 * The URL however is not relative, otherwise GeneralUtility::sanitizeLocalUrl() would say that
624 * the URL would be invalid
628 public function listURL($altId = '') {
629 return GeneralUtility
::linkThisScript(array(
630 'target' => rawurlencode($this->folderObject
->getCombinedIdentifier()),
631 'imagemode' => $this->thumbs
636 * This returns tablerows for the files in the array $items['sorting'].
638 * @param File[] $files File items
639 * @return string HTML table rows.
641 public function formatFileList(array $files) {
643 // first two keys are "0" (default) and "-1" (multiple), after that comes the "other languages"
644 $allSystemLanguages = GeneralUtility
::makeInstance(TranslationConfigurationProvider
::class)->getSystemLanguages();
645 $systemLanguages = array_filter($allSystemLanguages, function($languageRecord) {
646 if ($languageRecord['uid'] === -1 ||
$languageRecord['uid'] === 0 ||
!$this->getBackendUser()->checkLanguageAccess($languageRecord['uid'])) {
653 foreach ($files as $fileObject) {
656 $this->totalbytes +
= $fileObject->getSize();
657 $ext = $fileObject->getExtension();
658 $fileName = trim($fileObject->getName());
659 // The icon with link
660 $theIcon = IconUtility
::getSpriteIconForResource($fileObject, array('title' => $fileName . ' [' . (int)$fileObject->getUid() . ']'));
661 if ($this->clickMenus
) {
662 $theIcon = $GLOBALS['SOBE']->doc
->wrapClickMenuOnIcon($theIcon, $fileObject->getCombinedIdentifier());
664 // Preparing and getting the data-array
666 foreach ($this->fieldArray
as $field) {
669 $theData[$field] = GeneralUtility
::formatSize($fileObject->getSize(), $this->getLanguageService()->getLL('byteSizeUnits', TRUE));
672 $theData[$field] = '' . (!$fileObject->checkActionPermission('read') ?
' ' : '<strong class="text-danger">' . $this->getLanguageService()->getLL('read', TRUE) . '</strong>') . (!$fileObject->checkActionPermission('write') ?
'' : '<strong class="text-danger">' . $this->getLanguageService()->getLL('write', TRUE) . '</strong>');
675 $theData[$field] = strtoupper($ext);
678 $theData[$field] = BackendUtility
::date($fileObject->getModificationTime());
681 $theData[$field] = $this->makeEdit($fileObject);
684 $theData[$field] = $this->makeClip($fileObject);
686 case '_LOCALIZATION_':
687 if (!empty($systemLanguages) && $fileObject->isIndexed() && $fileObject->checkActionPermission('write') && $this->getBackendUser()->check('tables_modify', 'sys_file_metadata')) {
688 $metaDataRecord = $fileObject->_getMetaData();
689 $translations = $this->getTranslationsForMetaData($metaDataRecord);
692 foreach ($systemLanguages as $language) {
693 $languageId = $language['uid'];
694 $flagIcon = $language['flagIcon'];
695 if (array_key_exists($languageId, $translations)) {
696 $flagButtonIcon = IconUtility
::getSpriteIcon(
697 'actions-document-open',
698 array('title' => sprintf($GLOBALS['LANG']->getLL('editMetadataForLanguage'), $language['title'])),
699 array($flagIcon . '-overlay' => array()));
701 'sys_file_metadata' => array($translations[$languageId]['uid'] => 'edit')
703 $editOnClick = BackendUtility
::editOnClick(GeneralUtility
::implodeArrayForUrl('edit', $data), '', $this->listUrl());
704 $languageCode .= '<a href="#" class="btn btn-default" onclick="' . htmlspecialchars($editOnClick) . '">' . $flagButtonIcon . '</a>';
707 'justLocalized' => 'sys_file_metadata:' . $metaDataRecord['uid'] . ':' . $languageId,
708 'returnUrl' => $this->listURL()
710 $returnUrl = BackendUtility
::getModuleUrl('record_edit', $parameters) . BackendUtility
::getUrlToken('editRecord');
711 $href = $GLOBALS['SOBE']->doc
->issueCommand(
712 '&cmd[sys_file_metadata][' . $metaDataRecord['uid'] . '][localize]=' . $languageId,
715 $flagButtonIcon = IconUtility
::getSpriteIcon(
717 array('title' => sprintf($GLOBALS['LANG']->getLL('createMetadataForLanguage'), $language['title'])),
718 array($flagIcon . '-overlay' => array())
720 $languageCode .= '<a href="' . htmlspecialchars($href) . '" class="btn btn-default">' . $flagButtonIcon . '</a> ';
724 // Hide flag button bar when not translated yet
725 $theData[$field] = ' <div class="localisationData btn-group" data-fileid="' . $fileObject->getUid() . '"' .
726 (empty($translations) ?
' style="display: none;"' : '') . '>' . $languageCode . '</div>';
727 $theData[$field] .= '<a class="btn btn-default filelist-translationToggler" data-fileid="' . $fileObject->getUid() . '">' .
728 IconUtility
::getSpriteIcon(
729 'mimetypes-x-content-page-language-overlay',
731 'title' => $GLOBALS['LANG']->getLL('translateMetadata')
737 $theData[$field] = $this->makeRef($fileObject);
740 // Edit metadata of file
741 $theData[$field] = $this->linkWrapFile(htmlspecialchars($fileName), $fileObject);
743 if ($fileObject->isMissing()) {
744 $flashMessage = \TYPO3\CMS\Core\
Resource\Utility\BackendUtility
::getFlashMessageForMissingFile($fileObject);
745 $theData[$field] .= $flashMessage->render();
747 } elseif ($this->thumbs
&& $this->isImage($ext)) {
748 $processedFile = $fileObject->process(ProcessedFile
::CONTEXT_IMAGEPREVIEW
, array());
749 if ($processedFile) {
750 $thumbUrl = $processedFile->getPublicUrl(TRUE);
751 $theData[$field] .= '<br /><img src="' . $thumbUrl . '" ' .
752 'width="' . $processedFile->getProperty('width') . '" ' .
753 'height="' . $processedFile->getProperty('height') . '" ' .
754 'title="' . htmlspecialchars($fileName) . '" alt="" />';
759 $theData[$field] = '';
760 if ($fileObject->hasProperty($field)) {
761 $theData[$field] = htmlspecialchars(GeneralUtility
::fixed_lgd_cs($fileObject->getProperty($field), $this->fixedL
));
765 $out .= $this->addelement(1, $theIcon, $theData);
771 * Fetch the translations for a sys_file_metadata record
773 * @param $metaDataRecord
774 * @return array keys are the sys_language uids, values are the $rows
776 protected function getTranslationsForMetaData($metaDataRecord) {
777 $where = $GLOBALS['TCA']['sys_file_metadata']['ctrl']['transOrigPointerField'] . '=' . (int)$metaDataRecord['uid'] .
778 ' AND ' . $GLOBALS['TCA']['sys_file_metadata']['ctrl']['languageField'] . '>0';
779 $translationRecords = $this->getDatabaseConnection()->exec_SELECTgetRows('*', 'sys_file_metadata', $where);
780 $translations = array();
781 foreach ($translationRecords as $record) {
782 $translations[$record[$GLOBALS['TCA']['sys_file_metadata']['ctrl']['languageField']]] = $record;
784 return $translations;
788 * Returns TRUE if $ext is an image-extension according to $GLOBALS['TYPO3_CONF_VARS']['GFX']['imagefile_ext']
790 * @param string $ext File extension
793 public function isImage($ext) {
794 return GeneralUtility
::inList($GLOBALS['TYPO3_CONF_VARS']['GFX']['imagefile_ext'], strtolower($ext));
798 * Wraps the directory-titles ($code) in a link to filelist/Modules/Filelist/index.php (id=$path) and sorting commands...
800 * @param string $code String to be wrapped
801 * @param string $folderIdentifier ID (path)
802 * @param string $col Sorting column
803 * @return string HTML
805 public function linkWrapSort($code, $folderIdentifier, $col) {
806 if ($this->sort
=== $col) {
807 // Check reverse sorting
808 $params = '&SET[sort]=' . $col . '&SET[reverse]=' . ($this->sortRev ?
'0' : '1');
809 $sortArrow = IconUtility
::getSpriteIcon('status-status-sorting-light-' . ($this->sortRev ?
'desc' : 'asc'));
811 $params = '&SET[sort]=' . $col . '&SET[reverse]=0';
814 $href = GeneralUtility
::resolveBackPath($this->script
) . '&id=' . rawurlencode($folderIdentifier) . $params;
815 return '<a href="' . htmlspecialchars($href) . '">' . $code . ' ' . $sortArrow . '</a>';
819 * Creates the clipboard control pad
821 * @param File|Folder $fileOrFolderObject Array with information about the file/directory for which to make the clipboard panel for the listing.
822 * @return string HTML-table
824 public function makeClip($fileOrFolderObject) {
825 if (!$fileOrFolderObject->checkActionPermission('read')) {
829 $fullIdentifier = $fileOrFolderObject->getCombinedIdentifier();
830 $fullName = $fileOrFolderObject->getName();
831 $md5 = GeneralUtility
::shortmd5($fullIdentifier);
832 // For normal clipboard, add copy/cut buttons:
833 if ($this->clipObj
->current
=== 'normal') {
834 $isSel = $this->clipObj
->isSelected('_FILE', $md5);
835 $copyTitle = $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.copy', TRUE);
836 $cutTitle = $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.cut', TRUE);
837 $copyIcon = $this->iconFactory
->getIcon('actions-edit-copy', Icon
::SIZE_SMALL
);
838 $cutIcon = $this->iconFactory
->getIcon('actions-edit-cut', Icon
::SIZE_SMALL
);
840 if ($isSel === 'copy') {
841 $copyIcon = $this->iconFactory
->getIcon('actions-edit-copy-release', Icon
::SIZE_SMALL
);
842 } elseif ($isSel === 'cut') {
843 $cutIcon = $this->iconFactory
->getIcon('actions-edit-cut-release', Icon
::SIZE_SMALL
);
846 $cells[] = '<a class="btn btn-default"" href="' . htmlspecialchars($this->clipObj
->selUrlFile($fullIdentifier, 1, ($isSel === 'copy'))) . '" title="' . $copyTitle . '">' . $copyIcon . '</a>';
847 // we can only cut if file can be moved
848 if ($fileOrFolderObject->checkActionPermission('move')) {
849 $cells[] = '<a class="btn btn-default" href="' . htmlspecialchars($this->clipObj
->selUrlFile($fullIdentifier, 0, ($isSel === 'cut'))) . '" title="' . $cutTitle . '">' . $cutIcon . '</a>';
851 $cells[] = $this->spaceIcon
;
854 // For numeric pads, add select checkboxes:
855 $n = '_FILE|' . $md5;
856 $this->CBnames
[] = $n;
857 $checked = $this->clipObj
->isSelected('_FILE', $md5) ?
' checked="checked"' : '';
858 $cells[] = '<label class="btn btn-default btn-checkbox"><input type="hidden" name="CBH[' . $n . ']" value="0" /><input type="checkbox" name="CBC[' . $n . ']" value="' . htmlspecialchars($fullIdentifier) . '" ' . $checked . ' /><span class="t3-icon fa"></span></label>';
860 // Display PASTE button, if directory:
861 $elFromTable = $this->clipObj
->elFromTable('_FILE');
862 if ($fileOrFolderObject instanceof Folder
&& !empty($elFromTable) && $fileOrFolderObject->checkActionPermission('write')) {
863 $addPasteButton = TRUE;
864 $elToConfirm = array();
865 foreach ($elFromTable as $key => $element) {
866 $clipBoardElement = $this->resourceFactory
->retrieveFileOrFolderObject($element);
867 if ($clipBoardElement instanceof Folder
&& $clipBoardElement->getStorage()->isWithinFolder($clipBoardElement, $fileOrFolderObject)) {
868 $addPasteButton = FALSE;
870 $elToConfirm[$key] = $clipBoardElement->getName();
872 if ($addPasteButton) {
873 $cells[] = '<a class="btn btn-default" href="' . htmlspecialchars($this->clipObj
->pasteUrl('_FILE', $fullIdentifier)) . '" onclick="return ' . htmlspecialchars($this->clipObj
->confirmMsg('_FILE', $fullName, 'into', $elToConfirm)) . '" title="' . $this->getLanguageService()->getLL('clip_pasteInto', TRUE) . '">' . IconUtility
::getSpriteIcon('actions-document-paste-into') . '</a>';
876 // Compile items into a DIV-element:
877 return ' <div class="btn-group" role="group">' . implode('', $cells) . '</div>';
881 * Creates the edit control section
883 * @param File|Folder $fileOrFolderObject Array with information about the file/directory for which to make the edit control section for the listing.
884 * @return string HTML-table
886 public function makeEdit($fileOrFolderObject) {
888 $fullIdentifier = $fileOrFolderObject->getCombinedIdentifier();
890 // Edit file content (if editable)
891 if ($fileOrFolderObject instanceof File
&& $fileOrFolderObject->checkActionPermission('write') && GeneralUtility
::inList($GLOBALS['TYPO3_CONF_VARS']['SYS']['textfile_ext'], $fileOrFolderObject->getExtension())) {
892 $url = BackendUtility
::getModuleUrl('file_edit', array('target' => $fullIdentifier));
893 $editOnClick = 'top.content.list_frame.location.href=' . GeneralUtility
::quoteJSvalue($url) . '+\'&returnUrl=\'+top.rawurlencode(top.content.list_frame.document.location.pathname+top.content.list_frame.document.location.search);return false;';
894 $cells['edit'] = '<a href="#" class="btn btn-default" onclick="' . htmlspecialchars($editOnClick) . '" title="' . $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.editcontent') . '">' . IconUtility
::getSpriteIcon('actions-page-open') . '</a>';
896 $cells['edit'] = $this->spaceIcon
;
898 if ($fileOrFolderObject instanceof File
) {
899 $fileUrl = $fileOrFolderObject->getPublicUrl(TRUE);
901 $aOnClick = 'return top.openUrlInWindow(' . GeneralUtility
::quoteJSvalue($fileUrl) . ', \'WebFile\');';
902 $cells['view'] = '<a href="#" class="btn btn-default" onclick="' . htmlspecialchars($aOnClick) . '" title="' . $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.view') . '">' . $this->iconFactory
->getIcon('actions-document-view', Icon
::SIZE_SMALL
) . '</a>';
904 $cells['view'] = $this->spaceIcon
;
907 $cells['view'] = $this->spaceIcon
;
911 if ($fileOrFolderObject instanceof File
&& $fileOrFolderObject->checkActionPermission('replace')) {
912 $url = BackendUtility
::getModuleUrl('file_replace', array('target' => $fullIdentifier, 'uid' => $fileOrFolderObject->getUid()));
913 $replaceOnClick = 'top.content.list_frame.location.href = ' . GeneralUtility
::quoteJSvalue($url) . '+\'&returnUrl=\'+top.rawurlencode(top.content.list_frame.document.location.pathname+top.content.list_frame.document.location.search);return false;';
914 $cells['replace'] = '<a href="#" class="btn btn-default" onclick="' . $replaceOnClick . '" title="' . $GLOBALS['LANG']->sL('LLL:EXT:lang/locallang_core.xlf:cm.replace') . '">' . IconUtility
::getSpriteIcon('actions-edit-replace') . '</a>';
918 if ($fileOrFolderObject->checkActionPermission('rename')) {
919 $url = BackendUtility
::getModuleUrl('file_rename', array('target' => $fullIdentifier));
920 $renameOnClick = 'top.content.list_frame.location.href = ' . GeneralUtility
::quoteJSvalue($url) . '+\'&returnUrl=\'+top.rawurlencode(top.content.list_frame.document.location.pathname+top.content.list_frame.document.location.search);return false;';
921 $cells['rename'] = '<a href="#" class="btn btn-default" onclick="' . htmlspecialchars($renameOnClick) . '" title="' . $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.rename') . '">' . IconUtility
::getSpriteIcon('actions-edit-rename') . '</a>';
923 $cells['rename'] = $this->spaceIcon
;
925 if ($fileOrFolderObject->checkActionPermission('read')) {
927 if ($fileOrFolderObject instanceof Folder
) {
928 $infoOnClick = 'top.launchView( \'_FOLDER\', ' . GeneralUtility
::quoteJSvalue($fullIdentifier) . ');return false;';
929 } elseif ($fileOrFolderObject instanceof File
) {
930 $infoOnClick = 'top.launchView( \'_FILE\', ' . GeneralUtility
::quoteJSvalue($fullIdentifier) . ');return false;';
932 $cells['info'] = '<a href="#" class="btn btn-default" onclick="' . htmlspecialchars($infoOnClick) . '" title="' . $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.info') . '">' . IconUtility
::getSpriteIcon('status-dialog-information') . '</a>';
934 $cells['info'] = $this->spaceIcon
;
938 if ($fileOrFolderObject->checkActionPermission('delete')) {
939 $identifier = $fileOrFolderObject->getIdentifier();
940 if ($fileOrFolderObject instanceof Folder
) {
941 $referenceCountText = BackendUtility
::referenceCount('_FILE', $identifier, ' (There are %s reference(s) to this folder!)');
943 $referenceCountText = BackendUtility
::referenceCount('sys_file', $fileOrFolderObject->getUid(), ' (There are %s reference(s) to this file!)');
946 if ($this->getBackendUser()->jsConfirmation(JsConfirmation
::DELETE
)) {
947 $confirmationCheck = 'confirm(' . GeneralUtility
::quoteJSvalue(sprintf($this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:mess.delete'), $fileOrFolderObject->getName()) . $referenceCountText) . ')';
949 $confirmationCheck = '1 == 1';
952 $removeOnClick = 'if (' . $confirmationCheck . ') { top.content.list_frame.location.href=' . GeneralUtility
::quoteJSvalue(BackendUtility
::getModuleUrl('tce_file') .'&file[delete][0][data]=' . rawurlencode($fileOrFolderObject->getCombinedIdentifier()) . '&vC=' . $this->getBackendUser()->veriCode() . BackendUtility
::getUrlToken('tceAction') . '&redirect=') . '+top.rawurlencode(top.content.list_frame.document.location.pathname+top.content.list_frame.document.location.search);};';
954 $cells['delete'] = '<a href="#" class="btn btn-default" onclick="' . htmlspecialchars($removeOnClick) . '" title="' . $this->getLanguageService()->sL('LLL:EXT:lang/locallang_core.xlf:cm.delete') . '">' . IconUtility
::getSpriteIcon('actions-edit-delete') . '</a>';
956 $cells['delete'] = $this->spaceIcon
;
959 // Hook for manipulating edit icons.
960 if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['fileList']['editIconsHook'])) {
961 $cells['__fileOrFolderObject'] = $fileOrFolderObject;
962 foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['fileList']['editIconsHook'] as $classData) {
963 $hookObject = GeneralUtility
::getUserObj($classData);
964 if (!$hookObject instanceof FileListEditIconHookInterface
) {
965 throw new \
UnexpectedValueException(
966 '$hookObject must implement interface \\TYPO3\\CMS\\Filelist\\FileListEditIconHookInterface',
970 $hookObject->manipulateEditIcons($cells, $this);
972 unset($cells['__fileOrFolderObject']);
974 // Compile items into a DIV-element:
975 return '<div class="btn-group">' . implode('', $cells) . '</div>';
979 * Make reference count
981 * @param File|Folder $fileOrFolderObject Array with information about the file/directory for which to make the clipboard panel for the listing.
982 * @return string HTML
984 public function makeRef($fileOrFolderObject) {
985 if ($fileOrFolderObject instanceof FolderInterface
) {
988 // Look up the file in the sys_refindex.
989 // Exclude sys_file_metadata records as these are no use references
990 $databaseConnection = $this->getDatabaseConnection();
991 $table = 'sys_refindex';
992 $referenceCount = $databaseConnection->exec_SELECTcountRows(
995 'ref_table=' . $databaseConnection->fullQuoteStr('sys_file', $table)
996 . ' AND ref_uid=' . (int)$fileOrFolderObject->getUid()
998 . ' AND tablename != ' . $databaseConnection->fullQuoteStr('sys_file_metadata', $table)
1000 return $this->generateReferenceToolTip($referenceCount, '\'_FILE\', ' . GeneralUtility
::quoteJSvalue($fileOrFolderObject->getCombinedIdentifier()));
1004 * Returns the database connection
1005 * @return DatabaseConnection
1007 protected function getDatabaseConnection() {
1008 return $GLOBALS['TYPO3_DB'];
1012 * Returns an instance of LanguageService
1014 * @return \TYPO3\CMS\Lang\LanguageService
1016 protected function getLanguageService() {
1017 return $GLOBALS['LANG'];
1021 * Returns the current BE user.
1023 * @return \TYPO3\CMS\Core\Authentication\BackendUserAuthentication
1025 protected function getBackendUser() {
1026 return $GLOBALS['BE_USER'];