mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 06:05:56 +00:00
Merge remote-tracking branch 'origin/3.1' into 3
Conflicts: code/model/SiteConfig.php javascript/CMSMain.AddForm.js tests/model/SiteConfigTest.php tests/model/SiteTreeTest.php
This commit is contained in:
commit
cf4d9edafb
@ -54,6 +54,7 @@ class CMSMain extends LeftAndMain implements CurrentPageIdentifier, PermissionPr
|
|||||||
'treeview',
|
'treeview',
|
||||||
'listview',
|
'listview',
|
||||||
'ListViewForm',
|
'ListViewForm',
|
||||||
|
'childfilter',
|
||||||
);
|
);
|
||||||
|
|
||||||
public function init() {
|
public function init() {
|
||||||
@ -412,57 +413,35 @@ class CMSMain extends LeftAndMain implements CurrentPageIdentifier, PermissionPr
|
|||||||
$def['All'] = array();
|
$def['All'] = array();
|
||||||
|
|
||||||
// Identify disallows and set globals
|
// Identify disallows and set globals
|
||||||
$globalDisallowed = array();
|
|
||||||
foreach($classes as $class) {
|
|
||||||
$obj = singleton($class);
|
|
||||||
$needsPerm = $obj->stat('need_permission');
|
|
||||||
|
|
||||||
if(!($obj instanceof HiddenClass)) {
|
|
||||||
$def['All'][$class] = array(
|
|
||||||
'title' => $obj->i18n_singular_name()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(!$obj->stat('can_be_root')) {
|
|
||||||
$def['Root']['disallowedChildren'][] = $class;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(
|
|
||||||
($obj instanceof HiddenClass)
|
|
||||||
|| (!array_key_exists($class, $cacheCanCreate) || !$cacheCanCreate[$class])
|
|
||||||
|| ($needsPerm && !$this->can($needsPerm))
|
|
||||||
) {
|
|
||||||
$globalDisallowed[] = $class;
|
|
||||||
$def['Root']['disallowedChildren'][] = $class;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set disallows by class
|
|
||||||
foreach($classes as $class) {
|
foreach($classes as $class) {
|
||||||
$obj = singleton($class);
|
$obj = singleton($class);
|
||||||
if($obj instanceof HiddenClass) continue;
|
if($obj instanceof HiddenClass) continue;
|
||||||
|
|
||||||
|
// Name item
|
||||||
|
$def['All'][$class] = array(
|
||||||
|
'title' => $obj->i18n_singular_name()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Check if can be created at the root
|
||||||
|
$needsPerm = $obj->stat('need_permission');
|
||||||
|
if(
|
||||||
|
!$obj->stat('can_be_root')
|
||||||
|
|| (!array_key_exists($class, $cacheCanCreate) || !$cacheCanCreate[$class])
|
||||||
|
|| ($needsPerm && !$this->can($needsPerm))
|
||||||
|
) {
|
||||||
|
$def['Root']['disallowedChildren'][] = $class;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hint data specific to the class
|
||||||
$def[$class] = array();
|
$def[$class] = array();
|
||||||
|
|
||||||
$allowed = $obj->allowedChildren();
|
|
||||||
if($pos = array_search('SiteTree', $allowed)) unset($allowed[$pos]);
|
|
||||||
|
|
||||||
// Start by disallowing all classes which aren't specifically allowed,
|
|
||||||
// then add the ones which are globally disallowed.
|
|
||||||
$disallowed = array_diff($classes, (array)$allowed);
|
|
||||||
$disallowed = array_unique(array_merge($disallowed, $globalDisallowed));
|
|
||||||
// Re-index the array for JSON non sequential key issue
|
|
||||||
if($disallowed) $def[$class]['disallowedChildren'] = array_values($disallowed);
|
|
||||||
|
|
||||||
$defaultChild = $obj->defaultChild();
|
$defaultChild = $obj->defaultChild();
|
||||||
if($defaultChild != 'Page' && $defaultChild != null) {
|
if($defaultChild !== 'Page' && $defaultChild !== null) {
|
||||||
$def[$class]['defaultChild'] = $defaultChild;
|
$def[$class]['defaultChild'] = $defaultChild;
|
||||||
}
|
}
|
||||||
|
|
||||||
$defaultParent = $obj->defaultParent();
|
$defaultParent = $obj->defaultParent();
|
||||||
$parent = SiteTree::get_by_link($defaultParent);
|
if ($defaultParent !== 1 && $defaultParent !== null) {
|
||||||
$id = $parent ? $parent->id : null;
|
|
||||||
if ($defaultParent != 1 && $defaultParent != null) {
|
|
||||||
$def[$class]['defaultParent'] = $defaultParent;
|
$def[$class]['defaultParent'] = $defaultParent;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -491,8 +470,6 @@ class CMSMain extends LeftAndMain implements CurrentPageIdentifier, PermissionPr
|
|||||||
|
|
||||||
if($instance instanceof HiddenClass) continue;
|
if($instance instanceof HiddenClass) continue;
|
||||||
|
|
||||||
if(!$instance->canCreate()) continue;
|
|
||||||
|
|
||||||
// skip this type if it is restricted
|
// skip this type if it is restricted
|
||||||
if($instance->stat('need_permission') && !$this->can(singleton($class)->stat('need_permission'))) continue;
|
if($instance->stat('need_permission') && !$this->can(singleton($class)->stat('need_permission'))) continue;
|
||||||
|
|
||||||
@ -706,6 +683,39 @@ class CMSMain extends LeftAndMain implements CurrentPageIdentifier, PermissionPr
|
|||||||
return $this->renderWith($this->getTemplatesWithSuffix('_ListView'));
|
return $this->renderWith($this->getTemplatesWithSuffix('_ListView'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback to request the list of page types allowed under a given page instance.
|
||||||
|
* Provides a slower but more precise response over SiteTreeHints
|
||||||
|
*
|
||||||
|
* @param SS_HTTPRequest $request
|
||||||
|
* @return SS_HTTPResponse
|
||||||
|
*/
|
||||||
|
public function childfilter($request) {
|
||||||
|
// Check valid parent specified
|
||||||
|
$parentID = $request->requestVar('ParentID');
|
||||||
|
$parent = SiteTree::get()->byID($parentID);
|
||||||
|
if(!$parent || !$parent->exists()) return $this->httpError(404);
|
||||||
|
|
||||||
|
// Build hints specific to this class
|
||||||
|
// Identify disallows and set globals
|
||||||
|
$classes = SiteTree::page_type_classes();
|
||||||
|
$disallowedChildren = array();
|
||||||
|
foreach($classes as $class) {
|
||||||
|
$obj = singleton($class);
|
||||||
|
if($obj instanceof HiddenClass) continue;
|
||||||
|
|
||||||
|
if(!$obj->canCreate(null, array('Parent' => $parent))) {
|
||||||
|
$disallowedChildren[] = $class;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->extend('updateChildFilter', $disallowedChildren, $parentID);
|
||||||
|
return $this
|
||||||
|
->response
|
||||||
|
->addHeader('Content-Type', 'application/json; charset=utf-8')
|
||||||
|
->setBody(Convert::raw2json($disallowedChildren));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Safely reconstruct a selected filter from a given set of query parameters
|
* Safely reconstruct a selected filter from a given set of query parameters
|
||||||
*
|
*
|
||||||
|
@ -16,7 +16,7 @@ class CMSPageAddController extends CMSPageEditController {
|
|||||||
/**
|
/**
|
||||||
* @return Form
|
* @return Form
|
||||||
*/
|
*/
|
||||||
function AddForm() {
|
public function AddForm() {
|
||||||
$pageTypes = array();
|
$pageTypes = array();
|
||||||
foreach($this->PageTypes() as $type) {
|
foreach($this->PageTypes() as $type) {
|
||||||
$html = sprintf('<span class="page-icon class-%s"></span><strong class="title">%s</strong><span class="description">%s</span>',
|
$html = sprintf('<span class="page-icon class-%s"></span><strong class="title">%s</strong><span class="description">%s</span>',
|
||||||
@ -38,11 +38,6 @@ class CMSPageAddController extends CMSPageEditController {
|
|||||||
$childTitle = _t('CMSPageAddController.ParentMode_child', 'Under another page');
|
$childTitle = _t('CMSPageAddController.ParentMode_child', 'Under another page');
|
||||||
|
|
||||||
$fields = new FieldList(
|
$fields = new FieldList(
|
||||||
// TODO Should be part of the form attribute, but not possible in current form API
|
|
||||||
$hintsField = new LiteralField(
|
|
||||||
'Hints',
|
|
||||||
sprintf('<span class="hints" data-hints="%s"></span>', Convert::raw2xml($this->SiteTreeHints()))
|
|
||||||
),
|
|
||||||
new LiteralField('PageModeHeader', sprintf($numericLabelTmpl, 1, _t('CMSMain.ChoosePageParentMode', 'Choose where to create this page'))),
|
new LiteralField('PageModeHeader', sprintf($numericLabelTmpl, 1, _t('CMSMain.ChoosePageParentMode', 'Choose where to create this page'))),
|
||||||
$parentModeField = new SelectionGroup(
|
$parentModeField = new SelectionGroup(
|
||||||
"ParentModeField",
|
"ParentModeField",
|
||||||
@ -78,7 +73,7 @@ class CMSPageAddController extends CMSPageEditController {
|
|||||||
_t(
|
_t(
|
||||||
'CMSMain.AddPageRestriction',
|
'CMSMain.AddPageRestriction',
|
||||||
'Note: Some page types are not allowed for this selection'
|
'Note: Some page types are not allowed for this selection'
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
@ -122,6 +117,9 @@ class CMSPageAddController extends CMSPageEditController {
|
|||||||
$form = CMSForm::create(
|
$form = CMSForm::create(
|
||||||
$this, "AddForm", $fields, $actions
|
$this, "AddForm", $fields, $actions
|
||||||
)->setHTMLID('Form_AddForm');
|
)->setHTMLID('Form_AddForm');
|
||||||
|
$form->setAttribute('data-hints', $this->SiteTreeHints());
|
||||||
|
$form->setAttribute('data-childfilter', $this->Link('childfilter'));
|
||||||
|
|
||||||
$form->setResponseNegotiator($this->getResponseNegotiator());
|
$form->setResponseNegotiator($this->getResponseNegotiator());
|
||||||
$form->addExtraClass('cms-add-form stacked cms-content center cms-edit-form ' . $this->BaseCSSClasses());
|
$form->addExtraClass('cms-add-form stacked cms-content center cms-edit-form ' . $this->BaseCSSClasses());
|
||||||
$form->setTemplate($this->getTemplatesWithSuffix('_EditForm'));
|
$form->setTemplate($this->getTemplatesWithSuffix('_EditForm'));
|
||||||
@ -145,12 +143,8 @@ class CMSPageAddController extends CMSPageEditController {
|
|||||||
|
|
||||||
if(!$parentObj || !$parentObj->ID) $parentID = 0;
|
if(!$parentObj || !$parentObj->ID) $parentID = 0;
|
||||||
|
|
||||||
if($parentObj) {
|
if(!singleton($className)->canCreate(Member::currentUser(), array('Parent' => $parentObj))) {
|
||||||
if(!$parentObj->canAddChildren()) return Security::permissionFailure($this);
|
return Security::permissionFailure($this);
|
||||||
if(!singleton($className)->canCreate()) return Security::permissionFailure($this);
|
|
||||||
} else {
|
|
||||||
if(!SiteConfig::current_site_config()->canCreateTopLevel())
|
|
||||||
return Security::permissionFailure($this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$record = $this->getNewItem("new-$className-$parentID".$suffix, false);
|
$record = $this->getNewItem("new-$className-$parentID".$suffix, false);
|
||||||
|
@ -934,7 +934,7 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
// check for inherit
|
// check for inherit
|
||||||
if($this->CanViewType == 'Inherit') {
|
if($this->CanViewType == 'Inherit') {
|
||||||
if($this->ParentID) return $this->Parent()->canView($member);
|
if($this->ParentID) return $this->Parent()->canView($member);
|
||||||
else return $this->getSiteConfig()->canView($member);
|
else return $this->getSiteConfig()->canViewPages($member);
|
||||||
}
|
}
|
||||||
|
|
||||||
// check for any logged-in users
|
// check for any logged-in users
|
||||||
@ -1013,12 +1013,12 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* This function should return true if the current user can create new
|
* This function should return true if the current user can create new
|
||||||
* pages of this class. It can be overloaded to customise the security model for an
|
* pages of this class, regardless of context. It can be overloaded
|
||||||
* application.
|
* to customise the security model for an application.
|
||||||
*
|
*
|
||||||
* Denies permission if any of the following conditions is TRUE:
|
* By default, permission to create at the root level is based on the SiteConfig
|
||||||
* - canCreate() returns FALSE on any extension
|
* configuration, and permission to create beneath a parent is based on the
|
||||||
* - $can_create is set to FALSE and the site is not in "dev mode"
|
* ability to edit that parent page.
|
||||||
*
|
*
|
||||||
* Use {@link canAddChildren()} to control behaviour of creating children under this page.
|
* Use {@link canAddChildren()} to control behaviour of creating children under this page.
|
||||||
*
|
*
|
||||||
@ -1026,6 +1026,9 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
* @uses DataExtension->canCreate()
|
* @uses DataExtension->canCreate()
|
||||||
*
|
*
|
||||||
* @param Member $member
|
* @param Member $member
|
||||||
|
* @param array $context Optional array which may contain array('Parent' => $parentObj)
|
||||||
|
* If a parent page is known, it will be checked for validity.
|
||||||
|
* If omitted, it will be assumed this is to be created as a top level page.
|
||||||
* @return boolean True if the current user can create pages on this class.
|
* @return boolean True if the current user can create pages on this class.
|
||||||
*/
|
*/
|
||||||
public function canCreate($member = null) {
|
public function canCreate($member = null) {
|
||||||
@ -1033,16 +1036,31 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
$member = Member::currentUserID();
|
$member = Member::currentUserID();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check parent (custom canCreate option for SiteTree)
|
||||||
|
// Block children not allowed for this parent type
|
||||||
|
$context = func_num_args() > 1 ? func_get_arg(1) : array();
|
||||||
|
$parent = isset($context['Parent']) ? $context['Parent'] : null;
|
||||||
|
if($parent && !in_array(get_class($this), $parent->allowedChildren())) return false;
|
||||||
|
|
||||||
|
// Check permission
|
||||||
if($member && Permission::checkMember($member, "ADMIN")) return true;
|
if($member && Permission::checkMember($member, "ADMIN")) return true;
|
||||||
|
|
||||||
// Standard mechanism for accepting permission changes from extensions
|
// Standard mechanism for accepting permission changes from extensions
|
||||||
$extended = $this->extendedCan('canCreate', $member);
|
$results = $this->extend('canCreate', $member, $parent);
|
||||||
if($extended !== null) return $extended;
|
if(is_array($results) && ($results = array_filter($results, function($v) {return $v !== null;}))) {
|
||||||
|
return min($results);
|
||||||
|
}
|
||||||
|
|
||||||
return $this->stat('can_create') != false || Director::isDev();
|
// Fall over to inherited permissions
|
||||||
|
if($parent) {
|
||||||
|
return $parent->canAddChildren($member);
|
||||||
|
} else {
|
||||||
|
// This doesn't necessarily mean we are creating a root page, but that
|
||||||
|
// we don't know if there is a parent, so default to this permission
|
||||||
|
return SiteConfig::current_site_config()->canCreateTopLevel($member);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This function should return true if the current user can edit this
|
* This function should return true if the current user can edit this
|
||||||
* page. It can be overloaded to customise the security model for an
|
* page. It can be overloaded to customise the security model for an
|
||||||
@ -1083,7 +1101,7 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
|
|
||||||
// Default for unsaved pages
|
// Default for unsaved pages
|
||||||
} else {
|
} else {
|
||||||
return $this->getSiteConfig()->canEdit($member);
|
return $this->getSiteConfig()->canEditPages($member);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1289,7 +1307,7 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
if(empty(self::$cache_permissions[$cacheKey])) self::$cache_permissions[$cacheKey] = array();
|
if(empty(self::$cache_permissions[$cacheKey])) self::$cache_permissions[$cacheKey] = array();
|
||||||
self::$cache_permissions[$cacheKey] = $combinedStageResult + self::$cache_permissions[$cacheKey];
|
self::$cache_permissions[$cacheKey] = $combinedStageResult + self::$cache_permissions[$cacheKey];
|
||||||
|
|
||||||
return $combinedStageResult;
|
return $combinedStageResult;
|
||||||
} else {
|
} else {
|
||||||
return array();
|
return array();
|
||||||
}
|
}
|
||||||
@ -1305,7 +1323,7 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
* page can be edited.
|
* page can be edited.
|
||||||
*/
|
*/
|
||||||
static public function can_edit_multiple($ids, $memberID, $useCached = true) {
|
static public function can_edit_multiple($ids, $memberID, $useCached = true) {
|
||||||
return self::batch_permission_check($ids, $memberID, 'CanEditType', 'SiteTree_EditorGroups', 'canEdit', null, $useCached);
|
return self::batch_permission_check($ids, $memberID, 'CanEditType', 'SiteTree_EditorGroups', 'canEditPages', null, $useCached);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -2740,9 +2758,20 @@ class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvid
|
|||||||
* @return string a html string ready to be directly used in a template
|
* @return string a html string ready to be directly used in a template
|
||||||
*/
|
*/
|
||||||
public function getTreeTitle() {
|
public function getTreeTitle() {
|
||||||
|
// Build the list of candidate children
|
||||||
|
$children = array();
|
||||||
|
$candidates = static::page_type_classes();
|
||||||
|
foreach($this->allowedChildren() as $childClass) {
|
||||||
|
if(!in_array($childClass, $candidates)) continue;
|
||||||
|
$child = singleton($childClass);
|
||||||
|
if($child->canCreate(null, array('Parent' => $this))) {
|
||||||
|
$children[$childClass] = $child->i18n_singular_name();
|
||||||
|
}
|
||||||
|
}
|
||||||
$flags = $this->getStatusFlags();
|
$flags = $this->getStatusFlags();
|
||||||
$treeTitle = sprintf(
|
$treeTitle = sprintf(
|
||||||
"<span class=\"jstree-pageicon\"></span><span class=\"item\">%s</span>",
|
"<span class=\"jstree-pageicon\"></span><span class=\"item\" data-allowedchildren=\"%s\">%s</span>",
|
||||||
|
Convert::raw2att(Convert::raw2json($children)),
|
||||||
Convert::raw2xml(str_replace(array("\n","\r"),"",$this->MenuTitle))
|
Convert::raw2xml(str_replace(array("\n","\r"),"",$this->MenuTitle))
|
||||||
);
|
);
|
||||||
foreach($flags as $class => $data) {
|
foreach($flags as $class => $data) {
|
||||||
|
@ -26,7 +26,13 @@ class SiteTreeFileExtension extends DataExtension {
|
|||||||
* @param string $limit
|
* @param string $limit
|
||||||
* @return ManyManyList
|
* @return ManyManyList
|
||||||
*/
|
*/
|
||||||
public function BackLinkTracking($filter = "", $sort = "", $join = "", $limit = "") {
|
public function BackLinkTracking($filter = null, $sort = null, $join = null, $limit = null) {
|
||||||
|
if($filter !== null || $sort !== null || $join !== null || $limit !== null) {
|
||||||
|
Deprecation::notice('3.2', 'The $filter, $sort, $join and $limit parameters for
|
||||||
|
SiteTreeFileExtension::BackLinkTracking() have been deprecated.
|
||||||
|
Please manipluate the returned list directly.', Deprecation::SCOPE_GLOBAL);
|
||||||
|
}
|
||||||
|
|
||||||
if(class_exists("Subsite")){
|
if(class_exists("Subsite")){
|
||||||
$rememberSubsiteFilter = Subsite::$disable_subsite_filter;
|
$rememberSubsiteFilter = Subsite::$disable_subsite_filter;
|
||||||
Subsite::disable_subsite_filter(true);
|
Subsite::disable_subsite_filter(true);
|
||||||
|
@ -27,7 +27,7 @@ class SiteTreeFolderExtension extends DataExtension {
|
|||||||
$ids = $query->execute()->column();
|
$ids = $query->execute()->column();
|
||||||
if(!count($ids)) continue;
|
if(!count($ids)) continue;
|
||||||
|
|
||||||
foreach(singleton($className)->has_one() as $relName => $joinClass) {
|
foreach(singleton($className)->hasOne() as $relName => $joinClass) {
|
||||||
if($joinClass == 'Image' || $joinClass == 'File') {
|
if($joinClass == 'Image' || $joinClass == 'File') {
|
||||||
$fieldName = $relName .'ID';
|
$fieldName = $relName .'ID';
|
||||||
$query = DataList::create($className)->where("$fieldName > 0");
|
$query = DataList::create($className)->where("$fieldName > 0");
|
||||||
|
@ -109,7 +109,7 @@ class SiteTreeLinkTracking extends DataExtension {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Update the "LinkTracking" many_many
|
// Update the "LinkTracking" many_many
|
||||||
if($record->ID && $record->many_many('LinkTracking') && $tracker = $record->LinkTracking()) {
|
if($record->ID && $record->manyManyComponent('LinkTracking') && $tracker = $record->LinkTracking()) {
|
||||||
$tracker->removeByFilter(sprintf(
|
$tracker->removeByFilter(sprintf(
|
||||||
'"FieldName" = \'%s\' AND "%s" = %d',
|
'"FieldName" = \'%s\' AND "%s" = %d',
|
||||||
$fieldName,
|
$fieldName,
|
||||||
@ -123,7 +123,7 @@ class SiteTreeLinkTracking extends DataExtension {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Update the "ImageTracking" many_many
|
// Update the "ImageTracking" many_many
|
||||||
if($record->ID && $record->many_many('ImageTracking') && $tracker = $record->ImageTracking()) {
|
if($record->ID && $record->manyManyComponent('ImageTracking') && $tracker = $record->ImageTracking()) {
|
||||||
$tracker->removeByFilter(sprintf(
|
$tracker->removeByFilter(sprintf(
|
||||||
'"FieldName" = \'%s\' AND "%s" = %d',
|
'"FieldName" = \'%s\' AND "%s" = %d',
|
||||||
$fieldName,
|
$fieldName,
|
||||||
|
@ -54,7 +54,7 @@ class VirtualPage extends Page {
|
|||||||
$record = $this->CopyContentFrom();
|
$record = $this->CopyContentFrom();
|
||||||
|
|
||||||
$allFields = $record->db();
|
$allFields = $record->db();
|
||||||
if($hasOne = $record->has_one()) foreach($hasOne as $link) $allFields[$link . 'ID'] = "Int";
|
if($hasOne = $record->hasOne()) foreach($hasOne as $link) $allFields[$link . 'ID'] = "Int";
|
||||||
$virtualFields = array();
|
$virtualFields = array();
|
||||||
foreach($allFields as $field => $type) {
|
foreach($allFields as $field => $type) {
|
||||||
if(!in_array($field, $nonVirtualFields)) $virtualFields[] = $field;
|
if(!in_array($field, $nonVirtualFields)) $virtualFields[] = $field;
|
||||||
|
@ -15,43 +15,114 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
$(".cms-add-form").entwine({
|
$(".cms-add-form").entwine({
|
||||||
|
ParentID: 0, // Last selected parentID
|
||||||
|
ParentCache: {}, // Cache allowed children for each selected page
|
||||||
onadd: function() {
|
onadd: function() {
|
||||||
var self = this;
|
var self = this;
|
||||||
this.find('#Form_AddForm_ParentID_Holder .TreeDropdownField').bind('change', function() {
|
this.find('#Form_AddForm_ParentID_Holder .TreeDropdownField').bind('change', function() {
|
||||||
self.updateTypeList();
|
self.updateTypeList();
|
||||||
});
|
});
|
||||||
|
this.find(".SelectionGroup.parent-mode").bind('change', function() {
|
||||||
|
self.updateTypeList();
|
||||||
|
});
|
||||||
this.updateTypeList();
|
this.updateTypeList();
|
||||||
},
|
},
|
||||||
|
loadCachedChildren: function(parentID) {
|
||||||
|
var cache = this.getParentCache();
|
||||||
|
if(typeof cache[parentID] !== 'undefined') return cache[parentID];
|
||||||
|
else return null;
|
||||||
|
},
|
||||||
|
saveCachedChildren: function(parentID, children) {
|
||||||
|
var cache = this.getParentCache();
|
||||||
|
cache[parentID] = children;
|
||||||
|
this.setParentCache(cache);
|
||||||
|
},
|
||||||
/**
|
/**
|
||||||
* Limit page type selection based on parent class.
|
* Limit page type selection based on parent selection.
|
||||||
|
* Select of root classes is pre-computed, but selections with a given parent
|
||||||
|
* are updated on-demand.
|
||||||
* Similar implementation to LeftAndMain.Tree.js.
|
* Similar implementation to LeftAndMain.Tree.js.
|
||||||
*/
|
*/
|
||||||
updateTypeList: function() {
|
updateTypeList: function() {
|
||||||
var hints = this.find('.hints').data('hints'),
|
var hints = this.data('hints'),
|
||||||
metadata = this.find('#Form_AddForm_ParentID_Holder .TreeDropdownField').data('metadata'),
|
parentTree = this.find('#Form_AddForm_ParentID_Holder .TreeDropdownField'),
|
||||||
id = this.find('#Form_AddForm_ParentID_Holder .TreeDropdownField').getValue(),
|
parentMode = this.find("input[name=ParentModeField]:checked").val(),
|
||||||
newClassName = (id && metadata) ? metadata.ClassName : null,
|
metadata = parentTree.data('metadata'),
|
||||||
hintKey = (newClassName) ? newClassName : 'Root',
|
id = (metadata && parentMode === 'child')
|
||||||
hint = (typeof hints[hintKey] != 'undefined') ? hints[hintKey] : null,
|
? (parentTree.getValue() || this.getParentID())
|
||||||
allAllowed = true;
|
: null,
|
||||||
|
newClassName = metadata ? metadata.ClassName : null,
|
||||||
|
hintKey = (newClassName && parentMode === 'child')
|
||||||
|
? newClassName
|
||||||
|
: 'Root',
|
||||||
|
hint = (typeof hints[hintKey] !== 'undefined') ? hints[hintKey] : null,
|
||||||
|
self = this,
|
||||||
|
defaultChildClass = (hint && typeof hint.defaultChild !== 'undefined')
|
||||||
|
? hint.defaultChild
|
||||||
|
: null,
|
||||||
|
disallowedChildren = [];
|
||||||
|
|
||||||
var disallowedChildren = (hint && typeof hint.disallowedChildren != 'undefined') ? hint.disallowedChildren : [],
|
if(id) {
|
||||||
defaultChildClass = (hint && typeof hint.defaultChild != 'undefined') ? hint.defaultChild : null;
|
// Prevent interface operations
|
||||||
|
if(this.hasClass('loading')) return;
|
||||||
|
this.addClass('loading');
|
||||||
|
|
||||||
|
// Enable last parent ID to be re-selected from memory
|
||||||
|
this.setParentID(id);
|
||||||
|
if(!parentTree.getValue()) parentTree.setValue(id);
|
||||||
|
|
||||||
|
// Use cached data if available
|
||||||
|
disallowedChildren = this.loadCachedChildren(id);
|
||||||
|
if(disallowedChildren !== null) {
|
||||||
|
this.updateSelectionFilter(disallowedChildren, defaultChildClass);
|
||||||
|
this.removeClass('loading');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$.ajax({
|
||||||
|
url: self.data('childfilter'),
|
||||||
|
data: {'ParentID': id},
|
||||||
|
success: function(data) {
|
||||||
|
// reload current form and tree
|
||||||
|
self.saveCachedChildren(id, data);
|
||||||
|
self.updateSelectionFilter(data, defaultChildClass);
|
||||||
|
},
|
||||||
|
complete: function() {
|
||||||
|
self.removeClass('loading');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
disallowedChildren = (hint && typeof hint.disallowedChildren !== 'undefined')
|
||||||
|
? hint.disallowedChildren
|
||||||
|
: [],
|
||||||
|
this.updateSelectionFilter(disallowedChildren, defaultChildClass);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* Update the selection filter with the given blacklist and default selection
|
||||||
|
*
|
||||||
|
* @param array disallowedChildren
|
||||||
|
* @param string defaultChildClass
|
||||||
|
*/
|
||||||
|
updateSelectionFilter: function(disallowedChildren, defaultChildClass) {
|
||||||
// Limit selection
|
// Limit selection
|
||||||
|
var allAllowed = null; // troolian
|
||||||
this.find('#Form_AddForm_PageType_Holder li').each(function() {
|
this.find('#Form_AddForm_PageType_Holder li').each(function() {
|
||||||
var className = $(this).find('input').val(),
|
var className = $(this).find('input').val(),
|
||||||
isAllowed = ($.inArray(className, disallowedChildren) == -1);
|
isAllowed = ($.inArray(className, disallowedChildren) === -1);
|
||||||
|
|
||||||
$(this).setEnabled(isAllowed);
|
$(this).setEnabled(isAllowed);
|
||||||
if(!isAllowed) $(this).setSelected(false);
|
if(!isAllowed) $(this).setSelected(false);
|
||||||
allAllowed = allAllowed && isAllowed;
|
if(allAllowed === null) allAllowed = isAllowed;
|
||||||
|
else allAllowed = allAllowed && isAllowed;
|
||||||
});
|
});
|
||||||
|
|
||||||
// Set default child selection, or fall back to first available option
|
// Set default child selection, or fall back to first available option
|
||||||
if(defaultChildClass) {
|
if(defaultChildClass) {
|
||||||
var selectedEl = this.find('#Form_AddForm_PageType_Holder li input[value=' + defaultChildClass + ']').parents('li:first');
|
var selectedEl = this
|
||||||
|
.find('#Form_AddForm_PageType_Holder li input[value=' + defaultChildClass + ']')
|
||||||
|
.parents('li:first');
|
||||||
} else {
|
} else {
|
||||||
var selectedEl = this.find('#Form_AddForm_PageType_Holder li:not(.disabled):first');
|
var selectedEl = this.find('#Form_AddForm_PageType_Holder li:not(.disabled):first');
|
||||||
}
|
}
|
||||||
@ -59,7 +130,9 @@
|
|||||||
selectedEl.siblings().setSelected(false);
|
selectedEl.siblings().setSelected(false);
|
||||||
|
|
||||||
// Disable the "Create" button if none of the pagetypes are available
|
// Disable the "Create" button if none of the pagetypes are available
|
||||||
var buttonState = (this.find('#Form_AddForm_PageType_Holder li:not(.disabled)').length) ? 'enable' : 'disable';
|
var buttonState = this.find('#Form_AddForm_PageType_Holder li:not(.disabled)').length
|
||||||
|
? 'enable'
|
||||||
|
: 'disable';
|
||||||
this.find('button[name=action_doAdd]').button(buttonState);
|
this.find('button[name=action_doAdd]').button(buttonState);
|
||||||
|
|
||||||
this.find('.message-restricted')[allAllowed ? 'hide' : 'show']();
|
this.find('.message-restricted')[allAllowed ? 'hide' : 'show']();
|
||||||
@ -72,10 +145,13 @@
|
|||||||
},
|
},
|
||||||
setSelected: function(bool) {
|
setSelected: function(bool) {
|
||||||
var input = this.find('input');
|
var input = this.find('input');
|
||||||
this.toggleClass('selected', bool);
|
|
||||||
if(bool && !input.is(':disabled')) {
|
if(bool && !input.is(':disabled')) {
|
||||||
this.siblings().setSelected(false);
|
this.siblings().setSelected(false);
|
||||||
input.attr('checked', 'checked');
|
this.toggleClass('selected', true);
|
||||||
|
input.prop('checked', true);
|
||||||
|
} else {
|
||||||
|
this.toggleClass('selected', false);
|
||||||
|
input.prop('checked', false);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
setEnabled: function(bool) {
|
setEnabled: function(bool) {
|
||||||
|
@ -68,27 +68,15 @@
|
|||||||
// Build a list for allowed children as submenu entries
|
// Build a list for allowed children as submenu entries
|
||||||
var pagetype = node.data('pagetype'),
|
var pagetype = node.data('pagetype'),
|
||||||
id = node.data('id'),
|
id = node.data('id'),
|
||||||
disallowedChildren = (typeof hints[pagetype] != 'undefined') ? hints[pagetype].disallowedChildren : null,
|
allowedChildren = node.find('>a .item').data('allowedchildren'),
|
||||||
allowedChildren = $.extend(true, {}, hints['All']), // clone
|
|
||||||
disallowedClass,
|
|
||||||
menuAllowedChildren = {},
|
menuAllowedChildren = {},
|
||||||
hasAllowedChildren = false;
|
hasAllowedChildren = false;
|
||||||
|
|
||||||
// Filter allowed
|
|
||||||
if(disallowedChildren) {
|
|
||||||
for(var i=0; i<disallowedChildren.length; i++) {
|
|
||||||
disallowedClass = disallowedChildren[i];
|
|
||||||
if(allowedChildren[disallowedClass]) {
|
|
||||||
delete allowedChildren[disallowedClass];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert to menu entries
|
// Convert to menu entries
|
||||||
$.each(allowedChildren, function(klass, klassData){
|
$.each(allowedChildren, function(klass, title){
|
||||||
hasAllowedChildren = true;
|
hasAllowedChildren = true;
|
||||||
menuAllowedChildren["allowedchildren-" + klass ] = {
|
menuAllowedChildren["allowedchildren-" + klass ] = {
|
||||||
'label': '<span class="jstree-pageicon"></span>' + klassData.title,
|
'label': '<span class="jstree-pageicon"></span>' + title,
|
||||||
'_class': 'class-' + klass,
|
'_class': 'class-' + klass,
|
||||||
'action': function(obj) {
|
'action': function(obj) {
|
||||||
$('.cms-container').entwine('.ss').loadPanel(
|
$('.cms-container').entwine('.ss').loadPanel(
|
||||||
|
@ -19,7 +19,7 @@ $ExtraTreeTools
|
|||||||
</div>
|
</div>
|
||||||
<% end_if %>
|
<% end_if %>
|
||||||
|
|
||||||
<div class="cms-tree" data-url-tree="$LinkWithSearch($Link(getsubtree))" data-url-savetreenode="$Link(savetreenode)" data-url-updatetreenodes="$Link(updatetreenodes)" data-url-addpage="{$LinkPageAdd('AddForm/?action_doAdd=1', 'ParentID=%s&PageType=%s')}" data-url-editpage="$LinkPageEdit('%s')" data-url-duplicate="{$Link('duplicate/%s')}" data-url-duplicatewithchildren="{$Link('duplicatewithchildren/%s')}" data-url-listview="{$Link('?view=list')}" data-hints="$SiteTreeHints.XML" data-extra-params="SecurityID=$SecurityID">
|
<div class="cms-tree" data-url-tree="$LinkWithSearch($Link(getsubtree))" data-url-savetreenode="$Link(savetreenode)" data-url-updatetreenodes="$Link(updatetreenodes)" data-url-addpage="{$LinkPageAdd('AddForm/?action_doAdd=1', 'ParentID=%s&PageType=%s')}" data-url-editpage="$LinkPageEdit('%s')" data-url-duplicate="{$Link('duplicate/%s')}" data-url-duplicatewithchildren="{$Link('duplicatewithchildren/%s')}" data-url-listview="{$Link('?view=list')}" data-hints="$SiteTreeHints.XML" data-childfilter="$Link('childfilter')" data-extra-params="SecurityID=$SecurityID">
|
||||||
$SiteTreeAsUL
|
$SiteTreeAsUL
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -11,6 +11,9 @@ class CMSMainTest extends FunctionalTest {
|
|||||||
|
|
||||||
function testSiteTreeHints() {
|
function testSiteTreeHints() {
|
||||||
$cache = SS_Cache::factory('CMSMain_SiteTreeHints');
|
$cache = SS_Cache::factory('CMSMain_SiteTreeHints');
|
||||||
|
// Login as user with root creation privileges
|
||||||
|
$user = $this->objFromFixture('Member', 'rootedituser');
|
||||||
|
$user->logIn();
|
||||||
$cache->clean(Zend_Cache::CLEANING_MODE_ALL);
|
$cache->clean(Zend_Cache::CLEANING_MODE_ALL);
|
||||||
|
|
||||||
$rawHints = singleton('CMSMain')->SiteTreeHints();
|
$rawHints = singleton('CMSMain')->SiteTreeHints();
|
||||||
@ -46,23 +49,36 @@ class CMSMainTest extends FunctionalTest {
|
|||||||
$hints['Root']['disallowedChildren'],
|
$hints['Root']['disallowedChildren'],
|
||||||
'Limits root classes'
|
'Limits root classes'
|
||||||
);
|
);
|
||||||
$this->assertNotContains(
|
|
||||||
'CMSMainTest_ClassA',
|
}
|
||||||
// Lenient checks because other modules might influence state
|
|
||||||
(array)@$hints['Page']['disallowedChildren'],
|
public function testChildFilter() {
|
||||||
'Does not limit types on unlimited parent'
|
$this->logInWithPermission('ADMIN');
|
||||||
);
|
|
||||||
|
// Check page A
|
||||||
|
$pageA = new CMSMainTest_ClassA();
|
||||||
|
$pageA->write();
|
||||||
|
$pageB = new CMSMainTest_ClassB();
|
||||||
|
$pageB->write();
|
||||||
|
|
||||||
|
// Check query
|
||||||
|
$response = $this->get('CMSMain/childfilter?ParentID='.$pageA->ID);
|
||||||
|
$children = json_decode($response->getBody());
|
||||||
|
$this->assertFalse($response->isError());
|
||||||
|
|
||||||
|
// Page A can't have unrelated children
|
||||||
$this->assertContains(
|
$this->assertContains(
|
||||||
'Page',
|
'Page',
|
||||||
$hints['CMSMainTest_ClassA']['disallowedChildren'],
|
$children,
|
||||||
'Limited parent lists disallowed classes'
|
'Limited parent lists disallowed classes'
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// But it can create a ClassB
|
||||||
$this->assertNotContains(
|
$this->assertNotContains(
|
||||||
'CMSMainTest_ClassB',
|
'CMSMainTest_ClassB',
|
||||||
$hints['CMSMainTest_ClassA']['disallowedChildren'],
|
$children,
|
||||||
'Limited parent omits explicitly allowed classes in disallowedChildren'
|
'Limited parent omits explicitly allowed classes in disallowedChildren'
|
||||||
);
|
);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -302,11 +318,7 @@ class CMSMainTest extends FunctionalTest {
|
|||||||
'admin/pages/add/AddForm',
|
'admin/pages/add/AddForm',
|
||||||
array('ParentID' => $newPageId, 'PageType' => 'Page', 'Locale' => 'en_US', 'action_doAdd' => 1)
|
array('ParentID' => $newPageId, 'PageType' => 'Page', 'Locale' => 'en_US', 'action_doAdd' => 1)
|
||||||
);
|
);
|
||||||
$this->assertFalse($response->isError());
|
$this->assertEquals(403, $response->getStatusCode(), 'Add disallowed child should fail');
|
||||||
$this->assertContains(
|
|
||||||
htmlentities(_t('SiteTree.PageTypeNotAllowed', array('type' => 'Page'))),
|
|
||||||
$response->getBody()
|
|
||||||
);
|
|
||||||
|
|
||||||
$this->session()->inst_set('loggedInAs', NULL);
|
$this->session()->inst_set('loggedInAs', NULL);
|
||||||
|
|
||||||
|
@ -21,6 +21,13 @@ class SiteTreeTest extends SapphireTest {
|
|||||||
'SiteTreeTest_StageStatusInherit',
|
'SiteTreeTest_StageStatusInherit',
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ensure any current member is logged out
|
||||||
|
*/
|
||||||
|
public function logOut() {
|
||||||
|
if($member = Member::currentUser()) $member->logOut();
|
||||||
|
}
|
||||||
|
|
||||||
public function testCreateDefaultpages() {
|
public function testCreateDefaultpages() {
|
||||||
$remove = SiteTree::get();
|
$remove = SiteTree::get();
|
||||||
if($remove) foreach($remove as $page) $page->delete();
|
if($remove) foreach($remove as $page) $page->delete();
|
||||||
@ -442,10 +449,15 @@ class SiteTreeTest extends SapphireTest {
|
|||||||
$editor = $this->objFromFixture("Member", "editor");
|
$editor = $this->objFromFixture("Member", "editor");
|
||||||
|
|
||||||
$home = $this->objFromFixture("Page", "home");
|
$home = $this->objFromFixture("Page", "home");
|
||||||
|
$staff = $this->objFromFixture("Page", "staff");
|
||||||
$products = $this->objFromFixture("Page", "products");
|
$products = $this->objFromFixture("Page", "products");
|
||||||
$product1 = $this->objFromFixture("Page", "product1");
|
$product1 = $this->objFromFixture("Page", "product1");
|
||||||
$product4 = $this->objFromFixture("Page", "product4");
|
$product4 = $this->objFromFixture("Page", "product4");
|
||||||
|
|
||||||
|
// Test logged out users cannot edit
|
||||||
|
$this->logOut();
|
||||||
|
$this->assertFalse($staff->canEdit());
|
||||||
|
|
||||||
// Can't edit a page that is locked to admins
|
// Can't edit a page that is locked to admins
|
||||||
$this->assertFalse($home->canEdit($editor));
|
$this->assertFalse($home->canEdit($editor));
|
||||||
|
|
||||||
@ -469,6 +481,33 @@ class SiteTreeTest extends SapphireTest {
|
|||||||
$this->assertFalse($page->canEdit($securityAdminMember));
|
$this->assertFalse($page->canEdit($securityAdminMember));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testCreatePermissions() {
|
||||||
|
// Test logged out users cannot create
|
||||||
|
$this->logOut();
|
||||||
|
$this->assertFalse(singleton('SiteTree')->canCreate());
|
||||||
|
|
||||||
|
// Login with another permission
|
||||||
|
$this->logInWithPermission('DUMMY');
|
||||||
|
$this->assertFalse(singleton('SiteTree')->canCreate());
|
||||||
|
|
||||||
|
// Login with basic CMS permission
|
||||||
|
$perms = SiteConfig::config()->required_permission;
|
||||||
|
$this->logInWithPermission(reset($perms));
|
||||||
|
$this->assertTrue(singleton('SiteTree')->canCreate());
|
||||||
|
|
||||||
|
// Test creation underneath a parent which this user doesn't have access to
|
||||||
|
$parent = $this->objFromFixture('Page', 'about');
|
||||||
|
$this->assertFalse(singleton('SiteTree')->canCreate(null, array('Parent' => $parent)));
|
||||||
|
|
||||||
|
// Test creation underneath a parent which doesn't allow a certain child
|
||||||
|
$parentB = new SiteTreeTest_ClassB();
|
||||||
|
$parentB->Title = 'Only Allows SiteTreeTest_ClassC';
|
||||||
|
$parentB->write();
|
||||||
|
$this->assertTrue(singleton('SiteTreeTest_ClassA')->canCreate(null));
|
||||||
|
$this->assertFalse(singleton('SiteTreeTest_ClassA')->canCreate(null, array('Parent' => $parentB)));
|
||||||
|
$this->assertTrue(singleton('SiteTreeTest_ClassC')->canCreate(null, array('Parent' => $parentB)));
|
||||||
|
}
|
||||||
|
|
||||||
public function testEditPermissionsOnDraftVsLive() {
|
public function testEditPermissionsOnDraftVsLive() {
|
||||||
// Create an inherit-permission page
|
// Create an inherit-permission page
|
||||||
$page = new Page();
|
$page = new Page();
|
||||||
@ -506,7 +545,7 @@ class SiteTreeTest extends SapphireTest {
|
|||||||
// Confirm that Member.editor can still edit the page
|
// Confirm that Member.editor can still edit the page
|
||||||
$this->objFromFixture('Member','editor')->logIn();
|
$this->objFromFixture('Member','editor')->logIn();
|
||||||
$this->assertTrue($page->canEdit());
|
$this->assertTrue($page->canEdit());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCompareVersions() {
|
public function testCompareVersions() {
|
||||||
// Necessary to avoid
|
// Necessary to avoid
|
||||||
|
@ -1,3 +1,11 @@
|
|||||||
|
SiteConfig:
|
||||||
|
default:
|
||||||
|
Title: My test site
|
||||||
|
Tagline: Default site config
|
||||||
|
CanViewType: Anyone
|
||||||
|
CanEditType: LoggedInUsers
|
||||||
|
CanCreateTopLevelType: LoggedInUsers
|
||||||
|
|
||||||
Group:
|
Group:
|
||||||
editors:
|
editors:
|
||||||
Title: Editors
|
Title: Editors
|
||||||
|
Loading…
x
Reference in New Issue
Block a user