2011-03-18 16:01:06 +13:00
|
|
|
<?php
|
2016-06-16 16:57:19 +12:00
|
|
|
|
2016-07-22 11:32:32 +12:00
|
|
|
namespace SilverStripe\CMS\Model;
|
|
|
|
|
2017-06-21 16:29:40 +12:00
|
|
|
use Page;
|
2022-04-12 09:45:05 +12:00
|
|
|
use SilverStripe\Assets\File;
|
2017-05-16 16:17:36 +12:00
|
|
|
use SilverStripe\Forms\FieldList;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Forms\HeaderField;
|
|
|
|
use SilverStripe\Forms\OptionsetField;
|
2024-10-02 18:39:01 +13:00
|
|
|
use SilverStripe\Forms\UrlField;
|
2017-06-21 16:29:40 +12:00
|
|
|
use SilverStripe\Versioned\Versioned;
|
2016-06-16 16:57:19 +12:00
|
|
|
|
2011-03-18 16:01:06 +13:00
|
|
|
/**
|
|
|
|
* A redirector page redirects when the page is visited.
|
|
|
|
*
|
2022-04-12 09:45:05 +12:00
|
|
|
* @property string $RedirectionType Either 'Internal','External' or 'File'
|
2016-08-10 16:08:39 +12:00
|
|
|
* @property string $ExternalURL URL to redirect to if $RedirectionType is 'External'
|
|
|
|
* @property int $LinkToID
|
2023-12-15 11:58:32 +13:00
|
|
|
* @method SiteTree LinkTo()
|
|
|
|
* @method File LinkToFile()
|
2011-03-18 16:01:06 +13:00
|
|
|
*/
|
2017-01-26 09:59:25 +13:00
|
|
|
class RedirectorPage extends Page
|
|
|
|
{
|
2022-04-12 09:45:05 +12:00
|
|
|
private static $description = 'Redirects requests to another location';
|
2017-01-26 09:59:25 +13:00
|
|
|
|
2019-01-14 13:16:30 +13:00
|
|
|
private static $icon_class = 'font-icon-p-redirect';
|
|
|
|
|
2019-10-29 17:43:17 +00:00
|
|
|
private static $show_stage_link = false;
|
|
|
|
|
|
|
|
private static $show_live_link = false;
|
|
|
|
|
2020-01-24 22:07:29 +13:00
|
|
|
private static $db = [
|
2022-04-12 09:45:05 +12:00
|
|
|
"RedirectionType" => "Enum('Internal,External,File','Internal')",
|
2017-01-26 09:59:25 +13:00
|
|
|
"ExternalURL" => "Varchar(2083)" // 2083 is the maximum length of a URL in Internet Explorer.
|
2020-01-24 22:07:29 +13:00
|
|
|
];
|
2017-01-26 09:59:25 +13:00
|
|
|
|
2020-01-24 22:07:29 +13:00
|
|
|
private static $defaults = [
|
2017-01-26 09:59:25 +13:00
|
|
|
"RedirectionType" => "Internal"
|
2020-01-24 22:07:29 +13:00
|
|
|
];
|
2017-01-26 09:59:25 +13:00
|
|
|
|
2020-01-24 22:07:29 +13:00
|
|
|
private static $has_one = [
|
2018-03-14 16:34:46 +13:00
|
|
|
"LinkTo" => SiteTree::class,
|
2022-04-12 09:45:05 +12:00
|
|
|
"LinkToFile" => File::class,
|
2020-01-24 22:07:29 +13:00
|
|
|
];
|
2017-01-26 09:59:25 +13:00
|
|
|
|
2024-08-15 13:14:57 +12:00
|
|
|
private static array $scaffold_cms_fields_settings = [
|
|
|
|
'ignoreFields' => [
|
|
|
|
'RedirectionType',
|
|
|
|
'Content',
|
|
|
|
],
|
2024-10-02 18:39:01 +13:00
|
|
|
'fieldClasses' => [
|
|
|
|
'ExternalURL' => UrlField::class,
|
|
|
|
],
|
2024-08-15 13:14:57 +12:00
|
|
|
];
|
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
private static $table_name = 'RedirectorPage';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns this page if the redirect is external, otherwise
|
2022-04-12 09:45:05 +12:00
|
|
|
* returns the target page or file.
|
|
|
|
*
|
|
|
|
* @return SiteTree|File
|
2017-01-26 09:59:25 +13:00
|
|
|
*/
|
|
|
|
public function ContentSource()
|
|
|
|
{
|
|
|
|
if ($this->RedirectionType == 'Internal') {
|
|
|
|
return $this->LinkTo();
|
2022-04-12 09:45:05 +12:00
|
|
|
} elseif ($this->RedirectionType == 'File') {
|
|
|
|
return $this->LinkToFile();
|
2017-01-26 09:59:25 +13:00
|
|
|
} else {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the the link that should be used for this redirector page, in navigation, etc.
|
|
|
|
* If the redirectorpage has been appropriately configured, then it will return the redirection
|
|
|
|
* destination, to prevent unnecessary 30x redirections. However, if it's misconfigured, then
|
|
|
|
* it will return a link to itself, which will then display an error message.
|
|
|
|
*
|
|
|
|
* @param string $action
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function Link($action = null)
|
|
|
|
{
|
|
|
|
$link = $this->redirectionLink();
|
2022-04-12 09:45:05 +12:00
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
if ($link) {
|
|
|
|
return $link;
|
|
|
|
} else {
|
|
|
|
return $this->regularLink($action);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the normal link directly to this page. Once you visit this link, a 30x redirection
|
|
|
|
* will take you to your final destination.
|
|
|
|
*
|
|
|
|
* @param string $action
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function regularLink($action = null)
|
|
|
|
{
|
|
|
|
return parent::Link($action);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the link that we should redirect to.
|
|
|
|
* Only return a value if there is a legal redirection destination.
|
2020-01-24 22:07:29 +13:00
|
|
|
*
|
|
|
|
* @return string
|
2017-01-26 09:59:25 +13:00
|
|
|
*/
|
|
|
|
public function redirectionLink()
|
|
|
|
{
|
|
|
|
// Check external redirect
|
|
|
|
if ($this->RedirectionType == 'External') {
|
2020-01-24 22:07:29 +13:00
|
|
|
$result = $this->ExternalURL ?: null;
|
|
|
|
|
|
|
|
$this->extend('updateRedirectionLink', $result);
|
|
|
|
|
2022-04-12 09:45:05 +12:00
|
|
|
return $result;
|
|
|
|
} elseif ($this->RedirectionType == 'File') {
|
|
|
|
$result = $this->LinkToFile()->exists() ? $this->LinkToFile()->getURL() : null;
|
|
|
|
|
|
|
|
$this->extend('updateRedirectionLink', $result);
|
|
|
|
|
2020-01-24 22:07:29 +13:00
|
|
|
return $result;
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check internal redirect
|
|
|
|
$linkTo = $this->LinkToID ? SiteTree::get()->byID($this->LinkToID) : null;
|
|
|
|
|
2020-01-24 22:07:29 +13:00
|
|
|
if (empty($linkTo)) {
|
|
|
|
$link = null;
|
|
|
|
} elseif ($this->ID == $linkTo->ID) {
|
|
|
|
// We shouldn't point to ourselves
|
|
|
|
$link = null;
|
|
|
|
} elseif ($linkTo instanceof RedirectorPage) {
|
|
|
|
// If we're linking to another redirectorpage then just return the
|
|
|
|
// URLSegment, to prevent a cycle of redirector
|
|
|
|
// pages from causing an infinite loop. Instead, they will cause
|
|
|
|
// a 30x redirection loop in the browser, but
|
|
|
|
// this can be handled sufficiently gracefully by the browser.
|
|
|
|
$link = $linkTo->regularLink();
|
|
|
|
} else {
|
|
|
|
// For all other pages, just return the link of the page.
|
|
|
|
$link = $linkTo->Link();
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
|
2020-01-24 22:07:29 +13:00
|
|
|
$this->extend('updateRedirectionLink', $link);
|
2017-01-26 09:59:25 +13:00
|
|
|
|
2020-01-24 22:07:29 +13:00
|
|
|
return $link;
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function syncLinkTracking()
|
|
|
|
{
|
|
|
|
if ($this->RedirectionType == 'Internal') {
|
|
|
|
if ($this->LinkToID) {
|
2017-06-21 16:29:40 +12:00
|
|
|
$this->HasBrokenLink = Versioned::get_by_stage(SiteTree::class, Versioned::DRAFT)
|
|
|
|
->filter('ID', $this->LinkToID)
|
|
|
|
->count() === 0;
|
2017-01-26 09:59:25 +13:00
|
|
|
} else {
|
2022-04-12 09:45:05 +12:00
|
|
|
$this->HasBrokenLink = true;
|
|
|
|
}
|
|
|
|
} elseif ($this->RedirectionType == 'File') {
|
|
|
|
if ($this->LinkToFileID) {
|
|
|
|
$this->HasBrokenLink = Versioned::get_by_stage(File::class, Versioned::DRAFT)
|
|
|
|
->filter('ID', $this->LinkToFileID)
|
|
|
|
->count() === 0;
|
|
|
|
} else {
|
2017-01-26 09:59:25 +13:00
|
|
|
$this->HasBrokenLink = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$this->HasBrokenLink = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getCMSFields()
|
|
|
|
{
|
2017-05-16 16:17:36 +12:00
|
|
|
$this->beforeUpdateCMSFields(function (FieldList $fields) {
|
|
|
|
// Remove all metadata fields, does not apply for redirector pages
|
|
|
|
$fields->removeByName('Metadata');
|
2024-10-02 18:39:01 +13:00
|
|
|
$fields->dataFieldByName('ExternalURL')?->setAllowRelativeProtocol(true);
|
2017-05-16 16:17:36 +12:00
|
|
|
|
|
|
|
$fields->addFieldsToTab(
|
|
|
|
'Root.Main',
|
2020-04-19 16:18:01 +12:00
|
|
|
[
|
2024-08-15 13:14:57 +12:00
|
|
|
HeaderField::create(
|
|
|
|
'RedirectorDescHeader',
|
|
|
|
_t(__CLASS__.'.HEADER', "This page will redirect users to another page")
|
|
|
|
),
|
|
|
|
OptionsetField::create(
|
|
|
|
'RedirectionType',
|
|
|
|
$this->fieldLabel('RedirectionType'),
|
2020-04-19 16:18:01 +12:00
|
|
|
[
|
2017-05-16 16:17:36 +12:00
|
|
|
"Internal" => _t(__CLASS__.'.REDIRECTTOPAGE', "A page on your website"),
|
|
|
|
"External" => _t(__CLASS__.'.REDIRECTTOEXTERNAL', "Another website"),
|
2022-04-12 09:45:05 +12:00
|
|
|
"File" => _t(__CLASS__.'.REDIRECTTOFILE', "A file on your website"),
|
2020-04-19 16:18:01 +12:00
|
|
|
],
|
2017-05-16 16:17:36 +12:00
|
|
|
"Internal"
|
2017-01-26 09:59:25 +13:00
|
|
|
),
|
2024-08-15 13:14:57 +12:00
|
|
|
],
|
|
|
|
'ExternalURL'
|
2017-05-16 16:17:36 +12:00
|
|
|
);
|
|
|
|
});
|
2020-01-24 22:07:29 +13:00
|
|
|
|
2017-05-16 16:17:36 +12:00
|
|
|
return parent::getCMSFields();
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
// Don't cache RedirectorPages
|
|
|
|
public function subPagesToCache()
|
|
|
|
{
|
2020-01-24 22:07:29 +13:00
|
|
|
return [];
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
2011-03-18 16:01:06 +13:00
|
|
|
}
|