mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 06:05:56 +00:00
c965133da0
Trying to navigate directly to the edit view of a file by URL would fail if a folder other than the file’s parent was held in session data. Removed reliance on current ID in session and added folder detection for edit view.
745 lines
24 KiB
PHP
745 lines
24 KiB
PHP
<?php
|
|
/**
|
|
* AssetAdmin is the 'file store' section of the CMS.
|
|
* It provides an interface for manipulating the File and Folder objects in the system.
|
|
*
|
|
* @package cms
|
|
* @subpackage assets
|
|
*/
|
|
class AssetAdmin extends LeftAndMain implements PermissionProvider{
|
|
|
|
private static $url_segment = 'assets';
|
|
|
|
private static $url_rule = '/$Action/$ID';
|
|
|
|
private static $menu_title = 'Files';
|
|
|
|
private static $tree_class = 'Folder';
|
|
|
|
/**
|
|
* Amount of results showing on a single page.
|
|
*
|
|
* @config
|
|
* @var int
|
|
*/
|
|
private static $page_length = 15;
|
|
|
|
/**
|
|
* @config
|
|
* @see Upload->allowedMaxFileSize
|
|
* @var int
|
|
*/
|
|
private static $allowed_max_file_size;
|
|
|
|
private static $allowed_actions = array(
|
|
'addfolder',
|
|
'delete',
|
|
'AddForm',
|
|
'DeleteItemsForm',
|
|
'SearchForm',
|
|
'getsubtree',
|
|
'movemarked',
|
|
'removefile',
|
|
'savefile',
|
|
'deleteUnusedThumbnails' => 'ADMIN',
|
|
'doSync',
|
|
'filter',
|
|
);
|
|
|
|
/**
|
|
* Return fake-ID 0 (root) if no ID is found (needed to upload files into the root-folder)
|
|
*/
|
|
public function currentPageID() {
|
|
$id = 0;
|
|
$request = $this->getRequest();
|
|
if(is_numeric($request->requestVar('ID'))) {
|
|
$id = $request->requestVar('ID');
|
|
} elseif (is_numeric($request->param('ID'))) {
|
|
$id = $request->param('ID');
|
|
}
|
|
|
|
// Detect current folder in gridfield item edit view
|
|
if ($id && $id > 0) {
|
|
if (!Folder::get()->filter('ID', $id)->exists()) {
|
|
$file = File::get()->byID($id);
|
|
$id = ($file) ? $file->ParentID : 0;
|
|
}
|
|
}
|
|
|
|
$id = (int)$id;
|
|
$this->setCurrentPageID($id);
|
|
return $id;
|
|
}
|
|
|
|
/**
|
|
* Set up the controller, in particular, re-sync the File database with the assets folder./
|
|
*/
|
|
public function init() {
|
|
parent::init();
|
|
|
|
// Create base folder if it doesnt exist already
|
|
if(!file_exists(ASSETS_PATH)) Filesystem::makeFolder(ASSETS_PATH);
|
|
|
|
Requirements::javascript(CMS_DIR . "/javascript/AssetAdmin.js");
|
|
Requirements::javascript(CMS_DIR . '/javascript/CMSMain.GridField.js');
|
|
Requirements::add_i18n_javascript(CMS_DIR . '/javascript/lang', false, true);
|
|
Requirements::css(CMS_DIR . "/css/screen.css");
|
|
$frameworkDir = FRAMEWORK_DIR;
|
|
Requirements::customScript(<<<JS
|
|
_TREE_ICONS = {};
|
|
_TREE_ICONS['Folder'] = {
|
|
fileIcon: '$frameworkDir/javascript/tree/images/page-closedfolder.gif',
|
|
openFolderIcon: '$frameworkDir/javascript/tree/images/page-openfolder.gif',
|
|
closedFolderIcon: '$frameworkDir/javascript/tree/images/page-closedfolder.gif'
|
|
};
|
|
JS
|
|
);
|
|
|
|
CMSBatchActionHandler::register('delete', 'AssetAdmin_DeleteBatchAction', 'Folder');
|
|
}
|
|
|
|
/**
|
|
* Returns the files and subfolders contained in the currently selected folder,
|
|
* defaulting to the root node. Doubles as search results, if any search parameters
|
|
* are set through {@link SearchForm()}.
|
|
*
|
|
* @return SS_List
|
|
*/
|
|
public function getList() {
|
|
$folder = $this->currentPage();
|
|
$context = $this->getSearchContext();
|
|
// Overwrite name filter to search both Name and Title attributes
|
|
$context->removeFilterByName('Name');
|
|
$params = $this->getRequest()->requestVar('q');
|
|
$list = $context->getResults($params);
|
|
|
|
// Don't filter list when a detail view is requested,
|
|
// to avoid edge cases where the filtered list wouldn't contain the requested
|
|
// record due to faulty session state (current folder not always encoded in URL, see #7408).
|
|
if(!$folder->ID
|
|
&& $this->getRequest()->requestVar('ID') === null
|
|
&& ($this->getRequest()->param('ID') == 'field')
|
|
) {
|
|
return $list;
|
|
}
|
|
|
|
// Re-add previously removed "Name" filter as combined filter
|
|
// TODO Replace with composite SearchFilter once that API exists
|
|
if(!empty($params['Name'])) {
|
|
$list = $list->filterAny(array(
|
|
'Name:PartialMatch' => $params['Name'],
|
|
'Title:PartialMatch' => $params['Name']
|
|
));
|
|
}
|
|
|
|
// Always show folders at the top
|
|
$list = $list->sort('(CASE WHEN "File"."ClassName" = \'Folder\' THEN 0 ELSE 1 END), "Name"');
|
|
|
|
// If a search is conducted, check for the "current folder" limitation.
|
|
// Otherwise limit by the current folder as denoted by the URL.
|
|
if(empty($params) || !empty($params['CurrentFolderOnly'])) {
|
|
$list = $list->filter('ParentID', $folder->ID);
|
|
}
|
|
|
|
// Category filter
|
|
if(!empty($params['AppCategory'])
|
|
&& !empty(File::config()->app_categories[$params['AppCategory']])
|
|
) {
|
|
$exts = File::config()->app_categories[$params['AppCategory']];
|
|
$list = $list->filter('Name:PartialMatch', $exts);
|
|
}
|
|
|
|
// Date filter
|
|
if(!empty($params['CreatedFrom'])) {
|
|
$fromDate = new DateField(null, null, $params['CreatedFrom']);
|
|
$list = $list->filter("Created:GreaterThanOrEqual", $fromDate->dataValue().' 00:00:00');
|
|
}
|
|
if(!empty($params['CreatedTo'])) {
|
|
$toDate = new DateField(null, null, $params['CreatedTo']);
|
|
$list = $list->filter("Created:LessThanOrEqual", $toDate->dataValue().' 23:59:59');
|
|
}
|
|
|
|
return $list;
|
|
}
|
|
|
|
public function getEditForm($id = null, $fields = null) {
|
|
|
|
$form = parent::getEditForm($id, $fields);
|
|
$folder = $this->currentPage();
|
|
$fields = $form->Fields();
|
|
$title = ($folder && $folder->isInDB()) ? $folder->Title : _t('AssetAdmin.FILES', 'Files');
|
|
$fields->push(new HiddenField('ID', false, $folder ? $folder->ID : null));
|
|
|
|
// File listing
|
|
$gridFieldConfig = GridFieldConfig::create()->addComponents(
|
|
new GridFieldToolbarHeader(),
|
|
new GridFieldSortableHeader(),
|
|
new GridFieldFilterHeader(),
|
|
new GridFieldDataColumns(),
|
|
new GridFieldPaginator(self::config()->page_length),
|
|
new GridFieldEditButton(),
|
|
new GridFieldDeleteAction(),
|
|
new GridFieldDetailForm(),
|
|
GridFieldLevelup::create($folder->ID)->setLinkSpec('admin/assets/show/%d')
|
|
);
|
|
|
|
$gridField = GridField::create('File', $title, $this->getList(), $gridFieldConfig);
|
|
$columns = $gridField->getConfig()->getComponentByType('GridFieldDataColumns');
|
|
$columns->setDisplayFields(array(
|
|
'StripThumbnail' => '',
|
|
'Title' => _t('File.Title', 'Title'),
|
|
'Created' => _t('AssetAdmin.CREATED', 'Date'),
|
|
'Size' => _t('AssetAdmin.SIZE', 'Size'),
|
|
));
|
|
$columns->setFieldCasting(array(
|
|
'Created' => 'SS_Datetime->Nice'
|
|
));
|
|
$gridField->setAttribute(
|
|
'data-url-folder-template',
|
|
Controller::join_links($this->Link('show'), '%s')
|
|
);
|
|
|
|
if($folder->canCreate()) {
|
|
$uploadBtn = new LiteralField(
|
|
'UploadButton',
|
|
sprintf(
|
|
'<a class="ss-ui-button font-icon-upload cms-panel-link" data-pjax-target="Content" data-icon="drive-upload" href="%s">%s</a>',
|
|
Controller::join_links(singleton('CMSFileAddController')->Link(), '?ID=' . $folder->ID),
|
|
_t('Folder.UploadFilesButton', 'Upload')
|
|
)
|
|
);
|
|
} else {
|
|
$uploadBtn = null;
|
|
}
|
|
|
|
if(!$folder->hasMethod('canAddChildren') || ($folder->hasMethod('canAddChildren') && $folder->canAddChildren())) {
|
|
// TODO Will most likely be replaced by GridField logic
|
|
$addFolderBtn = new LiteralField(
|
|
'AddFolderButton',
|
|
sprintf(
|
|
'<a class="ss-ui-button font-icon-plus-circled cms-add-folder-link" data-icon="add" data-url="%s" href="%s">%s</a>',
|
|
Controller::join_links($this->Link('AddForm'), '?' . http_build_query(array(
|
|
'action_doAdd' => 1,
|
|
'ParentID' => $folder->ID,
|
|
'SecurityID' => $form->getSecurityToken()->getValue()
|
|
))),
|
|
Controller::join_links($this->Link('addfolder'), '?ParentID=' . $folder->ID),
|
|
_t('Folder.AddFolderButton', 'Add folder')
|
|
)
|
|
);
|
|
} else {
|
|
$addFolderBtn = '';
|
|
}
|
|
|
|
if($folder->canEdit()) {
|
|
$syncButton = new LiteralField(
|
|
'SyncButton',
|
|
sprintf(
|
|
'<a class="ss-ui-button ss-ui-action ui-button-text-icon-primary ss-ui-button-ajax font-icon-sync" data-icon="arrow-circle-double" title="%s" href="%s">%s</a>',
|
|
_t('AssetAdmin.FILESYSTEMSYNCTITLE', 'Update the CMS database entries of files on the filesystem. Useful when new files have been uploaded outside of the CMS, e.g. through FTP.'),
|
|
$this->Link('doSync'),
|
|
_t('AssetAdmin.FILESYSTEMSYNC','Sync files')
|
|
)
|
|
);
|
|
} else {
|
|
$syncButton = null;
|
|
}
|
|
|
|
// Move existing fields to a "details" tab, unless they've already been tabbed out through extensions.
|
|
// Required to keep Folder->getCMSFields() simple and reuseable,
|
|
// without any dependencies into AssetAdmin (e.g. useful for "add folder" views).
|
|
if(!$fields->hasTabset()) {
|
|
$tabs = new TabSet('Root',
|
|
$tabList = new Tab('ListView', _t('AssetAdmin.ListView', 'List View')),
|
|
$tabTree = new Tab('TreeView', _t('AssetAdmin.TreeView', 'Tree View'))
|
|
);
|
|
$tabList->addExtraClass("content-listview cms-tabset-icon list");
|
|
$tabTree->addExtraClass("content-treeview cms-tabset-icon tree");
|
|
if($fields->Count() && $folder && $folder->isInDB()) {
|
|
$tabs->push($tabDetails = new Tab('DetailsView', _t('AssetAdmin.DetailsView', 'Details')));
|
|
$tabDetails->addExtraClass("content-galleryview cms-tabset-icon edit");
|
|
foreach($fields as $field) {
|
|
$fields->removeByName($field->getName());
|
|
$tabDetails->push($field);
|
|
}
|
|
}
|
|
$fields->push($tabs);
|
|
}
|
|
|
|
// we only add buttons if they're available. User might not have permission and therefore
|
|
// the button shouldn't be available. Adding empty values into a ComposteField breaks template rendering.
|
|
$actionButtonsComposite = CompositeField::create()->addExtraClass('cms-actions-row');
|
|
if($uploadBtn) $actionButtonsComposite->push($uploadBtn);
|
|
if($addFolderBtn) $actionButtonsComposite->push($addFolderBtn);
|
|
if($syncButton) $actionButtonsComposite->push($syncButton);
|
|
|
|
// List view
|
|
$fields->addFieldsToTab('Root.ListView', array(
|
|
$actionsComposite = CompositeField::create(
|
|
$actionButtonsComposite
|
|
)->addExtraClass('cms-content-toolbar field'),
|
|
$gridField
|
|
));
|
|
|
|
$treeField = new LiteralField('Tree', '');
|
|
// Tree view
|
|
$fields->addFieldsToTab('Root.TreeView', array(
|
|
clone $actionsComposite,
|
|
// TODO Replace with lazy loading on client to avoid performance hit of rendering potentially unused views
|
|
new LiteralField(
|
|
'Tree',
|
|
FormField::create_tag(
|
|
'div',
|
|
array(
|
|
'class' => 'cms-tree',
|
|
'data-url-tree' => $this->Link('getsubtree'),
|
|
'data-url-savetreenode' => $this->Link('savetreenode')
|
|
),
|
|
$this->SiteTreeAsUL()
|
|
)
|
|
)
|
|
));
|
|
|
|
// Move actions to "details" tab (they don't make sense on list/tree view)
|
|
$actions = $form->Actions();
|
|
$saveBtn = $actions->fieldByName('action_save');
|
|
$deleteBtn = $actions->fieldByName('action_delete');
|
|
$actions->removeByName('action_save');
|
|
$actions->removeByName('action_delete');
|
|
if(($saveBtn || $deleteBtn) && $fields->fieldByName('Root.DetailsView')) {
|
|
$fields->addFieldToTab(
|
|
'Root.DetailsView',
|
|
CompositeField::create($saveBtn,$deleteBtn)->addExtraClass('Actions')
|
|
);
|
|
}
|
|
|
|
|
|
|
|
$fields->setForm($form);
|
|
$form->setTemplate($this->getTemplatesWithSuffix('_EditForm'));
|
|
// TODO Can't merge $FormAttributes in template at the moment
|
|
$form->addExtraClass('cms-edit-form ' . $this->BaseCSSClasses());
|
|
$form->setAttribute('data-pjax-fragment', 'CurrentForm');
|
|
$form->Fields()->findOrMakeTab('Root')->setTemplate('CMSTabSet');
|
|
|
|
$this->extend('updateEditForm', $form);
|
|
|
|
return $form;
|
|
}
|
|
|
|
public function addfolder($request) {
|
|
$obj = $this->customise(array(
|
|
'EditForm' => $this->AddForm()
|
|
));
|
|
|
|
if($request->isAjax()) {
|
|
// Rendering is handled by template, which will call EditForm() eventually
|
|
$content = $obj->renderWith($this->getTemplatesWithSuffix('_Content'));
|
|
} else {
|
|
$content = $obj->renderWith($this->getViewer('show'));
|
|
}
|
|
|
|
return $content;
|
|
}
|
|
|
|
public function delete($data, $form) {
|
|
$className = $this->stat('tree_class');
|
|
|
|
$record = DataObject::get_by_id($className, $data['ID']);
|
|
if($record && !$record->canDelete()) return Security::permissionFailure();
|
|
if(!$record || !$record->ID) throw new SS_HTTPResponse_Exception("Bad record ID #" . (int)$data['ID'], 404);
|
|
$parentID = $record->ParentID;
|
|
$this->setCurrentPageID($parentID);
|
|
$record->delete();
|
|
|
|
$this->getResponse()->addHeader('X-Status', rawurlencode(_t('LeftAndMain.DELETED', 'Deleted.')));
|
|
$this->getResponse()->addHeader('X-Pjax', 'Content');
|
|
return $this->redirect(Controller::join_links($this->Link('show'), $parentID ? $parentID : 0));
|
|
}
|
|
|
|
/**
|
|
* Get the search context
|
|
*
|
|
* @return SearchContext
|
|
*/
|
|
public function getSearchContext() {
|
|
$context = singleton('File')->getDefaultSearchContext();
|
|
|
|
// Namespace fields, for easier detection if a search is present
|
|
foreach($context->getFields() as $field) $field->setName(sprintf('q[%s]', $field->getName()));
|
|
foreach($context->getFilters() as $filter) $filter->setFullName(sprintf('q[%s]', $filter->getFullName()));
|
|
|
|
// Customize fields
|
|
$dateHeader = HeaderField::create('q[Date]', _t('CMSSearch.FILTERDATEHEADING', 'Date'), 4);
|
|
$dateFrom = DateField::create('q[CreatedFrom]', _t('CMSSearch.FILTERDATEFROM', 'From'))
|
|
->setConfig('showcalendar', true);
|
|
$dateTo = DateField::create('q[CreatedTo]',_t('CMSSearch.FILTERDATETO', 'To'))
|
|
->setConfig('showcalendar', true);
|
|
$dateGroup = FieldGroup::create(
|
|
$dateHeader,
|
|
$dateFrom,
|
|
$dateTo
|
|
);
|
|
$context->addField($dateGroup);
|
|
$appCategories = array(
|
|
'image' => _t('AssetAdmin.AppCategoryImage', 'Image'),
|
|
'audio' => _t('AssetAdmin.AppCategoryAudio', 'Audio'),
|
|
'mov' => _t('AssetAdmin.AppCategoryVideo', 'Video'),
|
|
'flash' => _t('AssetAdmin.AppCategoryFlash', 'Flash', 'The fileformat'),
|
|
'zip' => _t('AssetAdmin.AppCategoryArchive', 'Archive', 'A collection of files'),
|
|
'doc' => _t('AssetAdmin.AppCategoryDocument', 'Document')
|
|
);
|
|
$context->addField(
|
|
$typeDropdown = new DropdownField(
|
|
'q[AppCategory]',
|
|
_t('AssetAdmin.Filetype', 'File type'),
|
|
$appCategories
|
|
)
|
|
);
|
|
|
|
$typeDropdown->setEmptyString(' ');
|
|
|
|
$context->addField(
|
|
new CheckboxField('q[CurrentFolderOnly]', _t('AssetAdmin.CurrentFolderOnly', 'Limit to current folder?'))
|
|
);
|
|
$context->getFields()->removeByName('q[Title]');
|
|
|
|
return $context;
|
|
}
|
|
|
|
/**
|
|
* Returns a form for filtering of files and assets gridfield.
|
|
* Result filtering takes place in {@link getList()}.
|
|
*
|
|
* @return Form
|
|
* @see AssetAdmin.js
|
|
*/
|
|
public function SearchForm() {
|
|
$folder = $this->currentPage();
|
|
$context = $this->getSearchContext();
|
|
|
|
$fields = $context->getSearchFields();
|
|
$actions = new FieldList(
|
|
FormAction::create('doSearch', _t('CMSMain_left_ss.APPLY_FILTER', 'Apply Filter'))
|
|
->addExtraClass('ss-ui-action-constructive'),
|
|
Object::create('ResetFormAction', 'clear', _t('CMSMain_left_ss.RESET', 'Reset'))
|
|
);
|
|
|
|
$form = new Form($this, 'filter', $fields, $actions);
|
|
$form->setFormMethod('GET');
|
|
$form->setFormAction(Controller::join_links($this->Link('show'), $folder->ID));
|
|
$form->addExtraClass('cms-search-form');
|
|
$form->loadDataFrom($this->getRequest()->getVars());
|
|
$form->disableSecurityToken();
|
|
// This have to match data-name attribute on the gridfield so that the javascript selectors work
|
|
$form->setAttribute('data-gridfield', 'File');
|
|
return $form;
|
|
}
|
|
|
|
public function AddForm() {
|
|
$folder = singleton('Folder');
|
|
$form = CMSForm::create(
|
|
$this,
|
|
'AddForm',
|
|
new FieldList(
|
|
new TextField("Name", _t('File.Name')),
|
|
new HiddenField('ParentID', false, $this->getRequest()->getVar('ParentID'))
|
|
),
|
|
new FieldList(
|
|
FormAction::create('doAdd', _t('AssetAdmin_left_ss.GO','Go'))
|
|
->addExtraClass('ss-ui-action-constructive')->setAttribute('data-icon', 'accept')
|
|
->setTitle(_t('AssetAdmin.ActionAdd', 'Add folder'))
|
|
)
|
|
)->setHTMLID('Form_AddForm');
|
|
$form->setResponseNegotiator($this->getResponseNegotiator());
|
|
$form->setTemplate($this->getTemplatesWithSuffix('_EditForm'));
|
|
// TODO Can't merge $FormAttributes in template at the moment
|
|
$form->addExtraClass('add-form cms-add-form cms-edit-form cms-panel-padded center ' . $this->BaseCSSClasses());
|
|
|
|
return $form;
|
|
}
|
|
|
|
/**
|
|
* Add a new group and return its details suitable for ajax.
|
|
*
|
|
* @todo Move logic into Folder class, and use LeftAndMain->doAdd() default implementation.
|
|
*/
|
|
public function doAdd($data, $form) {
|
|
$class = $this->stat('tree_class');
|
|
|
|
// check create permissions
|
|
if(!singleton($class)->canCreate()) return Security::permissionFailure($this);
|
|
|
|
// check addchildren permissions
|
|
if(
|
|
singleton($class)->hasExtension('Hierarchy')
|
|
&& isset($data['ParentID'])
|
|
&& is_numeric($data['ParentID'])
|
|
&& $data['ParentID']
|
|
) {
|
|
$parentRecord = DataObject::get_by_id($class, $data['ParentID']);
|
|
if(
|
|
$parentRecord->hasMethod('canAddChildren')
|
|
&& !$parentRecord->canAddChildren()
|
|
) return Security::permissionFailure($this);
|
|
} else {
|
|
$parentRecord = null;
|
|
}
|
|
|
|
$parent = (isset($data['ParentID']) && is_numeric($data['ParentID'])) ? (int)$data['ParentID'] : 0;
|
|
$name = (isset($data['Name'])) ? basename($data['Name']) : _t('AssetAdmin.NEWFOLDER',"NewFolder");
|
|
if(!$parentRecord || !$parentRecord->ID) $parent = 0;
|
|
|
|
// Get the folder to be created
|
|
if($parentRecord && $parentRecord->ID) $filename = $parentRecord->FullPath . $name;
|
|
else $filename = ASSETS_PATH . '/' . $name;
|
|
|
|
// Actually create
|
|
if(!file_exists(ASSETS_PATH)) {
|
|
mkdir(ASSETS_PATH);
|
|
}
|
|
|
|
$record = new Folder();
|
|
$record->ParentID = $parent;
|
|
$record->Name = $record->Title = basename($filename);
|
|
|
|
// Ensure uniqueness
|
|
$i = 2;
|
|
$baseFilename = substr($record->Filename, 0, -1) . '-';
|
|
while(file_exists($record->FullPath)) {
|
|
$record->Filename = $baseFilename . $i . '/';
|
|
$i++;
|
|
}
|
|
|
|
$record->Name = $record->Title = basename($record->Filename);
|
|
$record->write();
|
|
|
|
mkdir($record->FullPath);
|
|
chmod($record->FullPath, Filesystem::config()->file_create_mask);
|
|
|
|
if($parentRecord) {
|
|
return $this->redirect(Controller::join_links($this->Link('show'), $parentRecord->ID));
|
|
} else {
|
|
return $this->redirect($this->Link());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Custom currentPage() method to handle opening the 'root' folder
|
|
*/
|
|
public function currentPage() {
|
|
$id = $this->currentPageID();
|
|
if ($id > 0) {
|
|
$folder = Folder::get()->byID($id);
|
|
if ($folder && $folder->isInDB()) {
|
|
return $folder;
|
|
}
|
|
}
|
|
// Fallback to root
|
|
$this->setCurrentPageID(null);
|
|
return new Folder();
|
|
}
|
|
|
|
public function getSiteTreeFor($className, $rootID = null, $childrenMethod = null, $numChildrenMethod = null, $filterFunction = null, $minNodeCount = 30) {
|
|
if (!$childrenMethod) $childrenMethod = 'ChildFolders';
|
|
if (!$numChildrenMethod) $numChildrenMethod = 'numChildFolders';
|
|
return parent::getSiteTreeFor($className, $rootID, $childrenMethod, $numChildrenMethod, $filterFunction, $minNodeCount);
|
|
}
|
|
|
|
public function getCMSTreeTitle() {
|
|
return Director::absoluteBaseURL() . "assets";
|
|
}
|
|
|
|
public function SiteTreeAsUL() {
|
|
return $this->getSiteTreeFor($this->stat('tree_class'), null, 'ChildFolders', 'numChildFolders');
|
|
}
|
|
|
|
//------------------------------------------------------------------------------------------//
|
|
|
|
// Data saving handlers
|
|
|
|
/**
|
|
* Can be queried with an ajax request to trigger the filesystem sync. It returns a FormResponse status message
|
|
* to display in the CMS
|
|
*/
|
|
public function doSync() {
|
|
$message = Filesystem::sync();
|
|
$this->getResponse()->addHeader('X-Status', rawurlencode($message));
|
|
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* #################################
|
|
* Garbage collection.
|
|
* #################################
|
|
*/
|
|
|
|
/**
|
|
* Removes all unused thumbnails from the file store
|
|
* and returns the status of the process to the user.
|
|
*/
|
|
public function deleteunusedthumbnails($request) {
|
|
// Protect against CSRF on destructive action
|
|
if(!SecurityToken::inst()->checkRequest($request)) return $this->httpError(400);
|
|
|
|
$count = 0;
|
|
$thumbnails = $this->getUnusedThumbnails();
|
|
|
|
if($thumbnails) {
|
|
foreach($thumbnails as $thumbnail) {
|
|
unlink(ASSETS_PATH . "/" . $thumbnail);
|
|
$count++;
|
|
}
|
|
}
|
|
|
|
$message = _t(
|
|
'AssetAdmin.THUMBSDELETED',
|
|
'{count} unused thumbnails have been deleted',
|
|
array('count' => $count)
|
|
);
|
|
$this->getResponse()->addHeader('X-Status', rawurlencode($message));
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Creates array containg all unused thumbnails.
|
|
*
|
|
* Array is created in three steps:
|
|
* 1. Scan assets folder and retrieve all thumbnails
|
|
* 2. Scan all HTMLField in system and retrieve thumbnails from them.
|
|
* 3. Count difference between two sets (array_diff)
|
|
*
|
|
* @return array
|
|
*/
|
|
private function getUnusedThumbnails() {
|
|
$allThumbnails = array();
|
|
$usedThumbnails = array();
|
|
$dirIterator = new RecursiveIteratorIterator(new RecursiveDirectoryIterator(ASSETS_PATH));
|
|
$classes = ClassInfo::subclassesFor('SiteTree');
|
|
|
|
if($dirIterator) {
|
|
foreach($dirIterator as $file) {
|
|
if($file->isFile()) {
|
|
if(strpos($file->getPathname(), '_resampled') !== false) {
|
|
$pathInfo = pathinfo($file->getPathname());
|
|
if(in_array(strtolower($pathInfo['extension']), array('jpeg', 'jpg', 'jpe', 'png', 'gif'))) {
|
|
$path = str_replace('\\','/', $file->getPathname());
|
|
$allThumbnails[] = substr($path, strpos($path, '/assets/') + 8);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if($classes) {
|
|
foreach($classes as $className) {
|
|
$SNG_class = singleton($className);
|
|
$objects = DataObject::get($className);
|
|
|
|
if($objects !== NULL) {
|
|
foreach($objects as $object) {
|
|
foreach($SNG_class->db() as $fieldName => $fieldType) {
|
|
if($fieldType == 'HTMLText') {
|
|
$url1 = HTTP::findByTagAndAttribute($object->$fieldName,array('img' => 'src'));
|
|
|
|
if($url1 != NULL) {
|
|
$usedThumbnails[] = substr($url1[0], strpos($url1[0], '/assets/') + 8);
|
|
}
|
|
|
|
if($object->latestPublished > 0) {
|
|
$object = Versioned::get_latest_version($className, $object->ID);
|
|
$url2 = HTTP::findByTagAndAttribute($object->$fieldName, array('img' => 'src'));
|
|
|
|
if($url2 != NULL) {
|
|
$usedThumbnails[] = substr($url2[0], strpos($url2[0], '/assets/') + 8);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return array_diff($allThumbnails, $usedThumbnails);
|
|
}
|
|
|
|
/**
|
|
* @param bool $unlinked
|
|
* @return ArrayList
|
|
*/
|
|
public function Breadcrumbs($unlinked = false) {
|
|
$items = parent::Breadcrumbs($unlinked);
|
|
|
|
// The root element should explicitly point to the root node.
|
|
// Uses session state for current record otherwise.
|
|
$items[0]->Link = Controller::join_links(singleton('AssetAdmin')->Link('show'), 0);
|
|
|
|
// If a search is in progress, don't show the path
|
|
if($this->getRequest()->requestVar('q')) {
|
|
$items = $items->limit(1);
|
|
$items->push(new ArrayData(array(
|
|
'Title' => _t('LeftAndMain.SearchResults', 'Search Results'),
|
|
'Link' => Controller::join_links($this->Link(), '?' . http_build_query(array('q' => $this->getRequest()->requestVar('q'))))
|
|
)));
|
|
}
|
|
|
|
// If we're adding a folder, note that in breadcrumbs as well
|
|
if($this->getRequest()->param('Action') == 'addfolder') {
|
|
$items->push(new ArrayData(array(
|
|
'Title' => _t('Folder.AddFolderButton', 'Add folder'),
|
|
'Link' => false
|
|
)));
|
|
}
|
|
|
|
return $items;
|
|
}
|
|
|
|
public function providePermissions() {
|
|
$title = _t("AssetAdmin.MENUTITLE", LeftAndMain::menu_title_for_class($this->class));
|
|
return array(
|
|
"CMS_ACCESS_AssetAdmin" => array(
|
|
'name' => _t('CMSMain.ACCESS', "Access to '{title}' section", array('title' => $title)),
|
|
'category' => _t('Permission.CMS_ACCESS_CATEGORY', 'CMS Access')
|
|
)
|
|
);
|
|
}
|
|
|
|
}
|
|
/**
|
|
* Delete multiple {@link Folder} records (and the associated filesystem nodes).
|
|
* Usually used through the {@link AssetAdmin} interface.
|
|
*
|
|
* @package cms
|
|
* @subpackage batchactions
|
|
*/
|
|
class AssetAdmin_DeleteBatchAction extends CMSBatchAction {
|
|
public function getActionTitle() {
|
|
// _t('AssetAdmin_left_ss.SELECTTODEL','Select the folders that you want to delete and then click the button below')
|
|
return _t('AssetAdmin_DeleteBatchAction.TITLE', 'Delete folders');
|
|
}
|
|
|
|
public function run(SS_List $records) {
|
|
$status = array(
|
|
'modified'=>array(),
|
|
'deleted'=>array()
|
|
);
|
|
|
|
foreach($records as $record) {
|
|
$id = $record->ID;
|
|
|
|
// Perform the action
|
|
if($record->canDelete()) $record->delete();
|
|
|
|
$status['deleted'][$id] = array();
|
|
|
|
$record->destroy();
|
|
unset($record);
|
|
}
|
|
|
|
return Convert::raw2json($status);
|
|
}
|
|
}
|
|
|