2011-03-22 22:26:53 +13:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* ModelAsController deals with mapping the initial request to the first {@link SiteTree}/{@link ContentController}
|
|
|
|
* pair, which are then used to handle the request.
|
|
|
|
*
|
|
|
|
* @package cms
|
|
|
|
* @subpackage control
|
|
|
|
*/
|
|
|
|
class ModelAsController extends Controller implements NestedController {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the appropriate {@link ContentController} for handling a {@link SiteTree} object, link it to the object and
|
|
|
|
* return it.
|
|
|
|
*
|
|
|
|
* @param SiteTree $sitetree
|
|
|
|
* @param string $action
|
|
|
|
* @return ContentController
|
|
|
|
*/
|
2012-09-19 12:07:46 +02:00
|
|
|
static public function controller_for(SiteTree $sitetree, $action = null) {
|
2011-03-22 22:26:53 +13:00
|
|
|
if($sitetree->class == 'SiteTree') $controller = "ContentController";
|
|
|
|
else $controller = "{$sitetree->class}_Controller";
|
|
|
|
|
|
|
|
if($action && class_exists($controller . '_' . ucfirst($action))) {
|
|
|
|
$controller = $controller . '_' . ucfirst($action);
|
|
|
|
}
|
|
|
|
|
2012-05-30 15:09:25 +10:00
|
|
|
return class_exists($controller) ? Injector::inst()->create($controller, $sitetree) : $sitetree;
|
2011-03-22 22:26:53 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function init() {
|
|
|
|
singleton('SiteTree')->extend('modelascontrollerInit', $this);
|
|
|
|
parent::init();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @uses ModelAsController::getNestedController()
|
|
|
|
* @return SS_HTTPResponse
|
|
|
|
*/
|
2011-05-01 17:42:27 +12:00
|
|
|
public function handleRequest(SS_HTTPRequest $request, DataModel $model) {
|
2011-03-22 22:26:53 +13:00
|
|
|
$this->request = $request;
|
2012-05-01 14:46:16 +12:00
|
|
|
$this->setDataModel($model);
|
2011-03-22 22:26:53 +13:00
|
|
|
|
|
|
|
$this->pushCurrent();
|
|
|
|
|
|
|
|
// Create a response just in case init() decides to redirect
|
|
|
|
$this->response = new SS_HTTPResponse();
|
|
|
|
|
|
|
|
$this->init();
|
|
|
|
|
|
|
|
// If we had a redirection or something, halt processing.
|
|
|
|
if($this->response->isFinished()) {
|
|
|
|
$this->popCurrent();
|
|
|
|
return $this->response;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the database has not yet been created, redirect to the build page.
|
|
|
|
if(!DB::isActive() || !ClassInfo::hasTable('SiteTree')) {
|
|
|
|
$this->response->redirect(Director::absoluteBaseURL() . 'dev/build?returnURL=' . (isset($_GET['url']) ? urlencode($_GET['url']) : null));
|
|
|
|
$this->popCurrent();
|
|
|
|
|
|
|
|
return $this->response;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
$result = $this->getNestedController();
|
|
|
|
|
|
|
|
if($result instanceof RequestHandler) {
|
2011-05-01 17:42:27 +12:00
|
|
|
$result = $result->handleRequest($this->request, $model);
|
2011-03-22 22:26:53 +13:00
|
|
|
} else if(!($result instanceof SS_HTTPResponse)) {
|
|
|
|
user_error("ModelAsController::getNestedController() returned bad object type '" .
|
|
|
|
get_class($result)."'", E_USER_WARNING);
|
|
|
|
}
|
|
|
|
} catch(SS_HTTPResponse_Exception $responseException) {
|
|
|
|
$result = $responseException->getResponse();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->popCurrent();
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return ContentController
|
|
|
|
*/
|
|
|
|
public function getNestedController() {
|
|
|
|
$request = $this->request;
|
|
|
|
|
|
|
|
if(!$URLSegment = $request->param('URLSegment')) {
|
|
|
|
throw new Exception('ModelAsController->getNestedController(): was not passed a URLSegment value.');
|
|
|
|
}
|
2012-05-08 22:10:44 +02:00
|
|
|
|
2011-03-22 22:26:53 +13:00
|
|
|
// Find page by link, regardless of current locale settings
|
2011-03-22 21:30:10 +13:00
|
|
|
if(class_exists('Translatable')) Translatable::disable_locale_filter();
|
2011-03-22 22:26:53 +13:00
|
|
|
$sitetree = DataObject::get_one(
|
|
|
|
'SiteTree',
|
|
|
|
sprintf(
|
|
|
|
'"URLSegment" = \'%s\' %s',
|
2012-05-08 22:10:44 +02:00
|
|
|
Convert::raw2sql(rawurlencode($URLSegment)),
|
2013-03-18 11:47:15 +01:00
|
|
|
(SiteTree::config()->nested_urls ? 'AND "ParentID" = 0' : null)
|
2011-03-22 22:26:53 +13:00
|
|
|
)
|
|
|
|
);
|
2011-03-22 21:30:10 +13:00
|
|
|
if(class_exists('Translatable')) Translatable::enable_locale_filter();
|
2011-03-22 22:26:53 +13:00
|
|
|
|
|
|
|
if(!$sitetree) {
|
|
|
|
// If a root page has been renamed, redirect to the new location.
|
|
|
|
// See ContentController->handleRequest() for similiar logic.
|
|
|
|
$redirect = self::find_old_page($URLSegment);
|
2012-04-11 14:51:33 +12:00
|
|
|
if($redirect) {
|
2011-03-22 22:26:53 +13:00
|
|
|
$params = $request->getVars();
|
|
|
|
if(isset($params['url'])) unset($params['url']);
|
|
|
|
$this->response = new SS_HTTPResponse();
|
|
|
|
$this->response->redirect(
|
|
|
|
Controller::join_links(
|
|
|
|
$redirect->Link(
|
|
|
|
Controller::join_links(
|
|
|
|
$request->param('Action'),
|
|
|
|
$request->param('ID'),
|
|
|
|
$request->param('OtherID')
|
|
|
|
)
|
|
|
|
),
|
|
|
|
// Needs to be in separate join links to avoid urlencoding
|
|
|
|
($params) ? '?' . http_build_query($params) : null
|
|
|
|
),
|
|
|
|
301
|
|
|
|
);
|
|
|
|
|
|
|
|
return $this->response;
|
|
|
|
}
|
|
|
|
|
2012-09-27 12:30:44 +12:00
|
|
|
$response = ErrorPage::response_for(404);
|
|
|
|
$this->httpError(404, $response ? $response : 'The requested page could not be found.');
|
2011-03-22 22:26:53 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
// Enforce current locale setting to the loaded SiteTree object
|
2011-03-22 21:30:10 +13:00
|
|
|
if(class_exists('Translatable') && $sitetree->Locale) Translatable::set_current_locale($sitetree->Locale);
|
2011-03-22 22:26:53 +13:00
|
|
|
|
|
|
|
if(isset($_REQUEST['debug'])) {
|
|
|
|
Debug::message("Using record #$sitetree->ID of type $sitetree->class with link {$sitetree->Link()}");
|
|
|
|
}
|
|
|
|
|
|
|
|
return self::controller_for($sitetree, $this->request->param('Action'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $URLSegment A subset of the url. i.e in /home/contact/ home and contact are URLSegment.
|
|
|
|
* @param int $parentID The ID of the parent of the page the URLSegment belongs to.
|
|
|
|
* @return SiteTree
|
|
|
|
*/
|
2012-09-19 12:07:46 +02:00
|
|
|
static public function find_old_page($URLSegment,$parentID = 0, $ignoreNestedURLs = false) {
|
2012-05-08 22:10:44 +02:00
|
|
|
$URLSegment = Convert::raw2sql(rawurlencode($URLSegment));
|
2011-03-22 22:26:53 +13:00
|
|
|
|
2013-03-18 11:47:15 +01:00
|
|
|
$useParentIDFilter = SiteTree::config()->nested_urls && $parentID;
|
2011-03-22 22:26:53 +13:00
|
|
|
|
|
|
|
// First look for a non-nested page that has a unique URLSegment and can be redirected to.
|
2013-03-18 11:47:15 +01:00
|
|
|
if(SiteTree::config()->nested_urls) {
|
2011-03-22 22:26:53 +13:00
|
|
|
$pages = DataObject::get(
|
|
|
|
'SiteTree',
|
|
|
|
"\"URLSegment\" = '$URLSegment'" . ($useParentIDFilter ? ' AND "ParentID" = ' . (int)$parentID : '')
|
|
|
|
);
|
2012-07-18 16:47:17 +12:00
|
|
|
|
|
|
|
if($pages && $pages->Count() == 1 && ($page = $pages->First())) {
|
|
|
|
$parent = $page->ParentID ? $page->Parent() : $page;
|
|
|
|
if($parent->isPublished()) return $page;
|
|
|
|
}
|
2011-03-22 22:26:53 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get an old version of a page that has been renamed.
|
|
|
|
$query = new SQLQuery (
|
|
|
|
'"RecordID"',
|
|
|
|
'"SiteTree_versions"',
|
|
|
|
"\"URLSegment\" = '$URLSegment' AND \"WasPublished\" = 1" . ($useParentIDFilter ? ' AND "ParentID" = ' . (int)$parentID : ''),
|
|
|
|
'"LastEdited" DESC',
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
1
|
|
|
|
);
|
|
|
|
$record = $query->execute()->first();
|
|
|
|
|
|
|
|
if($record && ($oldPage = DataObject::get_by_id('SiteTree', $record['RecordID']))) {
|
2012-07-18 16:47:17 +12:00
|
|
|
$oldParent = $oldPage->ParentID ? $oldPage->Parent() : $oldPage;
|
2011-04-15 19:37:15 +10:00
|
|
|
// Run the page through an extra filter to ensure that all extensions are applied.
|
2012-07-18 16:47:17 +12:00
|
|
|
if(SiteTree::get_by_link($oldPage->RelativeLink()) && $oldParent->isPublished()) return $oldPage;
|
2011-03-22 22:26:53 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-11 14:51:33 +12:00
|
|
|
}
|