2011-07-15 16:05:58 +02:00
< ? php
2016-06-16 16:57:19 +12:00
2016-07-22 11:32:32 +12:00
namespace SilverStripe\CMS\Controllers ;
2016-06-16 16:57:19 +12:00
use SilverStripe\ORM\ArrayList ;
use SilverStripe\ORM\DataObject ;
use SilverStripe\ORM\SS_List ;
use SilverStripe\ORM\Versioning\Versioned ;
use SilverStripe\ORM\FieldType\DBField ;
2016-08-10 16:08:39 +12:00
use SilverStripe\Security\Member ;
2016-07-22 11:32:32 +12:00
use ViewableData ;
use CMSPreviewable ;
use ClassInfo ;
use LeftAndMain ;
use Controller ;
use SiteTreeFutureState ;
use SilverStripe\CMS\Model\RedirectorPage ;
2016-06-02 15:56:45 +12:00
2011-07-15 16:05:58 +02:00
/**
* Utility class representing links to different views of a record
* for CMS authors , usually for { @ link SiteTree } objects with " stage " and " live " links .
* Useful both in the CMS and alongside the page template ( for logged in authors ) .
2011-07-21 20:00:48 +02:00
* The class can be used for any { @ link DataObject } subclass implementing the { @ link CMSPreviewable } interface .
2016-03-09 09:50:55 +13:00
*
2011-07-15 16:05:58 +02:00
* New item types can be defined by extending the { @ link SilverStripeNavigatorItem } class ,
* for example the " cmsworkflow " module defines a new " future state " item with a date selector
* to view embargoed data at a future point in time . So the item doesn ' t always have to be a simple link .
2016-03-09 09:50:55 +13:00
*
2011-07-15 16:05:58 +02:00
* @ package cms
* @ subpackage content
*/
2011-07-21 20:00:48 +02:00
class SilverStripeNavigator extends ViewableData {
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
2016-04-27 19:47:25 +12:00
* @ var DataObject | CMSPreviewable
2011-07-15 16:05:58 +02:00
*/
protected $record ;
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
2016-04-27 19:47:25 +12:00
* @ param DataObject | CMSPreviewable $record
2011-07-15 16:05:58 +02:00
*/
2016-04-27 19:47:25 +12:00
public function __construct ( CMSPreviewable $record ) {
parent :: __construct ();
2011-07-15 16:05:58 +02:00
$this -> record = $record ;
}
/**
2011-10-26 18:39:21 +13:00
* @ return SS_List of SilverStripeNavigatorItem
2011-07-15 16:05:58 +02:00
*/
2012-09-19 12:07:46 +02:00
public function getItems () {
2011-07-21 20:00:48 +02:00
$items = array ();
2016-03-09 09:50:55 +13:00
2016-07-22 11:32:32 +12:00
$classes = ClassInfo :: subclassesFor ( 'SilverStripe\\CMS\\Controllers\\SilverStripeNavigatorItem' );
unset ( $classes [ 'SilverStripe\\CMS\\Controllers\\SilverStripeNavigatorItem' ]);
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
// Sort menu items according to priority
foreach ( $classes as $class ) {
2016-04-27 19:47:25 +12:00
/** @var SilverStripeNavigatorItem $item */
2011-07-15 16:05:58 +02:00
$item = new $class ( $this -> record );
2016-04-27 19:47:25 +12:00
if ( ! $item -> canView ()) {
continue ;
}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
// This funny litle formula ensures that the first item added with the same priority will be left-most.
$priority = $item -> getPriority () * 100 - 1 ;
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
// Ensure that we can have duplicates with the same (default) priority
2016-04-27 19:47:25 +12:00
while ( isset ( $items [ $priority ])) {
$priority ++ ;
}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
$items [ $priority ] = $item ;
}
ksort ( $items );
2011-07-21 20:00:48 +02:00
2012-12-06 16:34:48 +13:00
// Drop the keys and let the ArrayList handle the numbering, so $First, $Last and others work properly.
return new ArrayList ( array_values ( $items ));
2011-07-15 16:05:58 +02:00
}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
2016-04-27 19:47:25 +12:00
* @ return DataObject | CMSPreviewable
2011-07-15 16:05:58 +02:00
*/
2012-09-19 12:07:46 +02:00
public function getRecord () {
2011-07-15 16:05:58 +02:00
return $this -> record ;
}
/**
2016-04-27 19:47:25 +12:00
* @ param DataObject | CMSPreviewable $record
* @ return array template data
2011-07-15 16:05:58 +02:00
*/
2012-09-19 12:07:46 +02:00
static public function get_for_record ( $record ) {
2011-07-15 16:05:58 +02:00
$html = '' ;
$message = '' ;
$navigator = new SilverStripeNavigator ( $record );
$items = $navigator -> getItems ();
2016-03-09 09:50:55 +13:00
foreach ( $items as $item ) {
2011-07-15 16:05:58 +02:00
$text = $item -> getHTML ();
if ( $text ) $html .= $text ;
$newMessage = $item -> getMessage ();
2013-03-18 15:57:20 +02:00
if ( $newMessage && $item -> isActive ()) $message = $newMessage ;
2011-07-15 16:05:58 +02:00
}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
return array (
'items' => $html ,
'message' => $message
);
}
}
/**
* Navigator items are links that appear in the $SilverStripeNavigator bar .
2011-07-21 20:00:48 +02:00
* To add an item , extend this class - it will be automatically picked up .
* When instanciating items manually , please ensure to call { @ link canView ()} .
2016-03-09 09:50:55 +13:00
*
2011-07-15 16:05:58 +02:00
* @ package cms
* @ subpackage content
*/
2016-04-27 19:47:25 +12:00
abstract class SilverStripeNavigatorItem extends ViewableData {
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
2016-04-27 19:47:25 +12:00
* @ param DataObject | CMSPreviewable
2011-07-15 16:05:58 +02:00
*/
protected $record ;
2016-03-09 09:50:55 +13:00
2016-08-10 16:08:39 +12:00
/** @var string */
protected $recordLink ;
2011-07-15 16:05:58 +02:00
/**
2016-04-27 19:47:25 +12:00
* @ param DataObject | CMSPreviewable $record
2011-07-15 16:05:58 +02:00
*/
2016-04-27 19:47:25 +12:00
public function __construct ( CMSPreviewable $record ) {
parent :: __construct ();
2011-07-15 16:05:58 +02:00
$this -> record = $record ;
}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
2015-09-28 22:31:31 +13:00
* @ return string HTML , mostly a link - but can be more complex as well .
2011-07-15 16:05:58 +02:00
* For example , a " future state " item might show a date selector .
*/
2016-04-27 19:47:25 +12:00
abstract public function getHTML ();
2012-05-04 11:26:40 +12:00
/**
2015-09-28 22:31:31 +13:00
* @ return string
2012-11-15 14:05:04 +13:00
* Get the Title of an item
2012-05-04 11:26:40 +12:00
*/
2016-04-27 19:47:25 +12:00
abstract public function getTitle ();
2016-03-09 09:50:55 +13:00
2012-11-29 17:41:01 +13:00
/**
* Machine - friendly name .
2016-04-27 19:47:25 +12:00
*
* @ return string
2012-11-29 17:41:01 +13:00
*/
public function getName () {
return substr ( get_class ( $this ), strpos ( get_class ( $this ), '_' ) + 1 );
}
2011-07-21 20:00:48 +02:00
/**
* Optional link to a specific view of this record .
* Not all items are simple links , please use { @ link getHTML ()}
* to represent an item in markup unless you know what you ' re doing .
2016-03-09 09:50:55 +13:00
*
2015-09-28 22:31:31 +13:00
* @ return string
2011-07-21 20:00:48 +02:00
*/
2012-09-19 12:07:46 +02:00
public function getLink () {}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
2015-09-28 22:31:31 +13:00
* @ return string
2011-07-15 16:05:58 +02:00
*/
2012-09-19 12:07:46 +02:00
public function getMessage () {}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
* @ return DataObject
*/
2012-09-19 12:07:46 +02:00
public function getRecord () {
2011-07-15 16:05:58 +02:00
return $this -> record ;
2016-03-09 09:50:55 +13:00
}
2011-07-15 16:05:58 +02:00
/**
2015-09-28 22:31:31 +13:00
* @ return int
2011-07-15 16:05:58 +02:00
*/
2012-09-19 12:07:46 +02:00
public function getPriority () {
2011-07-15 16:05:58 +02:00
return $this -> stat ( 'priority' );
}
2016-03-09 09:50:55 +13:00
2011-07-15 16:05:58 +02:00
/**
2011-07-21 20:00:48 +02:00
* As items might convey different record states like a " stage " or " live " table ,
* an item can be active ( showing the record in this state ) .
2016-03-09 09:50:55 +13:00
*
2011-07-21 20:00:48 +02:00
* @ return boolean
*/
2012-09-19 12:07:46 +02:00
public function isActive () {
2011-07-21 20:00:48 +02:00
return false ;
}
2016-03-09 09:50:55 +13:00
2011-07-21 20:00:48 +02:00
/**
* Filters items based on member permissions or other criteria ,
* such as if a state is generally available for the current record .
2016-03-09 09:50:55 +13:00
*
2016-04-27 19:47:25 +12:00
* @ param Member $member
2011-07-15 16:05:58 +02:00
* @ return Boolean
*/
2012-09-19 12:07:46 +02:00
public function canView ( $member = null ) {
2011-07-15 16:05:58 +02:00
return true ;
}
2013-01-25 11:35:48 +01:00
/**
* Counts as " archived " if the current record is a different version from both live and draft .
2016-03-09 09:50:55 +13:00
*
2013-01-25 11:35:48 +01:00
* @ return boolean
*/
public function isArchived () {
2016-06-16 16:57:19 +12:00
if ( ! $this -> record -> hasExtension ( 'SilverStripe\ORM\Versioning\Versioned' )) return false ;
2016-03-09 09:50:55 +13:00
2013-01-25 11:35:48 +01:00
if ( ! isset ( $this -> record -> _cached_isArchived )) {
2016-06-02 15:56:45 +12:00
$baseClass = $this -> record -> baseClass ();
$currentDraft = Versioned :: get_by_stage ( $baseClass , Versioned :: DRAFT ) -> byID ( $this -> record -> ID );
$currentLive = Versioned :: get_by_stage ( $baseClass , Versioned :: LIVE ) -> byID ( $this -> record -> ID );
2016-03-09 09:50:55 +13:00
2013-01-25 11:35:48 +01:00
$this -> record -> _cached_isArchived = (
2016-03-09 09:50:55 +13:00
( ! $currentDraft || ( $currentDraft && $this -> record -> Version != $currentDraft -> Version ))
2013-01-25 11:35:48 +01:00
&& ( ! $currentLive || ( $currentLive && $this -> record -> Version != $currentLive -> Version ))
);
2016-08-10 16:08:39 +12:00
}
2013-01-25 11:35:48 +01:00
return $this -> record -> _cached_isArchived ;
}
2011-07-15 16:05:58 +02:00
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_CMSLink extends SilverStripeNavigatorItem {
2013-03-18 11:47:15 +01:00
/** @config */
2016-03-09 09:50:55 +13:00
private static $priority = 10 ;
2012-09-19 12:07:46 +02:00
public function getHTML () {
2011-07-21 20:00:48 +02:00
return sprintf (
'<a href="%s">%s</a>' ,
$this -> record -> CMSEditLink (),
_t ( 'ContentController.CMS' , 'CMS' )
);
2011-07-15 16:05:58 +02:00
}
2016-03-09 09:50:55 +13:00
2012-11-15 14:05:04 +13:00
public function getTitle () {
2016-03-09 09:50:55 +13:00
return _t ( 'ContentController.CMS' , 'CMS' , 'Used in navigation. Should be a short label' );
2012-11-15 14:05:04 +13:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function getLink () {
2011-07-21 20:00:48 +02:00
return $this -> record -> CMSEditLink ();
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function isActive () {
2013-02-01 14:16:19 -04:00
return ( Controller :: curr () instanceof LeftAndMain );
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function canView ( $member = null ) {
2013-01-09 21:44:52 +01:00
return (
2011-07-21 20:00:48 +02:00
// Don't show in CMS
2013-02-07 21:44:41 +01:00
! ( Controller :: curr () instanceof LeftAndMain )
2013-01-09 21:44:52 +01:00
// Don't follow redirects in preview, they break the CMS editing form
&& ! ( $this -> record instanceof RedirectorPage )
);
2011-07-15 16:05:58 +02:00
}
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_StageLink extends SilverStripeNavigatorItem {
2013-03-18 11:47:15 +01:00
/** @config */
private static $priority = 20 ;
2011-07-15 16:05:58 +02:00
2012-09-19 12:07:46 +02:00
public function getHTML () {
2011-07-21 20:00:48 +02:00
$draftPage = $this -> getDraftPage ();
if ( $draftPage ) {
$this -> recordLink = Controller :: join_links ( $draftPage -> AbsoluteLink (), " ?stage=Stage " );
2013-03-18 15:57:20 +02:00
return " <a " . ( $this -> isActive () ? 'class="current" ' : '' ) . " href= \" $this->recordLink\ " > " . _t('ContentController.DRAFTSITE', 'Draft Site') . " </ a > " ;
2011-07-15 16:05:58 +02:00
}
}
2012-05-04 11:26:40 +12:00
2012-11-15 14:05:04 +13:00
public function getTitle () {
2012-11-27 12:22:45 +13:00
return _t ( 'ContentController.DRAFT' , 'Draft' , 'Used for the Switch between draft and published view mode. Needs to be a short label' );
2012-05-04 11:26:40 +12:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function getMessage () {
2011-07-21 20:00:48 +02:00
return " <div id= \" SilverStripeNavigatorMessage \" title= \" " . _t ( 'ContentControl.NOTEWONTBESHOWN' , 'Note: this message will not be shown to your visitors' ) . " \" > " . _t ( 'ContentController.DRAFTSITE' , 'Draft Site' ) . " </div> " ;
2011-07-15 16:05:58 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function getLink () {
2013-01-25 11:35:48 +01:00
$date = Versioned :: current_archived_date ();
return Controller :: join_links (
2016-03-09 09:50:55 +13:00
$this -> record -> PreviewLink (),
2013-01-25 11:35:48 +01:00
'?stage=Stage' ,
$date ? '?archiveDate=' . $date : null
);
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function canView ( $member = null ) {
2013-01-09 21:44:52 +01:00
return (
2016-06-16 16:57:19 +12:00
$this -> record -> hasExtension ( 'SilverStripe\ORM\Versioning\Versioned' )
2013-01-09 21:44:52 +01:00
&& $this -> getDraftPage ()
// Don't follow redirects in preview, they break the CMS editing form
&& ! ( $this -> record instanceof RedirectorPage )
);
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function isActive () {
2011-07-21 20:00:48 +02:00
return (
2016-03-17 13:02:50 +13:00
Versioned :: get_stage () == 'Stage'
2011-07-21 20:00:48 +02:00
&& ! ( ClassInfo :: exists ( 'SiteTreeFutureState' ) && SiteTreeFutureState :: get_future_datetime ())
2013-01-25 11:35:48 +01:00
&& ! $this -> isArchived ()
2011-07-21 20:00:48 +02:00
);
}
2016-03-09 09:50:55 +13:00
2011-07-21 20:00:48 +02:00
protected function getDraftPage () {
2016-06-02 15:56:45 +12:00
$baseClass = $this -> record -> baseClass ();
return Versioned :: get_by_stage ( $baseClass , Versioned :: DRAFT ) -> byID ( $this -> record -> ID );
2011-07-15 16:05:58 +02:00
}
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_LiveLink extends SilverStripeNavigatorItem {
2013-03-18 11:47:15 +01:00
/** @config */
private static $priority = 30 ;
2011-07-15 16:05:58 +02:00
2012-09-19 12:07:46 +02:00
public function getHTML () {
2011-07-21 20:00:48 +02:00
$livePage = $this -> getLivePage ();
if ( $livePage ) {
$this -> recordLink = Controller :: join_links ( $livePage -> AbsoluteLink (), " ?stage=Live " );
2013-03-18 15:57:20 +02:00
return " <a " . ( $this -> isActive () ? 'class="current" ' : '' ) . " href= \" $this->recordLink\ " > " . _t('ContentController.PUBLISHEDSITE', 'Published Site') . " </ a > " ;
2011-07-15 16:05:58 +02:00
}
}
2012-05-04 11:26:40 +12:00
2012-11-15 14:05:04 +13:00
public function getTitle () {
2012-11-27 12:22:45 +13:00
return _t ( 'ContentController.PUBLISHED' , 'Published' , 'Used for the Switch between draft and published view mode. Needs to be a short label' );
2012-05-04 11:26:40 +12:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function getMessage () {
2011-07-21 20:00:48 +02:00
return " <div id= \" SilverStripeNavigatorMessage \" title= \" " . _t ( 'ContentControl.NOTEWONTBESHOWN' , 'Note: this message will not be shown to your visitors' ) . " \" > " . _t ( 'ContentController.PUBLISHEDSITE' , 'Published Site' ) . " </div> " ;
2011-07-15 16:05:58 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function getLink () {
2013-01-14 12:06:34 +01:00
return Controller :: join_links ( $this -> record -> PreviewLink (), '?stage=Live' );
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function canView ( $member = null ) {
2013-01-09 21:44:52 +01:00
return (
2016-06-16 16:57:19 +12:00
$this -> record -> hasExtension ( 'SilverStripe\ORM\Versioning\Versioned' )
2013-01-09 21:44:52 +01:00
&& $this -> getLivePage ()
// Don't follow redirects in preview, they break the CMS editing form
&& ! ( $this -> record instanceof RedirectorPage )
);
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function isActive () {
2013-01-25 11:35:48 +01:00
return (
2016-03-17 13:02:50 +13:00
( ! Versioned :: get_stage () || Versioned :: get_stage () == 'Live' )
2013-01-25 11:35:48 +01:00
&& ! $this -> isArchived ()
);
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2011-07-21 20:00:48 +02:00
protected function getLivePage () {
2016-06-02 15:56:45 +12:00
$baseClass = $this -> record -> baseClass ();
return Versioned :: get_by_stage ( $baseClass , Versioned :: LIVE ) -> byID ( $this -> record -> ID );
2011-07-15 16:05:58 +02:00
}
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_ArchiveLink extends SilverStripeNavigatorItem {
2013-03-18 11:47:15 +01:00
/** @config */
private static $priority = 40 ;
2011-07-15 16:05:58 +02:00
2012-09-19 12:07:46 +02:00
public function getHTML () {
2016-06-02 15:56:45 +12:00
$this -> recordLink = $this -> record -> AbsoluteLink ();
return " <a class= \" ss-ui-button " . ( $this -> isActive () ? ' current' : '' ) . " \" href= \" $this->recordLink ?archiveDate= { $this -> record -> LastEdited } \" target= \" _blank \" > " . _t ( 'ContentController.ARCHIVEDSITE' , 'Preview version' ) . " </a> " ;
2011-07-15 16:05:58 +02:00
}
2016-03-09 09:50:55 +13:00
2012-11-15 14:05:04 +13:00
public function getTitle () {
2013-01-25 11:35:48 +01:00
return _t ( 'SilverStripeNavigator.ARCHIVED' , 'Archived' );
2012-11-15 14:05:04 +13:00
}
2016-03-09 09:50:55 +13:00
public function getMessage () {
2011-10-31 12:15:07 +13:00
if ( $date = Versioned :: current_archived_date ()) {
2013-03-18 15:57:20 +02:00
$dateObj = DBField :: create_field ( 'Datetime' , $date );
2011-10-31 12:15:07 +13:00
return " <div id= \" SilverStripeNavigatorMessage \" title= \" " . _t ( 'ContentControl.NOTEWONTBESHOWN' , 'Note: this message will not be shown to your visitors' ) . " \" > " . _t ( 'ContentController.ARCHIVEDSITEFROM' , 'Archived site from' ) . " <br> " . $dateObj -> Nice () . " </div> " ;
}
2011-07-15 16:05:58 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function getLink () {
2013-01-25 11:35:48 +01:00
return $this -> record -> PreviewLink () . '?archiveDate=' . urlencode ( $this -> record -> LastEdited );
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function canView ( $member = null ) {
2013-01-09 21:44:52 +01:00
return (
2016-06-16 16:57:19 +12:00
$this -> record -> hasExtension ( 'SilverStripe\ORM\Versioning\Versioned' )
2013-01-09 21:44:52 +01:00
&& $this -> isArchived ()
// Don't follow redirects in preview, they break the CMS editing form
&& ! ( $this -> record instanceof RedirectorPage )
);
2011-07-21 20:00:48 +02:00
}
2016-03-09 09:50:55 +13:00
2012-09-19 12:07:46 +02:00
public function isActive () {
2013-01-25 11:35:48 +01:00
return $this -> isArchived ();
2011-07-15 16:05:58 +02:00
}
2016-04-27 19:47:25 +12:00
}