2011-07-15 16:05:58 +02:00
< ? php
/**
* 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 .
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 .
*
* @ package cms
* @ subpackage content
*/
2011-07-21 20:00:48 +02:00
class SilverStripeNavigator extends ViewableData {
2011-07-15 16:05:58 +02:00
/**
* @ var DataObject
*/
protected $record ;
/**
* @ param DataObject
*/
function __construct ( $record ) {
2011-07-21 20:00:48 +02:00
if ( ! in_array ( 'CMSPreviewable' , class_implements ( $record ))) {
throw new InvalidArgumentException ( 'SilverStripeNavigator: Record of type %s doesn\'t implement CMSPreviewable' , get_class ( $record ));
}
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
*/
function getItems () {
2011-07-21 20:00:48 +02:00
$items = array ();
2011-07-15 16:05:58 +02:00
$classes = ClassInfo :: subclassesFor ( 'SilverStripeNavigatorItem' );
array_shift ( $classes );
// Sort menu items according to priority
$i = 0 ;
foreach ( $classes as $class ) {
// Skip base class
if ( $class == 'SilverStripeNavigatorItem' ) continue ;
$i ++ ;
$item = new $class ( $this -> record );
if ( ! $item -> canView ()) continue ;
// This funny litle formula ensures that the first item added with the same priority will be left-most.
$priority = $item -> getPriority () * 100 - 1 ;
// Ensure that we can have duplicates with the same (default) priority
while ( isset ( $items [ $priority ])) $priority ++ ;
$items [ $priority ] = $item ;
}
ksort ( $items );
2011-07-21 20:00:48 +02:00
2011-10-26 18:39:21 +13:00
return new ArrayList ( $items );
2011-07-15 16:05:58 +02:00
}
/**
* @ return DataObject
*/
function getRecord () {
return $this -> record ;
}
/**
2011-07-21 20:00:48 +02:00
* @ param DataObject $record
2011-07-15 16:05:58 +02:00
* @ return Array template data
*/
static function get_for_record ( $record ) {
$html = '' ;
$message = '' ;
$navigator = new SilverStripeNavigator ( $record );
$items = $navigator -> getItems ();
foreach ( $items as $item ) {
$text = $item -> getHTML ();
if ( $text ) $html .= $text ;
$newMessage = $item -> getMessage ();
if ( $newMessage ) $message = $newMessage ;
}
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 ()} .
2011-07-15 16:05:58 +02:00
*
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem extends ViewableData {
/**
* @ param DataObject
*/
protected $record ;
/**
* @ param DataObject
*/
function __construct ( $record ) {
$this -> record = $record ;
}
/**
* @ return String HTML , mostly a link - but can be more complex as well .
* For example , a " future state " item might show a date selector .
*/
function getHTML () {}
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 .
*
* @ return String
*/
function getLink () {}
2011-07-15 16:05:58 +02:00
/**
* @ return String
*/
function getMessage () {}
/**
* @ return DataObject
*/
function getRecord () {
return $this -> record ;
}
/**
* @ return Int
*/
function getPriority () {
return $this -> stat ( 'priority' );
}
/**
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 ) .
*
* @ return boolean
*/
function isActive () {
return false ;
}
/**
* Filters items based on member permissions or other criteria ,
* such as if a state is generally available for the current record .
2011-07-15 16:05:58 +02:00
*
* @ param Member
* @ return Boolean
*/
function canView ( $member = null ) {
return true ;
}
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_CMSLink extends SilverStripeNavigatorItem {
static $priority = 10 ;
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
}
function getLink () {
2011-07-21 20:00:48 +02:00
return $this -> record -> CMSEditLink ();
}
function isActive () {
return ( Controller :: curr () instanceof CMSMain );
}
function canView () {
// Don't show in CMS
return ! ( Controller :: curr () instanceof CMSMain );
2011-07-15 16:05:58 +02:00
}
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_StageLink extends SilverStripeNavigatorItem {
static $priority = 20 ;
function getHTML () {
2011-07-21 20:00:48 +02:00
$draftPage = $this -> getDraftPage ();
if ( $draftPage ) {
$this -> recordLink = Controller :: join_links ( $draftPage -> AbsoluteLink (), " ?stage=Stage " );
return " <a href= \" $this->recordLink\ " > " . _t('ContentController.DRAFTSITE', 'Draft Site') . " </ a > " ;
2011-07-15 16:05:58 +02:00
}
}
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
}
function getLink () {
2011-07-21 20:00:48 +02:00
return Controller :: join_links ( $this -> record -> AbsoluteLink (), '?stage=Stage' );
}
function canView () {
return ( $this -> record -> hasExtension ( 'Versioned' ) && $this -> getDraftPage ());
}
function isActive () {
return (
Versioned :: current_stage () == 'Stage'
&& ! ( ClassInfo :: exists ( 'SiteTreeFutureState' ) && SiteTreeFutureState :: get_future_datetime ())
);
}
protected function getDraftPage () {
$baseTable = ClassInfo :: baseDataClass ( $this -> record -> class );
return Versioned :: get_one_by_stage (
$baseTable ,
'Stage' ,
sprintf ( '"%s"."ID" = %d' , $baseTable , $this -> record -> ID )
);
2011-07-15 16:05:58 +02:00
}
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_LiveLink extends SilverStripeNavigatorItem {
static $priority = 30 ;
function getHTML () {
2011-07-21 20:00:48 +02:00
$livePage = $this -> getLivePage ();
if ( $livePage ) {
$this -> recordLink = Controller :: join_links ( $livePage -> AbsoluteLink (), " ?stage=Live " );
return " <a href= \" $this->recordLink\ " > " . _t('ContentController.PUBLISHEDSITE', 'Published Site') . " </ a > " ;
2011-07-15 16:05:58 +02:00
}
}
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
}
function getLink () {
2011-07-21 20:00:48 +02:00
return Controller :: join_links ( $this -> record -> AbsoluteLink (), '?stage=Live' );
}
function canView () {
return ( $this -> record -> hasExtension ( 'Versioned' ) && $this -> getLivePage ());
}
function isActive () {
return ( ! Versioned :: current_stage () || Versioned :: current_stage () == 'Live' );
}
protected function getLivePage () {
$baseTable = ClassInfo :: baseDataClass ( $this -> record -> class );
return Versioned :: get_one_by_stage (
$baseTable ,
'Live' ,
sprintf ( '"%s"."ID" = %d' , $baseTable , $this -> record -> ID )
);
2011-07-15 16:05:58 +02:00
}
}
/**
* @ package cms
* @ subpackage content
*/
class SilverStripeNavigatorItem_ArchiveLink extends SilverStripeNavigatorItem {
static $priority = 40 ;
function getHTML () {
2011-07-21 20:00:48 +02:00
$this -> recordLink = $this -> record -> AbsoluteLink ();
return " <a href= \" $this->recordLink ?archiveDate= { $this -> record -> LastEdited } \" target= \" _blank \" > " . _t ( 'ContentController.ARCHIVEDSITE' , 'Archived Site' ) . " </a> " ;
2011-07-15 16:05:58 +02:00
}
function getMessage () {
2011-10-31 12:15:07 +13:00
if ( $date = Versioned :: current_archived_date ()) {
$dateObj = Object :: create ( 'Datetime' );
$dateObj -> setValue ( $date );
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
}
function getLink () {
2011-07-21 20:00:48 +02:00
return $this -> record -> AbsoluteLink () . '?archiveDate=' . $date ;
}
function canView () {
return ( $this -> record -> hasExtension ( 'Versioned' ) && $this -> isArchived ());
}
function isActive () {
return ( Versioned :: current_archived_date ());
}
/**
* Counts as " archived " if the current record is a different version from both live and draft .
*
* @ return boolean
*/
function isArchived () {
if ( ! $this -> record -> hasExtension ( 'Versioned' )) return false ;
$baseTable = ClassInfo :: baseDataClass ( $this -> record -> class );
$currentDraft = Versioned :: get_one_by_stage (
$baseTable ,
'Stage' ,
sprintf ( '"%s"."ID" = %d' , $baseTable , $this -> record -> ID )
);
$currentLive = Versioned :: get_one_by_stage (
$baseTable ,
'Live' ,
sprintf ( '"%s"."ID" = %d' , $baseTable , $this -> record -> ID )
);
return (
( ! $currentDraft || ( $currentDraft && $this -> record -> Version != $currentDraft -> Version ))
&& ( ! $currentLive || ( $currentLive && $this -> record -> Version != $currentLive -> Version ))
);
2011-07-15 16:05:58 +02:00
}
}