mirror of
https://github.com/silverstripe/silverstripe-subsites
synced 2024-10-22 11:05:55 +02:00
Merge branch '2.1'
# Conflicts: # src/Extensions/SiteTreeSubsites.php
This commit is contained in:
commit
bf7dd9c37b
@ -8,3 +8,5 @@ SilverStripe\Core\Injector\Injector:
|
|||||||
properties:
|
properties:
|
||||||
Middlewares:
|
Middlewares:
|
||||||
SubsitesStateMiddleware: %$SilverStripe\Subsites\Middleware\InitStateMiddleware
|
SubsitesStateMiddleware: %$SilverStripe\Subsites\Middleware\InitStateMiddleware
|
||||||
|
SilverStripe\Dev\Tasks\MigrateFileTask:
|
||||||
|
class: SilverStripe\Subsites\Tasks\SubsiteMigrateFileTask
|
||||||
|
@ -170,57 +170,89 @@ class SiteTreeSubsites extends DataExtension
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Does the basic duplication, but doesn't write anything
|
* Does the basic duplication, but doesn't write anything this means we can subclass this easier and do more
|
||||||
* this means we can subclass this easier and do more complex
|
* complex relation duplication.
|
||||||
* relation duplication.
|
*
|
||||||
|
* Note that when duplicating including children, everything is written.
|
||||||
|
*
|
||||||
|
* @param Subsite|int $subsiteID
|
||||||
|
* @param bool $includeChildren
|
||||||
|
* @return SiteTree
|
||||||
*/
|
*/
|
||||||
public function duplicateToSubsitePrep($subsiteID)
|
public function duplicateToSubsitePrep($subsiteID, $includeChildren)
|
||||||
{
|
{
|
||||||
if (is_object($subsiteID)) {
|
if (is_object($subsiteID)) {
|
||||||
$subsiteID = $subsiteID->ID;
|
$subsiteID = $subsiteID->ID;
|
||||||
}
|
}
|
||||||
|
|
||||||
$oldSubsite = SubsiteState::singleton()->getSubsiteId();
|
return SubsiteState::singleton()
|
||||||
if ($subsiteID) {
|
->withState(function (SubsiteState $newState) use ($subsiteID, $includeChildren) {
|
||||||
Subsite::changeSubsite($subsiteID);
|
$newState->setSubsiteId($subsiteID);
|
||||||
} else {
|
|
||||||
$subsiteID = $oldSubsite;
|
/** @var SiteTree $page */
|
||||||
|
$page = $this->owner;
|
||||||
|
|
||||||
|
try {
|
||||||
|
// We have no idea what the ParentID should be, but it shouldn't be the same as it was since
|
||||||
|
// we're now in a different subsite. As a workaround use the url-segment and subsite ID.
|
||||||
|
if ($page->Parent()) {
|
||||||
|
$parentSeg = $page->Parent()->URLSegment;
|
||||||
|
$newParentPage = Page::get()->filter('URLSegment', $parentSeg)->first();
|
||||||
|
$originalParentID = $page->ParentID;
|
||||||
|
if ($newParentPage) {
|
||||||
|
$page->ParentID = (int) $newParentPage->ID;
|
||||||
|
} else {
|
||||||
|
// reset it to the top level, so the user can decide where to put it
|
||||||
|
$page->ParentID = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Disable query filtering by subsite during actual duplication
|
||||||
|
$originalFilter = Subsite::$disable_subsite_filter;
|
||||||
|
Subsite::disable_subsite_filter(true);
|
||||||
|
|
||||||
|
return $includeChildren ? $page->duplicateWithChildren() : $page->duplicate(false);
|
||||||
|
} finally {
|
||||||
|
Subsite::disable_subsite_filter($originalFilter);
|
||||||
|
|
||||||
|
// Re-set the original parent ID for the current page
|
||||||
|
$page->ParentID = $originalParentID;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When duplicating a page, assign the current subsite ID from the state
|
||||||
|
*/
|
||||||
|
public function onBeforeDuplicate()
|
||||||
|
{
|
||||||
|
$subsiteId = SubsiteState::singleton()->getSubsiteId();
|
||||||
|
if ($subsiteId !== null) {
|
||||||
|
$this->owner->SubsiteID = $subsiteId;
|
||||||
}
|
}
|
||||||
// doesn't write as we need to reset the SubsiteID, ParentID etc
|
|
||||||
$clone = $this->owner->duplicate(false);
|
|
||||||
$clone->CheckedPublicationDifferences = $clone->AddedToStage = true;
|
|
||||||
$subsiteID = ($subsiteID ? $subsiteID : $oldSubsite);
|
|
||||||
$clone->SubsiteID = $subsiteID;
|
|
||||||
// We have no idea what the parentID should be, so as a workaround use the url-segment and subsite ID
|
|
||||||
if ($this->owner->Parent()) {
|
|
||||||
$parentSeg = $this->owner->Parent()->URLSegment;
|
|
||||||
$newParentPage = Page::get()->filter('URLSegment', $parentSeg)->first();
|
|
||||||
if ($newParentPage) {
|
|
||||||
$clone->ParentID = $newParentPage->ID;
|
|
||||||
} else {
|
|
||||||
// reset it to the top level, so the user can decide where to put it
|
|
||||||
$clone->ParentID = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// MasterPageID is here for legacy purposes, to satisfy the subsites_relatedpages module
|
|
||||||
$clone->MasterPageID = $this->owner->ID;
|
|
||||||
return $clone;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a duplicate of this page and save it to another subsite
|
* Create a duplicate of this page and save it to another subsite
|
||||||
* @param $subsiteID int|Subsite The Subsite to copy to, or its ID
|
*
|
||||||
|
* @param Subsite|int $subsiteID The Subsite to copy to, or its ID
|
||||||
|
* @param boolean $includeChildren Whether to duplicate child pages too
|
||||||
|
* @return SiteTree The duplicated page
|
||||||
*/
|
*/
|
||||||
public function duplicateToSubsite($subsiteID = null)
|
public function duplicateToSubsite($subsiteID = null, $includeChildren = false)
|
||||||
{
|
{
|
||||||
$clone = $this->owner->duplicateToSubsitePrep($subsiteID);
|
$clone = $this->owner->duplicateToSubsitePrep($subsiteID, $includeChildren);
|
||||||
$clone->invokeWithExtensions('onBeforeDuplicateToSubsite', $this->owner);
|
$clone->invokeWithExtensions('onBeforeDuplicateToSubsite', $this->owner);
|
||||||
$clone->write();
|
|
||||||
|
if (!$includeChildren) {
|
||||||
|
// Write the new page if "include children" is false, because it is written by default when it's true.
|
||||||
|
$clone->write();
|
||||||
|
}
|
||||||
|
// Deprecated: manually duplicate any configured relationships
|
||||||
$clone->duplicateSubsiteRelations($this->owner);
|
$clone->duplicateSubsiteRelations($this->owner);
|
||||||
// new extension hooks which happens after write,
|
|
||||||
// onAfterDuplicate isn't reliable due to
|
|
||||||
// https://github.com/silverstripe/silverstripe-cms/issues/1253
|
|
||||||
$clone->invokeWithExtensions('onAfterDuplicateToSubsite', $this->owner);
|
$clone->invokeWithExtensions('onAfterDuplicateToSubsite', $this->owner);
|
||||||
|
|
||||||
return $clone;
|
return $clone;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -42,7 +42,7 @@ class GridFieldSubsiteDetailFormItemRequest extends GridFieldDetailForm_ItemRequ
|
|||||||
$templateArray
|
$templateArray
|
||||||
);
|
);
|
||||||
$templateDropdown->setEmptyString('(' . _t('Subsite.NOTEMPLATE', 'No template') . ')');
|
$templateDropdown->setEmptyString('(' . _t('Subsite.NOTEMPLATE', 'No template') . ')');
|
||||||
$form->Fields()->addFieldToTab('Root.Configuration', $templateDropdown);
|
$form->Fields()->addFieldToTab('Root.Main', $templateDropdown);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $form;
|
return $form;
|
||||||
|
@ -43,8 +43,12 @@ class InitStateMiddleware implements HTTPMiddleware
|
|||||||
|
|
||||||
return $delegate($request);
|
return $delegate($request);
|
||||||
} catch (DatabaseException $ex) {
|
} catch (DatabaseException $ex) {
|
||||||
// Database is not ready
|
$message = $ex->getMessage();
|
||||||
return $delegate($request);
|
if (strpos($message, 'No database selected') !== false) {
|
||||||
|
// Database is not ready, ignore and continue
|
||||||
|
return $delegate($request);
|
||||||
|
}
|
||||||
|
throw $ex;
|
||||||
} finally {
|
} finally {
|
||||||
// Persist to the session if using the CMS
|
// Persist to the session if using the CMS
|
||||||
if ($state->getUseSessions()) {
|
if ($state->getUseSessions()) {
|
||||||
@ -62,7 +66,7 @@ class InitStateMiddleware implements HTTPMiddleware
|
|||||||
public function getIsAdmin(HTTPRequest $request)
|
public function getIsAdmin(HTTPRequest $request)
|
||||||
{
|
{
|
||||||
$adminPaths = static::config()->get('admin_url_paths');
|
$adminPaths = static::config()->get('admin_url_paths');
|
||||||
$adminPaths[] = AdminRootController::config()->get('url_base') . '/';
|
$adminPaths[] = AdminRootController::admin_url();
|
||||||
$currentPath = rtrim($request->getURL(), '/') . '/';
|
$currentPath = rtrim($request->getURL(), '/') . '/';
|
||||||
foreach ($adminPaths as $adminPath) {
|
foreach ($adminPaths as $adminPath) {
|
||||||
if (substr($currentPath, 0, strlen($adminPath)) === $adminPath) {
|
if (substr($currentPath, 0, strlen($adminPath)) === $adminPath) {
|
||||||
|
19
src/Tasks/SubsiteMigrateFileTask.php
Normal file
19
src/Tasks/SubsiteMigrateFileTask.php
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace SilverStripe\Subsites\Tasks;
|
||||||
|
|
||||||
|
use SilverStripe\Dev\Tasks\MigrateFileTask;
|
||||||
|
use SilverStripe\Subsites\Model\Subsite;
|
||||||
|
|
||||||
|
class SubsiteMigrateFileTask extends MigrateFileTask
|
||||||
|
{
|
||||||
|
public function run($request)
|
||||||
|
{
|
||||||
|
$origDisableSubsiteFilter = Subsite::$disable_subsite_filter;
|
||||||
|
Subsite::disable_subsite_filter(true);
|
||||||
|
|
||||||
|
parent::run($request);
|
||||||
|
|
||||||
|
Subsite::disable_subsite_filter($origDisableSubsiteFilter);
|
||||||
|
}
|
||||||
|
}
|
@ -359,28 +359,41 @@ class SiteTreeSubsitesTest extends BaseSubsiteTest
|
|||||||
$this->assertEquals('important-page', $mainSubsiteImportantPage->URLSegment);
|
$this->assertEquals('important-page', $mainSubsiteImportantPage->URLSegment);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCopySubsiteWithChildren()
|
/**
|
||||||
|
* @param bool $withChildren
|
||||||
|
* @param int $expectedChildren
|
||||||
|
* @dataProvider duplicateToSubsiteProvider
|
||||||
|
*/
|
||||||
|
public function testDuplicateToSubsite($withChildren, $expectedChildren)
|
||||||
{
|
{
|
||||||
$page = $this->objFromFixture('Page', 'about');
|
/** @var SiteTree $page */
|
||||||
|
$page = $this->objFromFixture(Page::class, 'about');
|
||||||
|
/** @var Subsite $newSubsite */
|
||||||
$newSubsite = $this->objFromFixture(Subsite::class, 'subsite1');
|
$newSubsite = $this->objFromFixture(Subsite::class, 'subsite1');
|
||||||
|
|
||||||
$moved = $page->duplicateToSubsite($newSubsite->ID, true);
|
/** @var SiteTree $duplicatedPage */
|
||||||
$this->assertEquals($moved->SubsiteID, $newSubsite->ID, 'Ensure returned records are on new subsite');
|
$duplicatedPage = $page->duplicateToSubsite($newSubsite->ID, $withChildren);
|
||||||
$this->assertEquals(
|
$this->assertInstanceOf(SiteTree::class, $duplicatedPage, 'A new page is returned');
|
||||||
$moved->AllChildren()->count(),
|
|
||||||
$page->AllChildren()->count(),
|
$this->assertEquals($newSubsite->ID, $duplicatedPage->SubsiteID, 'Ensure returned records are on new subsite');
|
||||||
'All pages are copied across'
|
|
||||||
|
$this->assertCount(1, $page->AllChildren());
|
||||||
|
$this->assertCount(
|
||||||
|
$expectedChildren,
|
||||||
|
$duplicatedPage->AllChildren(),
|
||||||
|
'Duplicated page also duplicates children'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCopySubsiteWithoutChildren()
|
/**
|
||||||
|
* @return array[]
|
||||||
|
*/
|
||||||
|
public function duplicateToSubsiteProvider()
|
||||||
{
|
{
|
||||||
$page = $this->objFromFixture('Page', 'about');
|
return [
|
||||||
$newSubsite = $this->objFromFixture(Subsite::class, 'subsite2');
|
[true, 1],
|
||||||
|
[false, 0],
|
||||||
$moved = $page->duplicateToSubsite($newSubsite->ID, false);
|
];
|
||||||
$this->assertEquals($moved->SubsiteID, $newSubsite->ID, 'Ensure returned records are on new subsite');
|
|
||||||
$this->assertEquals($moved->AllChildren()->count(), 0, 'All pages are copied across');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testIfSubsiteThemeIsSetToThemeList()
|
public function testIfSubsiteThemeIsSetToThemeList()
|
||||||
|
Loading…
Reference in New Issue
Block a user