2011-08-19 12:32:31 +12:00
|
|
|
<?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;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Control\Controller;
|
2016-09-09 11:26:24 +12:00
|
|
|
use SilverStripe\Control\HTTPRequest;
|
|
|
|
use SilverStripe\Control\HTTPResponse;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Forms\CheckboxField;
|
2017-02-28 13:39:30 +13:00
|
|
|
use SilverStripe\Forms\CompositeField;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Forms\FieldList;
|
|
|
|
use SilverStripe\Forms\Form;
|
|
|
|
use SilverStripe\Forms\FormAction;
|
|
|
|
use SilverStripe\Forms\HiddenField;
|
2017-02-28 13:39:30 +13:00
|
|
|
use SilverStripe\Forms\HTMLReadonlyField;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Forms\LiteralField;
|
2016-08-10 16:08:39 +12:00
|
|
|
use SilverStripe\ORM\FieldType\DBField;
|
|
|
|
use SilverStripe\ORM\FieldType\DBHTMLText;
|
2016-06-16 16:57:19 +12:00
|
|
|
use SilverStripe\ORM\Versioning\Versioned;
|
2016-06-23 11:51:20 +12:00
|
|
|
use SilverStripe\Security\Security;
|
2016-10-12 15:09:59 +13:00
|
|
|
use SilverStripe\View\ArrayData;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\View\ViewableData;
|
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
class CMSPageHistoryController extends CMSMain
|
|
|
|
{
|
|
|
|
|
|
|
|
private static $url_segment = 'pages/history';
|
|
|
|
|
|
|
|
private static $url_rule = '/$Action/$ID/$VersionID/$OtherVersionID';
|
|
|
|
|
|
|
|
private static $url_priority = 42;
|
|
|
|
|
|
|
|
private static $menu_title = 'History';
|
|
|
|
|
|
|
|
private static $required_permission_codes = 'CMS_ACCESS_CMSMain';
|
|
|
|
|
|
|
|
private static $allowed_actions = array(
|
|
|
|
'VersionsForm',
|
|
|
|
'CompareVersionsForm',
|
|
|
|
'show',
|
|
|
|
'compare'
|
|
|
|
);
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param HTTPRequest $request
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function show($request)
|
|
|
|
{
|
|
|
|
$form = $this->ShowVersionForm($request->param('VersionID'));
|
|
|
|
|
|
|
|
$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);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param HTTPRequest $request
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function compare($request)
|
|
|
|
{
|
|
|
|
$form = $this->CompareVersionsForm(
|
|
|
|
$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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* @param int $id ID of the record to show
|
|
|
|
* @param array $fields optional
|
|
|
|
* @param int $versionID
|
|
|
|
* @param int $compareID Compare mode
|
|
|
|
*
|
|
|
|
* @return Form
|
|
|
|
*/
|
|
|
|
public function getEditForm($id = null, $fields = null, $versionID = null, $compareID = null)
|
|
|
|
{
|
|
|
|
if (!$id) {
|
|
|
|
$id = $this->currentPageID();
|
|
|
|
}
|
|
|
|
|
|
|
|
$record = $this->getRecord($id, $versionID);
|
|
|
|
$versionID = ($record) ? $record->Version : $versionID;
|
|
|
|
|
|
|
|
$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)
|
|
|
|
));
|
|
|
|
|
|
|
|
$fields = $form->Fields();
|
|
|
|
$fields->removeByName("Status");
|
|
|
|
$fields->push(new HiddenField("ID"));
|
|
|
|
$fields->push(new HiddenField("Version"));
|
|
|
|
|
|
|
|
$fields = $fields->makeReadonly();
|
|
|
|
|
|
|
|
if ($compareID) {
|
|
|
|
$link = Controller::join_links(
|
|
|
|
$this->Link('show'),
|
|
|
|
$id
|
|
|
|
);
|
|
|
|
|
|
|
|
$view = _t('CMSPageHistoryController.VIEW', "view");
|
|
|
|
|
|
|
|
$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)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$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
|
|
|
|
2016-11-04 17:55:24 +13:00
|
|
|
$fields->fieldByName('Root.Main')->unshift(
|
|
|
|
new LiteralField('CurrentlyViewingMessage', ArrayData::create(array(
|
|
|
|
'Content' => DBField::create_field('HTMLFragment', $message),
|
|
|
|
'Classes' => 'notice'
|
|
|
|
))->renderWith($this->getTemplatesWithSuffix('_notice')))
|
|
|
|
);
|
2011-08-26 14:03:21 +12:00
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
$form->setFields($fields->makeReadonly());
|
|
|
|
$form->loadDataFrom(array(
|
|
|
|
"ID" => $id,
|
|
|
|
"Version" => $versionID,
|
|
|
|
));
|
|
|
|
|
|
|
|
if (($record && $record->isLatestVersion())) {
|
|
|
|
$revert->setReadonly(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
$form->removeExtraClass('cms-content');
|
|
|
|
|
|
|
|
return $form;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Version select form. Main interface between selecting versions to view
|
|
|
|
* and comparing multiple versions.
|
|
|
|
*
|
|
|
|
* Because we can reload the page directly to a compare view (history/compare/1/2/3)
|
|
|
|
* 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');
|
|
|
|
|
|
|
|
$showUnpublishedChecked = 0;
|
|
|
|
$compareModeChecked = ($action == "compare");
|
|
|
|
|
|
|
|
if ($page) {
|
|
|
|
$versions = $page->allVersions();
|
|
|
|
$versionID = (!$versionID) ? $page->Version : $versionID;
|
|
|
|
|
|
|
|
if ($versions) {
|
|
|
|
foreach ($versions as $k => $version) {
|
|
|
|
$active = false;
|
|
|
|
|
|
|
|
if ($version->Version == $versionID || $version->Version == $otherVersionID) {
|
|
|
|
$active = true;
|
|
|
|
|
|
|
|
if (!$version->WasPublished) {
|
|
|
|
$showUnpublishedChecked = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$version->Active = ($active);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$vd = new ViewableData();
|
|
|
|
|
|
|
|
$versionsHtml = $vd->customise(array(
|
|
|
|
'Versions' => $versions
|
|
|
|
))->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(
|
|
|
|
'doShowVersion',
|
|
|
|
_t('CMSPageHistoryController.SHOWVERSION', 'Show Version')
|
|
|
|
)
|
|
|
|
);
|
2012-11-29 14:43:05 +01:00
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
// Use <button> to allow full jQuery UI styling
|
|
|
|
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();
|
|
|
|
|
|
|
|
$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'));
|
|
|
|
|
|
|
|
return $form;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Process the {@link VersionsForm} compare function between two pages.
|
|
|
|
*
|
|
|
|
* @param array $data
|
|
|
|
* @param Form $form
|
|
|
|
* @return HTTPResponse|DBHTMLText
|
|
|
|
*/
|
|
|
|
public function doCompare($data, $form)
|
|
|
|
{
|
|
|
|
$versions = $data['Versions'];
|
|
|
|
if (count($versions) < 2) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$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'
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
// non javascript, redirect the user to the page
|
|
|
|
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
|
|
|
|
*
|
|
|
|
* @return DBHTMLText|HTTPResponse
|
|
|
|
*/
|
|
|
|
public function doShowVersion($data, $form)
|
|
|
|
{
|
|
|
|
$versionID = null;
|
|
|
|
|
|
|
|
if (isset($data['Versions']) && is_array($data['Versions'])) {
|
|
|
|
$versionID = array_shift($data['Versions']);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$versionID) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$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
|
|
|
|
return $this->redirect(Controller::join_links(
|
|
|
|
$this->Link('version'),
|
|
|
|
$versionID
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @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;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$toVersion || !$fromVersion) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$id = $this->currentPageID();
|
|
|
|
/** @var SiteTree $page */
|
|
|
|
$page = SiteTree::get()->byID($id);
|
|
|
|
|
|
|
|
$record = null;
|
|
|
|
if ($page && $page->exists()) {
|
|
|
|
if (!$page->canView()) {
|
|
|
|
return Security::permissionFailure($this);
|
|
|
|
}
|
|
|
|
|
|
|
|
$record = $page->compareVersions($fromVersion, $toVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
$fromVersionRecord = Versioned::get_version('SilverStripe\\CMS\\Model\\SiteTree', $id, $fromVersion);
|
|
|
|
$toVersionRecord = Versioned::get_version('SilverStripe\\CMS\\Model\\SiteTree', $id, $toVersion);
|
|
|
|
|
|
|
|
if (!$fromVersionRecord) {
|
|
|
|
user_error("Can't find version $fromVersion of page $id", E_USER_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$toVersionRecord) {
|
|
|
|
user_error("Can't find version $toVersion of page $id", E_USER_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$record) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
$form = $this->getEditForm($id, null, $fromVersion, $toVersion);
|
|
|
|
$form->setActions(new FieldList());
|
|
|
|
$form->addExtraClass('compare');
|
|
|
|
|
|
|
|
$form->loadDataFrom($record);
|
|
|
|
$form->loadDataFrom(array(
|
|
|
|
"ID" => $id,
|
|
|
|
"Version" => $fromVersion,
|
|
|
|
));
|
2017-02-28 16:34:46 +13:00
|
|
|
|
2017-02-28 13:39:30 +13:00
|
|
|
// Comparison views shouldn't be editable.
|
|
|
|
// As the comparison output is HTML and not valid values for the various field types
|
|
|
|
$readonlyFields = $this->transformReadonly($form->Fields());
|
|
|
|
$form->setFields($readonlyFields);
|
2017-02-28 16:34:46 +13:00
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
return $form;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function Breadcrumbs($unlinked = false)
|
|
|
|
{
|
|
|
|
$crumbs = parent::Breadcrumbs($unlinked);
|
|
|
|
$crumbs[0]->Title = _t('CMSPagesController.MENUTITLE', 'Pages');
|
|
|
|
return $crumbs;
|
|
|
|
}
|
2017-02-28 16:34:46 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Replace all data fields with HTML readonly fields to display diff
|
|
|
|
*
|
|
|
|
* @param FieldList $fields
|
|
|
|
* @return FieldList
|
|
|
|
*/
|
2017-02-28 13:39:30 +13:00
|
|
|
public function transformReadonly(FieldList $fields)
|
|
|
|
{
|
2017-02-28 16:34:46 +13:00
|
|
|
foreach ($fields->dataFields() as $field) {
|
|
|
|
if ($field instanceof HiddenField) {
|
|
|
|
continue;
|
2017-02-28 13:39:30 +13:00
|
|
|
}
|
2017-02-28 16:34:46 +13:00
|
|
|
$newField = $field->castedCopy(HTMLReadonlyField::class);
|
|
|
|
$fields->replaceField($field->getName(), $newField);
|
2017-02-28 13:39:30 +13:00
|
|
|
}
|
|
|
|
return $fields;
|
|
|
|
}
|
2012-04-12 19:23:20 +12:00
|
|
|
}
|