2007-07-19 10:40:28 +00:00
< ? php
/**
* Base controller class .
* Controllers are the cornerstone of all site functionality in Sapphire . The { @ link Director }
* selects a controller to pass control to , and then calls { @ link run ()} . This method will execute
* the appropriate action - either by calling the action method , or displaying the action ' s template .
2007-09-14 18:02:03 +00:00
*
2007-07-19 10:40:28 +00:00
* See { @ link getTemplate ()} for information on how the template is chosen .
2008-02-25 02:10:37 +00:00
* @ package sapphire
* @ subpackage control
2007-07-19 10:40:28 +00:00
*/
2008-10-30 22:03:21 +00:00
class Controller extends RequestHandler {
2008-10-05 22:35:14 +00:00
2008-02-25 02:10:37 +00:00
/**
2008-10-05 22:35:14 +00:00
* @ var array $urlParams An array of arguments extracted from the URL
2008-08-09 03:19:54 +00:00
*/
2007-07-19 10:40:28 +00:00
protected $urlParams ;
2008-10-05 22:35:14 +00:00
/**
* @ var array $requestParams Contains all GET and POST parameters
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* passed to the current { @ link SS_HTTPRequest } .
* @ uses SS_HTTPRequest -> requestVars ()
2008-10-05 22:35:14 +00:00
*/
2007-07-19 10:40:28 +00:00
protected $requestParams ;
2008-10-05 22:35:14 +00:00
/**
* @ var string $action The URL part matched on the current controller as
* determined by the " $Action " part of the { @ link $url_handlers } definition .
* Should correlate to a public method on this controller .
* Used in { @ link render ()} and { @ link getViewer ()} to determine
* action - specific templates .
*/
2007-07-19 10:40:28 +00:00
protected $action ;
2007-08-17 05:45:15 +00:00
/**
* The { @ link Session } object for this controller
*/
protected $session ;
2007-07-19 10:40:28 +00:00
2007-08-17 05:45:15 +00:00
/**
* Stack of current controllers .
* Controller :: $controller_stack [ 0 ] is the current controller .
*/
protected static $controller_stack = array ();
2007-07-19 10:40:28 +00:00
protected $basicAuthEnabled = true ;
2007-08-17 03:09:46 +00:00
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* @ var SS_HTTPResponse $response The response object that the controller returns .
2008-10-05 19:45:38 +00:00
* Set in { @ link handleRequest ()} .
2007-08-17 03:09:46 +00:00
*/
protected $response ;
2007-07-19 10:40:28 +00:00
2008-08-09 03:19:54 +00:00
/**
* Default URL handlers - ( Action ) / ( ID ) / ( OtherID )
*/
static $url_handlers = array (
2008-08-09 03:29:30 +00:00
'$Action//$ID/$OtherID' => 'handleAction' ,
2008-08-09 03:19:54 +00:00
);
static $allowed_actions = array (
'handleAction' ,
'handleIndex' ,
);
2008-10-16 11:08:51 +00:00
/**
* Initialisation function that is run before any action on the controller is called .
*
* @ uses BasicAuth :: requireLogin ()
*/
function init () {
2009-11-15 21:24:58 +00:00
if ( $this -> basicAuthEnabled ) BasicAuth :: protect_site_if_necessary ();
2008-10-16 11:08:51 +00:00
2008-11-06 04:51:25 +00:00
// Directly access the session variable just in case the Group or Member tables don't yet exist
2008-11-09 22:11:25 +00:00
if ( Session :: get ( 'loggedInAs' ) && Security :: database_is_ready ()) {
2008-11-06 04:51:25 +00:00
$member = Member :: currentUser ();
2008-11-24 09:31:14 +00:00
if ( $member ) {
2010-10-18 22:53:03 +00:00
if ( ! headers_sent ()) Cookie :: set ( " PastMember " , true , 90 , null , null , false , true );
2009-03-11 21:47:31 +00:00
DB :: query ( " UPDATE \" Member \" SET \" LastVisited \" = " . DB :: getConn () -> now () . " WHERE \" ID \" = $member->ID " , null );
2008-11-24 09:31:14 +00:00
}
2008-10-16 11:08:51 +00:00
}
// This is used to test that subordinate controllers are actually calling parent::init() - a common bug
$this -> baseInitCalled = true ;
}
2008-08-09 03:19:54 +00:00
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* Executes this controller , and return an { @ link SS_HTTPResponse } object with the result .
2009-09-07 00:14:11 +00:00
*
* This method first does a few set - up activities :
* - Push this controller ont to the controller stack -
* see { @ link Controller :: curr ()} for information about this .
* - Call { @ link init ()}
* - Defer to { @ link RequestHandler -> handleRequest ()} to determine which action
* should be executed
2008-10-04 04:45:43 +00:00
*
2009-09-07 00:14:11 +00:00
* Note : $requestParams [ 'executeForm' ] support was removed ,
* make the following change in your URLs :
* " /?executeForm=FooBar " -> " /FooBar "
* Also make sure " FooBar " is in the $allowed_actions of your controller class .
2008-10-04 04:45:43 +00:00
*
2009-09-07 00:14:11 +00:00
* Note : You should rarely need to overload run () -
* this kind of change is only really appropriate for things like nested
* controllers - { @ link ModelAsController } and { @ link RootURLController }
* are two examples here . If you want to make more
* orthodox functionality , it ' s better to overload { @ link init ()} or { @ link index ()} .
*
* Important : If you are going to overload handleRequest ,
* make sure that you start the method with $this -> pushCurrent ()
* and end the method with $this -> popCurrent () .
* Failure to do this will create weird session errors .
*
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* @ param $request The { @ link SS_HTTPRequest } object that is responsible
2009-09-07 00:14:11 +00:00
* for distributing request parsing .
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* @ return SS_HTTPResponse The response that this controller produces ,
2009-09-07 00:14:11 +00:00
* including HTTP headers such as redirection info
2008-08-09 03:19:54 +00:00
*/
2011-05-01 17:33:02 +12:00
function handleRequest ( SS_HTTPRequest $request , DataModel $model ) {
2008-08-11 04:48:45 +00:00
if ( ! $request ) user_error ( " Controller::handleRequest() not passed a request! " , E_USER_ERROR );
2008-08-09 03:19:54 +00:00
$this -> pushCurrent ();
$this -> urlParams = $request -> allParams ();
2008-10-05 19:45:38 +00:00
$this -> request = $request ;
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
$this -> response = new SS_HTTPResponse ();
2011-05-01 17:33:02 +12:00
$this -> setModel ( $model );
2009-05-28 00:22:07 +00:00
$this -> extend ( 'onBeforeInit' );
2008-08-09 03:19:54 +00:00
// Init
$this -> baseInitCalled = false ;
$this -> init ();
if ( ! $this -> baseInitCalled ) user_error ( " init() method on class ' $this->class ' doesn't call Controller::init(). Make sure that you have parent::init() included. " , E_USER_WARNING );
2009-05-28 00:22:07 +00:00
$this -> extend ( 'onAfterInit' );
2008-08-09 03:19:54 +00:00
// If we had a redirection or something, halt processing.
if ( $this -> response -> isFinished ()) {
$this -> popCurrent ();
return $this -> response ;
}
2011-05-01 17:33:02 +12:00
$body = parent :: handleRequest ( $request , $model );
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
if ( $body instanceof SS_HTTPResponse ) {
if ( isset ( $_REQUEST [ 'debug_request' ])) Debug :: message ( " Request handler returned SS_HTTPResponse object to $this->class controller; returning it without modification. " );
2008-08-09 03:19:54 +00:00
$this -> response = $body ;
} else {
2008-08-11 02:57:59 +00:00
if ( is_object ( $body )) {
if ( isset ( $_REQUEST [ 'debug_request' ])) Debug :: message ( " Request handler $body->class object to $this->class controller;, rendering with template returned by $body->class ::getViewer() " );
$body = $body -> getViewer ( $request -> latestParam ( 'Action' )) -> process ( $body );
}
2008-08-09 03:19:54 +00:00
$this -> response -> setBody ( $body );
}
ContentNegotiator :: process ( $this -> response );
HTTP :: add_cache_headers ( $this -> response );
$this -> popCurrent ();
return $this -> response ;
}
/**
* Controller ' s default action handler . It will call the method named in $Action , if that method exists .
* If $Action isn ' t given , it will use " index " as a default .
*/
2009-10-11 00:07:01 +00:00
public function handleAction ( $request ) {
2008-08-09 03:19:54 +00:00
// urlParams, requestParams, and action are set for backward compatability
2008-08-10 22:49:59 +00:00
foreach ( $request -> latestParams () as $k => $v ) {
if ( $v || ! isset ( $this -> urlParams [ $k ])) $this -> urlParams [ $k ] = $v ;
}
2008-08-09 03:19:54 +00:00
$this -> action = str_replace ( " - " , " _ " , $request -> param ( 'Action' ));
$this -> requestParams = $request -> requestVars ();
if ( ! $this -> action ) $this -> action = 'index' ;
2009-10-11 00:07:01 +00:00
2009-10-11 00:07:24 +00:00
if ( ! $this -> hasAction ( $this -> action )) {
$this -> httpError ( 404 , " The action ' $this->action ' does not exist in class $this->class " );
}
2008-08-09 03:19:54 +00:00
// run & init are manually disabled, because they create infinite loops and other dodgy situations
2009-10-11 00:07:01 +00:00
if ( ! $this -> checkAccessAction ( $this -> action ) || in_array ( strtolower ( $this -> action ), array ( 'run' , 'init' ))) {
return $this -> httpError ( 403 , " Action ' $this->action ' isn't allowed on class $this->class " );
}
if ( $this -> hasMethod ( $this -> action )) {
$result = $this -> { $this -> action }( $request );
2008-08-09 03:19:54 +00:00
2009-10-11 00:07:01 +00:00
// If the action returns an array, customise with it before rendering the template.
if ( is_array ( $result )) {
return $this -> getViewer ( $this -> action ) -> process ( $this -> customise ( $result ));
2008-08-09 03:19:54 +00:00
} else {
2009-10-11 00:07:01 +00:00
return $result ;
2008-08-09 03:19:54 +00:00
}
} else {
2009-10-11 00:07:24 +00:00
return $this -> getViewer ( $this -> action ) -> process ( $this );
2008-08-09 03:19:54 +00:00
}
}
2007-07-19 10:40:28 +00:00
function setURLParams ( $urlParams ) {
$this -> urlParams = $urlParams ;
}
2007-08-20 23:22:30 +00:00
/**
2008-02-25 02:10:37 +00:00
* @ return array The parameters extracted from the URL by the { @ link Director } .
2007-08-20 23:22:30 +00:00
*/
2007-07-19 10:40:28 +00:00
function getURLParams () {
return $this -> urlParams ;
}
2007-12-02 21:29:31 +00:00
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* Returns the SS_HTTPResponse object that this controller is building up .
2007-12-02 21:29:31 +00:00
* Can be used to set the status code and headers
*/
function getResponse () {
return $this -> response ;
}
2008-10-05 19:45:38 +00:00
2008-03-02 23:24:10 +00:00
protected $baseInitCalled = false ;
2008-03-03 08:48:52 +00:00
/**
* Return the object that is going to own a form that ' s being processed , and handle its execution .
* Note that the result needn ' t be an actual controller object .
*/
function getFormOwner () {
// Get the appropraite ocntroller: sometimes we want to get a form from another controller
if ( isset ( $this -> requestParams [ 'formController' ])) {
$formController = Director :: getControllerForURL ( $this -> requestParams [ 'formController' ]);
while ( is_a ( $formController , 'NestedController' )) {
$formController = $formController -> getNestedController ();
}
return $formController ;
} else {
return $this ;
}
}
2007-07-19 10:40:28 +00:00
2008-03-02 23:24:10 +00:00
/**
* This is the default action handler used if a method doesn ' t exist .
* It will process the controller object with the template returned by { @ link getViewer ()}
*/
2007-07-19 10:40:28 +00:00
function defaultAction ( $action ) {
return $this -> getViewer ( $action ) -> process ( $this );
}
2007-09-14 18:02:03 +00:00
2008-03-02 23:24:10 +00:00
/**
* Returns the action that is being executed on this controller .
*/
2007-07-19 10:40:28 +00:00
function getAction () {
return $this -> action ;
}
2007-09-14 18:02:03 +00:00
2007-07-19 10:40:28 +00:00
/**
* Return an SSViewer object to process the data
2007-08-20 23:22:30 +00:00
* @ return SSViewer The viewer identified being the default handler for this Controller / Action combination
2007-07-19 10:40:28 +00:00
*/
function getViewer ( $action ) {
// Hard-coded templates
if ( $this -> templates [ $action ]) {
$templates = $this -> templates [ $action ];
} else if ( $this -> templates [ 'index' ]) {
$templates = $this -> templates [ 'index' ];
} else if ( $this -> template ) {
$templates = $this -> template ;
} else {
2008-10-06 19:25:45 +00:00
// Add action-specific templates for inheritance chain
2008-08-14 03:35:13 +00:00
$parentClass = $this -> class ;
if ( $action && $action != 'index' ) {
$parentClass = $this -> class ;
while ( $parentClass != " Controller " ) {
$templates [] = strtok ( $parentClass , '_' ) . '_' . $action ;
$parentClass = get_parent_class ( $parentClass );
}
}
2008-10-06 19:25:45 +00:00
// Add controller templates for inheritance chain
2007-07-19 10:40:28 +00:00
$parentClass = $this -> class ;
while ( $parentClass != " Controller " ) {
2008-08-14 03:35:13 +00:00
$templates [] = strtok ( $parentClass , '_' );
2007-07-19 10:40:28 +00:00
$parentClass = get_parent_class ( $parentClass );
}
2008-08-14 03:35:13 +00:00
2008-10-06 19:25:45 +00:00
// remove duplicates
2007-07-19 10:40:28 +00:00
$templates = array_unique ( $templates );
}
return new SSViewer ( $templates );
}
2009-10-11 00:07:23 +00:00
public function hasAction ( $action ) {
return parent :: hasAction ( $action ) || $this -> hasActionTemplate ( $action );
}
2009-10-11 00:07:01 +00:00
/**
* Returns TRUE if this controller has a template that is specifically designed to handle a specific action .
*
* @ param string $action
* @ return bool
*/
2009-10-11 00:07:23 +00:00
public function hasActionTemplate ( $action ) {
2009-10-11 00:07:01 +00:00
if ( isset ( $this -> templates [ $action ])) return true ;
$parentClass = $this -> class ;
$templates = array ();
while ( $parentClass != 'Controller' ) {
$templates [] = strtok ( $parentClass , '_' ) . '_' . $action ;
$parentClass = get_parent_class ( $parentClass );
}
return SSViewer :: hasTemplate ( $templates );
}
2008-10-05 19:27:21 +00:00
/**
* Render the current controller with the templates determined
* by { @ link getViewer ()} .
*
* @ param array $params Key - value array for custom template variables ( Optional )
* @ return string Parsed template content
*/
function render ( $params = null ) {
$template = $this -> getViewer ( $this -> getAction ());
// if the object is already customised (e.g. through Controller->run()), use it
$obj = ( $this -> customisedObj ) ? $this -> customisedObj : $this ;
if ( $params ) $obj = $this -> customise ( $params );
2009-05-28 00:22:07 +00:00
2008-10-05 19:27:21 +00:00
return $template -> process ( $obj );
}
2007-07-19 10:40:28 +00:00
/**
2009-11-15 21:24:58 +00:00
* Call this to disable site - wide basic authentication for a specific contoller .
* This must be called before Controller :: init () . That is , you must call it in your controller ' s
* init method before it calls parent :: init () .
2007-07-19 10:40:28 +00:00
*/
function disableBasicAuth () {
$this -> basicAuthEnabled = false ;
}
2007-08-17 03:09:46 +00:00
/**
* Returns the current controller
2007-08-20 23:22:30 +00:00
* @ returns Controller
2007-08-17 03:09:46 +00:00
*/
public static function curr () {
2007-08-17 05:45:15 +00:00
if ( Controller :: $controller_stack ) {
return Controller :: $controller_stack [ 0 ];
} else {
user_error ( " No current controller available " , E_USER_WARNING );
}
2007-08-17 03:09:46 +00:00
}
2007-08-20 22:39:44 +00:00
/**
* Tests whether we have a currently active controller or not
2007-08-20 23:22:30 +00:00
* @ return boolean True if there is at least 1 controller in the stack .
2007-08-20 22:39:44 +00:00
*/
public static function has_curr () {
return Controller :: $controller_stack ? true : false ;
}
2007-07-19 10:40:28 +00:00
/**
* Returns true if the member is allowed to do the given action .
* @ param perm The permission to be checked , such as 'View' .
* @ param member The member whose permissions need checking . Defaults to the currently logged
* in user .
2007-08-20 23:22:30 +00:00
* @ return boolean
2007-07-19 10:40:28 +00:00
*/
function can ( $perm , $member = null ) {
if ( ! $member ) $member = Member :: currentUser ();
2011-10-29 12:09:22 +13:00
if ( is_array ( $perm )) {
$perm = array_map ( array ( $this , 'can' ), $perm , array_fill ( 0 , count ( $perm ), $member ));
return min ( $perm );
}
2007-07-19 10:40:28 +00:00
if ( $this -> hasMethod ( $methodName = 'can' . $perm )) {
return $this -> $methodName ( $member );
} else {
return true ;
}
}
2007-09-14 18:02:03 +00:00
2007-07-19 10:40:28 +00:00
//-----------------------------------------------------------------------------------
2007-09-14 18:02:03 +00:00
2007-07-19 10:40:28 +00:00
/**
* returns a date object for use within a template
* Usage : $Now . Year - Returns 2006
2007-08-20 23:22:30 +00:00
* @ return Date The current date
2007-07-19 10:40:28 +00:00
*/
function Now () {
$d = new Date ( null );
2008-10-16 08:40:08 +00:00
$d -> setValue ( date ( " Y-m-d h:i:s " ));
2007-07-19 10:40:28 +00:00
return $d ;
}
2007-09-14 18:02:03 +00:00
2007-07-19 10:40:28 +00:00
/**
* Returns the currently logged in user
*/
function CurrentMember () {
return Member :: currentUser ();
}
2007-09-14 18:02:03 +00:00
2007-07-19 10:40:28 +00:00
/**
* Return true if the visitor has signed up for a login account before
2007-08-20 23:22:30 +00:00
* @ return boolean
2007-07-19 10:40:28 +00:00
*/
function PastMember () {
return Cookie :: get ( " PastMember " ) ? true : false ;
}
2007-08-17 05:45:15 +00:00
/**
* Pushes this controller onto the stack of current controllers .
* This means that any redirection , session setting , or other things that rely on Controller :: curr () will now write to this
* controller object .
*/
function pushCurrent () {
array_unshift ( self :: $controller_stack , $this );
// Create a new session object
2007-08-17 23:11:43 +00:00
if ( ! $this -> session ) {
2007-10-02 04:33:18 +00:00
if ( isset ( self :: $controller_stack [ 1 ])) {
$this -> session = self :: $controller_stack [ 1 ] -> getSession ();
} else {
$this -> session = new Session ( null );
}
2007-08-17 23:11:43 +00:00
}
2007-08-17 05:45:15 +00:00
}
/**
* Pop this controller off the top of the stack .
*/
function popCurrent () {
2007-08-17 23:31:51 +00:00
if ( $this === self :: $controller_stack [ 0 ]) {
2007-08-17 05:45:15 +00:00
array_shift ( self :: $controller_stack );
} else {
2007-08-17 23:14:30 +00:00
user_error ( " popCurrent called on $this->class controller, but it wasn't at the top of the stack " , E_USER_WARNING );
2007-08-17 05:45:15 +00:00
}
}
2007-08-17 03:09:46 +00:00
/**
2008-03-02 23:24:10 +00:00
* Redirct to the given URL .
* It is generally recommended to call Director :: redirect () rather than calling this function directly .
2007-08-17 03:09:46 +00:00
*/
2008-04-22 01:45:55 +00:00
function redirect ( $url , $code = 302 ) {
2010-10-13 04:59:22 +00:00
if ( ! $this -> response ) $this -> response = new SS_HTTPResponse ();
2008-02-25 02:10:37 +00:00
if ( $this -> response -> getHeader ( 'Location' )) {
2008-04-26 06:53:13 +00:00
user_error ( " Already directed to " . $this -> response -> getHeader ( 'Location' ) . " ; now trying to direct to $url " , E_USER_WARNING );
return ;
2008-02-25 02:10:37 +00:00
}
2007-08-17 03:09:46 +00:00
// Attach site-root to relative links, if they have a slash in them
2007-08-28 02:49:31 +00:00
if ( $url == " " || $url [ 0 ] == '?' || ( substr ( $url , 0 , 4 ) != " http " && $url [ 0 ] != " / " && strpos ( $url , '/' ) !== false )){
2007-08-17 03:09:46 +00:00
$url = Director :: baseURL () . $url ;
}
2008-04-22 01:45:55 +00:00
$this -> response -> redirect ( $url , $code );
2007-08-17 03:09:46 +00:00
}
2007-08-17 05:45:15 +00:00
2008-10-14 19:59:01 +00:00
/**
2011-10-10 10:10:20 +02:00
* Redirect back . Uses either the HTTP_REFERER or a manually set request - variable called " BackURL " .
2008-10-14 19:59:01 +00:00
* This variable is needed in scenarios where not HTTP - Referer is sent (
* e . g when calling a page by location . href in IE ) .
* If none of the two variables is available , it will redirect to the base
* URL ( see { @ link Director :: baseURL ()}) .
* @ uses redirect ()
*/
function redirectBack () {
2010-10-13 04:59:22 +00:00
$url = null ;
// In edge-cases, this will be called outside of a handleRequest() context; in that case,
// redirect to the homepage - don't break into the global state at this stage because we'll
// be calling from a test context or something else where the global state is inappropraite
if ( $this -> request ) {
2011-10-10 10:10:20 +02:00
if ( $this -> request -> requestVar ( 'BackURL' )) {
$url = $this -> request -> requestVar ( 'BackURL' );
2010-10-13 04:59:22 +00:00
} else if ( $this -> request -> getHeader ( 'Referer' )) {
$url = $this -> request -> getHeader ( 'Referer' );
}
2008-10-14 19:59:01 +00:00
}
2010-10-13 04:59:22 +00:00
if ( ! $url ) $url = Director :: baseURL ();
2008-10-14 19:59:01 +00:00
2009-03-17 22:24:50 +00:00
// absolute redirection URLs not located on this site may cause phishing
if ( Director :: is_site_url ( $url )) {
return $this -> redirect ( $url );
} else {
return false ;
}
2008-10-14 19:59:01 +00:00
}
2007-08-31 00:26:41 +00:00
/**
* Tests whether a redirection has been requested .
* @ return string If redirect () has been called , it will return the URL redirected to . Otherwise , it will return null ;
*/
function redirectedTo () {
2010-10-13 04:59:22 +00:00
return $this -> response && $this -> response -> getHeader ( 'Location' );
2007-08-31 00:26:41 +00:00
}
2007-08-17 05:45:15 +00:00
/**
* Get the Session object representing this Controller ' s session
2007-08-20 23:22:30 +00:00
* @ return Session
2007-08-17 05:45:15 +00:00
*/
function getSession () {
return $this -> session ;
}
/**
* Set the Session object .
*/
function setSession ( Session $session ) {
$this -> session = $session ;
}
2007-08-20 22:39:44 +00:00
/**
* Returns true if this controller is processing an ajax request
2007-08-20 23:22:30 +00:00
* @ return boolean True if this controller is processing an ajax request
2007-08-20 22:39:44 +00:00
*/
function isAjax () {
return (
2009-04-28 23:40:35 +00:00
isset ( $this -> requestParams [ 'ajax' ]) || isset ( $_REQUEST [ 'ajax' ]) ||
2007-08-20 22:39:44 +00:00
( isset ( $_SERVER [ 'HTTP_X_REQUESTED_WITH' ]) && $_SERVER [ 'HTTP_X_REQUESTED_WITH' ] == " XMLHttpRequest " )
);
}
2008-08-09 03:54:55 +00:00
/**
2010-04-13 03:20:59 +00:00
* Joins two or more link segments together , putting a slash between them if necessary .
* Use this for building the results of { @ link Link ()} methods .
* If either of the links have query strings ,
* then they will be combined and put at the end of the resulting url .
*
* Caution : All parameters are expected to be URI - encoded already .
*
* @ param String
* @ return String
2008-08-09 03:54:55 +00:00
*/
static function join_links () {
$args = func_get_args ();
2008-08-19 10:06:43 +00:00
$result = " " ;
$querystrings = array ();
2010-10-15 00:29:29 +00:00
$fragmentIdentifier = null ;
2008-08-09 03:54:55 +00:00
foreach ( $args as $arg ) {
2010-10-15 00:29:29 +00:00
// Find fragment identifier - keep the last one
if ( strpos ( $arg , '#' ) !== false ) {
list ( $arg , $fragmentIdentifier ) = explode ( '#' , $arg , 2 );
}
// Find querystrings
2008-08-19 10:06:43 +00:00
if ( strpos ( $arg , '?' ) !== false ) {
list ( $arg , $suffix ) = explode ( '?' , $arg , 2 );
$querystrings [] = $suffix ;
}
2008-08-28 04:25:13 +00:00
if ( $arg ) {
if ( $result && substr ( $result , - 1 ) != '/' && $arg [ 0 ] != '/' ) $result .= " / $arg " ;
2009-10-11 00:07:15 +00:00
else $result .= ( substr ( $result , - 1 ) == '/' && $arg [ 0 ] == '/' ) ? ltrim ( $arg , '/' ) : $arg ;
2008-08-28 04:25:13 +00:00
}
2008-08-09 03:54:55 +00:00
}
2008-08-19 10:06:43 +00:00
if ( $querystrings ) $result .= '?' . implode ( '&' , $querystrings );
2010-10-15 00:29:29 +00:00
if ( $fragmentIdentifier ) $result .= " # $fragmentIdentifier " ;
2008-08-19 10:06:43 +00:00
2008-08-09 03:54:55 +00:00
return $result ;
}
2007-07-19 10:40:28 +00:00
}
2009-04-29 01:20:24 +00:00
?>