silverstripe-cms/code/Controllers/CMSPageHistoryController.php

456 lines
12 KiB
PHP
Raw Normal View History

<?php
2016-07-22 11:32:32 +12:00
namespace SilverStripe\CMS\Controllers;
2016-08-10 16:08:39 +12:00
use SilverStripe\CMS\Model\SiteTree;
use SilverStripe\Control\Controller;
2016-09-09 11:26:24 +12:00
use SilverStripe\Control\HTTPRequest;
use SilverStripe\Control\HTTPResponse;
use SilverStripe\Forms\CheckboxField;
use SilverStripe\Forms\FieldList;
use SilverStripe\Forms\Form;
use SilverStripe\Forms\FormAction;
use SilverStripe\Forms\HiddenField;
use SilverStripe\Forms\LiteralField;
2016-08-10 16:08:39 +12:00
use SilverStripe\ORM\FieldType\DBField;
use SilverStripe\ORM\FieldType\DBHTMLText;
use SilverStripe\ORM\Versioning\Versioned;
use SilverStripe\Security\Security;
2016-10-12 15:09:59 +13:00
use SilverStripe\View\ArrayData;
use SilverStripe\View\ViewableData;
class CMSPageHistoryController extends CMSMain {
private static $url_segment = 'pages/history';
2016-08-10 16:08:39 +12:00
private static $url_rule = '/$Action/$ID/$VersionID/$OtherVersionID';
2016-08-10 16:08:39 +12:00
private static $url_priority = 42;
2016-08-10 16:08:39 +12:00
private static $menu_title = 'History';
2016-08-10 16:08:39 +12:00
private static $required_permission_codes = 'CMS_ACCESS_CMSMain';
2016-03-09 09:50:55 +13:00
private static $allowed_actions = array(
'VersionsForm',
'CompareVersionsForm',
'show',
'compare'
);
2016-03-09 09:50:55 +13:00
private static $url_handlers = array(
'$Action/$ID/$VersionID/$OtherVersionID' => 'handleAction'
);
public function getResponseNegotiator() {
$negotiator = parent::getResponseNegotiator();
$controller = $this;
$negotiator->setCallback('CurrentForm', function() use(&$controller) {
$form = $controller->ShowVersionForm($controller->getRequest()->param('VersionID'));
if($form) return $form->forTemplate();
else return $controller->renderWith($controller->getTemplatesWithSuffix('_Content'));
});
$negotiator->setCallback('default', function() use(&$controller) {
return $controller->renderWith($controller->getViewer('show'));
});
return $negotiator;
}
2016-03-09 09:50:55 +13:00
/**
2016-09-09 11:26:24 +12:00
* @param HTTPRequest $request
* @return array
*/
public function show($request) {
$form = $this->ShowVersionForm($request->param('VersionID'));
2016-03-09 09:50:55 +13:00
$negotiator = $this->getResponseNegotiator();
$controller = $this;
$negotiator->setCallback('CurrentForm', function() use(&$controller, &$form) {
return $form ? $form->forTemplate() : $controller->renderWith($controller->getTemplatesWithSuffix('_Content'));
});
$negotiator->setCallback('default', function() use(&$controller, &$form) {
return $controller->customise(array('EditForm' => $form))->renderWith($controller->getViewer('show'));
});
return $negotiator->respond($request);
}
2016-03-09 09:50:55 +13:00
/**
2016-09-09 11:26:24 +12:00
* @param HTTPRequest $request
* @return array
*/
public function compare($request) {
$form = $this->CompareVersionsForm(
2016-03-09 09:50:55 +13:00
$request->param('VersionID'),
$request->param('OtherVersionID')
);
$negotiator = $this->getResponseNegotiator();
$controller = $this;
$negotiator->setCallback('CurrentForm', function() use(&$controller, &$form) {
return $form ? $form->forTemplate() : $controller->renderWith($controller->getTemplatesWithSuffix('_Content'));
});
$negotiator->setCallback('default', function() use(&$controller, &$form) {
return $controller->customise(array('EditForm' => $form))->renderWith($controller->getViewer('show'));
});
return $negotiator->respond($request);
}
public function getSilverStripeNavigator() {
$record = $this->getRecord($this->currentPageID(), $this->getRequest()->param('VersionID'));
if($record) {
$navigator = new SilverStripeNavigator($record);
return $navigator->renderWith($this->getTemplatesWithSuffix('_SilverStripeNavigator'));
} else {
return false;
}
}
2016-03-09 09:50:55 +13:00
/**
2016-03-09 09:50:55 +13:00
* Returns the read only version of the edit form. Detaches all {@link FormAction}
* instances attached since only action relates to revert.
*
* Permission checking is done at the {@link CMSMain::getEditForm()} level.
2016-03-09 09:50:55 +13:00
*
* @param int $id ID of the record to show
* @param array $fields optional
* @param int $versionID
2014-02-10 15:35:13 -05:00
* @param int $compareID Compare mode
*
* @return Form
*/
public function getEditForm($id = null, $fields = null, $versionID = null, $compareID = null) {
if(!$id) $id = $this->currentPageID();
2016-03-09 09:50:55 +13:00
$record = $this->getRecord($id, $versionID);
$versionID = ($record) ? $record->Version : $versionID;
2016-03-09 09:50:55 +13:00
$form = parent::getEditForm($record, ($record) ? $record->getCMSFields() : null);
// Respect permission failures from parent implementation
if(!($form instanceof Form)) return $form;
// TODO: move to the SilverStripeNavigator structure so the new preview can pick it up.
//$nav = new SilverStripeNavigatorItem_ArchiveLink($record);
$form->setActions(new FieldList(
$revert = FormAction::create('doRollback', _t('CMSPageHistoryController.REVERTTOTHISVERSION', 'Revert to this version'))->setUseButtonTag(true)
));
2016-03-09 09:50:55 +13:00
$fields = $form->Fields();
$fields->removeByName("Status");
$fields->push(new HiddenField("ID"));
$fields->push(new HiddenField("Version"));
2016-03-09 09:50:55 +13:00
$fields = $fields->makeReadonly();
if($compareID) {
$link = Controller::join_links(
$this->Link('show'),
$id
);
$view = _t('CMSPageHistoryController.VIEW',"view");
2016-03-09 09:50:55 +13:00
$message = _t(
'CMSPageHistoryController.COMPARINGVERSION',
"Comparing versions {version1} and {version2}.",
array(
'version1' => sprintf('%s (<a href="%s">%s</a>)', $versionID, Controller::join_links($link, $versionID), $view),
'version2' => sprintf('%s (<a href="%s">%s</a>)', $compareID, Controller::join_links($link, $compareID), $view)
)
);
2016-03-09 09:50:55 +13:00
$revert->setReadonly(true);
} else {
if($record->isLatestVersion()) {
$message = _t('CMSPageHistoryController.VIEWINGLATEST', 'Currently viewing the latest version.');
} else {
$message = _t(
'CMSPageHistoryController.VIEWINGVERSION',
"Currently viewing version {version}.",
array('version' => $versionID)
);
}
}
2016-03-09 09:50:55 +13:00
$fields->addFieldToTab('Root.Main',
2016-10-12 15:09:59 +13:00
new LiteralField('CurrentlyViewingMessage', ArrayData::create(array(
2016-08-10 16:08:39 +12:00
'Content' => DBField::create_field('HTMLFragment', $message),
'Classes' => 'notice'
))->renderWith($this->getTemplatesWithSuffix('_notice'))),
"Title"
);
$form->setFields($fields->makeReadonly());
$form->loadDataFrom(array(
"ID" => $id,
"Version" => $versionID,
));
2016-03-09 09:50:55 +13:00
if(($record && $record->isLatestVersion())) {
$revert->setReadonly(true);
}
2016-03-09 09:50:55 +13:00
$form->removeExtraClass('cms-content');
return $form;
}
2016-03-09 09:50:55 +13:00
/**
2016-03-09 09:50:55 +13:00
* Version select form. Main interface between selecting versions to view
* and comparing multiple versions.
2016-03-09 09:50:55 +13:00
*
* Because we can reload the page directly to a compare view (history/compare/1/2/3)
2016-03-09 09:50:55 +13:00
* this form has to adapt to those parameters as well.
*
* @return Form
*/
public function VersionsForm() {
$id = $this->currentPageID();
$page = $this->getRecord($id);
$versionsHtml = '';
$action = $this->getRequest()->param('Action');
$versionID = $this->getRequest()->param('VersionID');
$otherVersionID = $this->getRequest()->param('OtherVersionID');
2016-03-09 09:50:55 +13:00
$showUnpublishedChecked = 0;
$compareModeChecked = ($action == "compare");
if($page) {
$versions = $page->allVersions();
$versionID = (!$versionID) ? $page->Version : $versionID;
if($versions) {
foreach($versions as $k => $version) {
$active = false;
2016-03-09 09:50:55 +13:00
if($version->Version == $versionID || $version->Version == $otherVersionID) {
$active = true;
2016-03-09 09:50:55 +13:00
if(!$version->WasPublished) $showUnpublishedChecked = 1;
}
$version->Active = ($active);
}
}
2016-03-09 09:50:55 +13:00
$vd = new ViewableData();
2016-03-09 09:50:55 +13:00
$versionsHtml = $vd->customise(array(
'Versions' => $versions
2016-08-10 16:08:39 +12:00
))->renderWith($this->getTemplatesWithSuffix('_versions'));
}
$fields = new FieldList(
new CheckboxField(
'ShowUnpublished',
_t('CMSPageHistoryController.SHOWUNPUBLISHED','Show unpublished versions'),
$showUnpublishedChecked
),
new CheckboxField(
'CompareMode',
_t('CMSPageHistoryController.COMPAREMODE', 'Compare mode (select two)'),
$compareModeChecked
),
new LiteralField('VersionsHtml', $versionsHtml),
$hiddenID = new HiddenField('ID', false, "")
);
$actions = new FieldList(
new FormAction(
'doCompare', _t('CMSPageHistoryController.COMPAREVERSIONS','Compare Versions')
),
new FormAction(
2016-03-09 09:50:55 +13:00
'doShowVersion', _t('CMSPageHistoryController.SHOWVERSION','Show Version')
)
);
// Use <button> to allow full jQuery UI styling
2016-08-10 16:08:39 +12:00
foreach($actions->dataFields() as $action) {
/** @var FormAction $action */
$action->setUseButtonTag(true);
}
$form = Form::create(
$this,
'VersionsForm',
$fields,
$actions
)->setHTMLID('Form_VersionsForm');
$form->loadDataFrom($this->getRequest()->requestVars());
$hiddenID->setValue($id);
$form->unsetValidator();
2016-03-09 09:50:55 +13:00
$form
->addExtraClass('cms-versions-form') // placeholder, necessary for $.metadata() to work
->setAttribute('data-link-tmpl-compare', Controller::join_links($this->Link('compare'), '%s', '%s', '%s'))
->setAttribute('data-link-tmpl-show', Controller::join_links($this->Link('show'), '%s', '%s'));
2016-03-09 09:50:55 +13:00
return $form;
}
2016-03-09 09:50:55 +13:00
/**
* Process the {@link VersionsForm} compare function between two pages.
*
2016-08-10 16:08:39 +12:00
* @param array $data
* @param Form $form
2016-09-09 11:26:24 +12:00
* @return HTTPResponse|DBHTMLText
*/
public function doCompare($data, $form) {
$versions = $data['Versions'];
2016-08-10 16:08:39 +12:00
if(count($versions) < 2) {
return null;
}
2016-03-09 09:50:55 +13:00
$version1 = array_shift($versions);
$version2 = array_shift($versions);
$form = $this->CompareVersionsForm($version1, $version2);
// javascript solution, render into template
if($this->getRequest()->isAjax()) {
return $this->customise(array(
"EditForm" => $form
))->renderWith(array(
static::class . '_EditForm',
'LeftAndMain_Content'
));
}
2016-03-09 09:50:55 +13:00
// non javascript, redirect the user to the page
2016-08-10 16:08:39 +12:00
return $this->redirect(Controller::join_links(
$this->Link('compare'),
$version1,
$version2
));
}
/**
* Process the {@link VersionsForm} show version function. Only requires
* one page to be selected.
*
* @param array
* @param Form
*
2016-09-09 11:26:24 +12:00
* @return DBHTMLText|HTTPResponse
2016-03-09 09:50:55 +13:00
*/
public function doShowVersion($data, $form) {
$versionID = null;
2016-03-09 09:50:55 +13:00
if(isset($data['Versions']) && is_array($data['Versions'])) {
$versionID = array_shift($data['Versions']);
}
2016-03-09 09:50:55 +13:00
2016-08-10 16:08:39 +12:00
if(!$versionID) {
return null;
}
2016-03-09 09:50:55 +13:00
2016-08-10 16:08:39 +12:00
$request = $this->getRequest();
if($request->isAjax()) {
return $this->customise(array(
"EditForm" => $this->ShowVersionForm($versionID)
))->renderWith(array(
static::class . '_EditForm',
'LeftAndMain_Content'
));
}
// non javascript, redirect the user to the page
2016-08-10 16:08:39 +12:00
return $this->redirect(Controller::join_links(
$this->Link('version'),
$versionID
));
}
/**
2014-02-10 15:35:13 -05:00
* @param int|null $versionID
* @return Form
*/
public function ShowVersionForm($versionID = null) {
if(!$versionID) return null;
$id = $this->currentPageID();
$form = $this->getEditForm($id, null, $versionID);
return $form;
}
2016-03-09 09:50:55 +13:00
/**
2014-02-10 15:35:13 -05:00
* @param int $versionID
* @param int $otherVersionID
* @return mixed
*/
public function CompareVersionsForm($versionID, $otherVersionID) {
if($versionID > $otherVersionID) {
$toVersion = $versionID;
$fromVersion = $otherVersionID;
} else {
$toVersion = $otherVersionID;
$fromVersion = $versionID;
}
2016-08-10 16:08:39 +12:00
if(!$toVersion || !$fromVersion) {
return null;
}
2016-03-09 09:50:55 +13:00
$id = $this->currentPageID();
2016-08-10 16:08:39 +12:00
/** @var SiteTree $page */
$page = SiteTree::get()->byID($id);
2016-03-09 09:50:55 +13:00
$record = null;
if($page && $page->exists()) {
if(!$page->canView()) {
return Security::permissionFailure($this);
}
$record = $page->compareVersions($fromVersion, $toVersion);
}
2016-07-22 11:32:32 +12:00
$fromVersionRecord = Versioned::get_version('SilverStripe\\CMS\\Model\\SiteTree', $id, $fromVersion);
$toVersionRecord = Versioned::get_version('SilverStripe\\CMS\\Model\\SiteTree', $id, $toVersion);
2016-03-09 09:50:55 +13:00
if(!$fromVersionRecord) {
user_error("Can't find version $fromVersion of page $id", E_USER_ERROR);
}
2016-03-09 09:50:55 +13:00
if(!$toVersionRecord) {
user_error("Can't find version $toVersion of page $id", E_USER_ERROR);
}
2016-08-10 16:08:39 +12:00
if(!$record) {
return null;
}
2016-11-03 15:32:47 +00:00
$form = $this->getEditForm($id, null, $fromVersion, $toVersion);
2016-08-10 16:08:39 +12:00
$form->setActions(new FieldList());
$form->addExtraClass('compare');
2016-03-09 09:50:55 +13:00
2016-08-10 16:08:39 +12:00
// Comparison views shouldn't be editable.
// Its important to convert fields *before* loading data,
// as the comparison output is HTML and not valid values for the various field types
$readonlyFields = $form->Fields()->makeReadonly();
$form->setFields($readonlyFields);
2016-03-09 09:50:55 +13:00
2016-08-10 16:08:39 +12:00
$form->loadDataFrom($record);
$form->loadDataFrom(array(
"ID" => $id,
"Version" => $fromVersion,
));
foreach($form->Fields()->dataFields() as $field) {
$field->dontEscape = true;
}
2016-07-25 14:12:31 +01:00
2016-08-10 16:08:39 +12:00
return $form;
}
public function Breadcrumbs($unlinked = false) {
$crumbs = parent::Breadcrumbs($unlinked);
$crumbs[0]->Title = _t('CMSPagesController.MENUTITLE');
return $crumbs;
}
}