mirror of
https://github.com/silverstripe/silverstripe-subsites
synced 2024-10-22 11:05:55 +02:00
966 lines
32 KiB
PHP
966 lines
32 KiB
PHP
<?php
|
|
|
|
namespace SilverStripe\Subsites\Model;
|
|
|
|
use SilverStripe\Admin\CMSMenu;
|
|
use SilverStripe\CMS\Model\SiteTree;
|
|
use SilverStripe\Control\Director;
|
|
use SilverStripe\Control\Session;
|
|
use SilverStripe\Core\Convert;
|
|
use SilverStripe\Core\Injector\Injector;
|
|
use SilverStripe\Forms\CheckboxField;
|
|
use SilverStripe\Forms\CheckboxSetField;
|
|
use SilverStripe\Forms\DropdownField;
|
|
use SilverStripe\Forms\FieldList;
|
|
use SilverStripe\Forms\GridField\GridField;
|
|
use SilverStripe\Forms\GridField\GridFieldConfig_RecordEditor;
|
|
use SilverStripe\Forms\HeaderField;
|
|
use SilverStripe\Forms\HiddenField;
|
|
use SilverStripe\Forms\LiteralField;
|
|
use SilverStripe\Forms\Tab;
|
|
use SilverStripe\Forms\TabSet;
|
|
use SilverStripe\Forms\TextField;
|
|
use SilverStripe\i18n\Data\Intl\IntlLocales;
|
|
use SilverStripe\i18n\i18n;
|
|
use SilverStripe\ORM\ArrayLib;
|
|
use SilverStripe\ORM\ArrayList;
|
|
use SilverStripe\ORM\DataList;
|
|
use SilverStripe\ORM\DataObject;
|
|
use SilverStripe\ORM\DB;
|
|
use SilverStripe\ORM\SS_List;
|
|
use SilverStripe\Security\Group;
|
|
use SilverStripe\Security\Member;
|
|
use SilverStripe\Security\Permission;
|
|
use SilverStripe\Security\Security;
|
|
use SilverStripe\Subsites\State\SubsiteState;
|
|
use SilverStripe\Versioned\Versioned;
|
|
use UnexpectedValueException;
|
|
|
|
/**
|
|
* A dynamically created subsite. SiteTree objects can now belong to a subsite.
|
|
* You can simulate subsite access without setting up virtual hosts by appending ?SubsiteID=<ID> to the request.
|
|
*
|
|
* @package subsites
|
|
*/
|
|
class Subsite extends DataObject
|
|
{
|
|
|
|
private static $table_name = 'Subsite';
|
|
|
|
/**
|
|
* @var boolean $disable_subsite_filter If enabled, bypasses the query decoration
|
|
* to limit DataObject::get*() calls to a specific subsite. Useful for debugging.
|
|
*/
|
|
public static $disable_subsite_filter = false;
|
|
|
|
/**
|
|
* Allows you to force a specific subsite ID, or comma separated list of IDs.
|
|
* Only works for reading. An object cannot be written to more than 1 subsite.
|
|
*/
|
|
public static $force_subsite = null;
|
|
|
|
/**
|
|
*
|
|
* @var boolean
|
|
*/
|
|
public static $write_hostmap = true;
|
|
|
|
/**
|
|
* Memory cache of accessible sites
|
|
*
|
|
* @array
|
|
*/
|
|
private static $_cache_accessible_sites = [];
|
|
|
|
/**
|
|
* Memory cache of subsite id for domains
|
|
*
|
|
* @var array
|
|
*/
|
|
private static $_cache_subsite_for_domain = [];
|
|
|
|
/**
|
|
* @var array $allowed_themes Numeric array of all themes which are allowed to be selected for all subsites.
|
|
* Corresponds to subfolder names within the /themes folder. By default, all themes contained in this folder
|
|
* are listed.
|
|
*/
|
|
private static $allowed_themes = [];
|
|
|
|
/**
|
|
* @var Boolean If set to TRUE, don't assume 'www.example.com' and 'example.com' are the same.
|
|
* Doesn't affect wildcard matching, so '*.example.com' will match 'www.example.com' (but not 'example.com')
|
|
* in both TRUE or FALSE setting.
|
|
*/
|
|
public static $strict_subdomain_matching = false;
|
|
|
|
/**
|
|
* @var boolean Respects the IsPublic flag when retrieving subsites
|
|
*/
|
|
public static $check_is_public = true;
|
|
|
|
/*** @return array
|
|
*/
|
|
private static $summary_fields = [
|
|
'Title',
|
|
'PrimaryDomain',
|
|
'IsPublic'
|
|
];
|
|
|
|
/**
|
|
* Set allowed themes
|
|
*
|
|
* @param array $themes - Numeric array of all themes which are allowed to be selected for all subsites.
|
|
*/
|
|
public static function set_allowed_themes($themes)
|
|
{
|
|
self::$allowed_themes = $themes;
|
|
}
|
|
|
|
/**
|
|
* Gets the subsite currently set in the session.
|
|
*
|
|
* @uses ControllerSubsites->controllerAugmentInit()
|
|
* @return DataObject The current Subsite
|
|
*/
|
|
public static function currentSubsite()
|
|
{
|
|
return Subsite::get()->byID(SubsiteState::singleton()->getSubsiteId());
|
|
}
|
|
|
|
/**
|
|
* This function gets the current subsite ID from the session. It used in the backend so Ajax requests
|
|
* use the correct subsite. The frontend handles subsites differently. It calls getSubsiteIDForDomain
|
|
* directly from ModelAsController::getNestedController. Only gets Subsite instances which have their
|
|
* {@link IsPublic} flag set to TRUE.
|
|
*
|
|
* You can simulate subsite access without creating virtual hosts by appending ?SubsiteID=<ID> to the request.
|
|
*
|
|
* @return int ID of the current subsite instance
|
|
*
|
|
* @deprecated 2.0..3.0 Use SubsiteState::singleton()->getSubsiteId() instead
|
|
*/
|
|
public static function currentSubsiteID()
|
|
{
|
|
Deprecation::notice('3.0', 'Use SubsiteState::singleton()->getSubsiteId() instead');
|
|
return SubsiteState::singleton()->getSubsiteId();
|
|
}
|
|
|
|
/**
|
|
* Switch to another subsite through storing the subsite identifier in the current PHP session.
|
|
* Only takes effect when {@link SubsiteState::singleton()->getUseSessions()} is set to TRUE.
|
|
*
|
|
* @param int|Subsite $subsite Either the ID of the subsite, or the subsite object itself
|
|
*/
|
|
public static function changeSubsite($subsite)
|
|
{
|
|
// Session subsite change only meaningful if the session is active.
|
|
// Otherwise we risk setting it to wrong value, e.g. if we rely on currentSubsiteID.
|
|
if (!SubsiteState::singleton()->getUseSessions()) {
|
|
return;
|
|
}
|
|
|
|
if (is_object($subsite)) {
|
|
$subsiteID = $subsite->ID;
|
|
} else {
|
|
$subsiteID = $subsite;
|
|
}
|
|
|
|
SubsiteState::singleton()->setSubsiteId($subsiteID);
|
|
|
|
// Set locale
|
|
if (is_object($subsite) && $subsite->Language !== '') {
|
|
$locale = (new IntlLocales())->localeFromLang($subsite->Language);
|
|
if ($locale) {
|
|
i18n::set_locale($locale);
|
|
}
|
|
}
|
|
|
|
Permission::reset();
|
|
}
|
|
|
|
/**
|
|
* Get a matching subsite for the given host, or for the current HTTP_HOST.
|
|
* Supports "fuzzy" matching of domains by placing an asterisk at the start of end of the string,
|
|
* for example matching all subdomains on *.example.com with one subsite,
|
|
* and all subdomains on *.example.org on another.
|
|
*
|
|
* @param $host string The host to find the subsite for. If not specified, $_SERVER['HTTP_HOST'] is used.
|
|
* @param bool $checkPermissions
|
|
* @return int Subsite ID
|
|
*/
|
|
public static function getSubsiteIDForDomain($host = null, $checkPermissions = true)
|
|
{
|
|
if ($host == null && isset($_SERVER['HTTP_HOST'])) {
|
|
$host = $_SERVER['HTTP_HOST'];
|
|
}
|
|
|
|
$matchingDomains = null;
|
|
$cacheKey = null;
|
|
if ($host) {
|
|
if (!self::$strict_subdomain_matching) {
|
|
$host = preg_replace('/^www\./', '', $host);
|
|
}
|
|
|
|
$currentUserId = Security::getCurrentUser() ? Security::getCurrentUser()->ID : 0;
|
|
$cacheKey = implode('_', [$host, $currentUserId, self::$check_is_public]);
|
|
if (isset(self::$_cache_subsite_for_domain[$cacheKey])) {
|
|
return self::$_cache_subsite_for_domain[$cacheKey];
|
|
}
|
|
|
|
$SQL_host = Convert::raw2sql($host);
|
|
|
|
if (!in_array('SubsiteDomain', DB::table_list())) {
|
|
// Table hasn't been created yet. Might be a dev/build, skip.
|
|
return 0;
|
|
}
|
|
|
|
$matchingDomains = DataObject::get(
|
|
SubsiteDomain::class,
|
|
"'$SQL_host' LIKE replace(\"SubsiteDomain\".\"Domain\",'*','%')",
|
|
'"IsPrimary" DESC'
|
|
)->innerJoin(
|
|
'Subsite',
|
|
'"Subsite"."ID" = "SubsiteDomain"."SubsiteID" AND "Subsite"."IsPublic"=1'
|
|
);
|
|
}
|
|
|
|
if ($matchingDomains && $matchingDomains->count()) {
|
|
$subsiteIDs = array_unique($matchingDomains->column('SubsiteID'));
|
|
$subsiteDomains = array_unique($matchingDomains->column('Domain'));
|
|
if (sizeof($subsiteIDs) > 1) {
|
|
throw new UnexpectedValueException(sprintf(
|
|
"Multiple subsites match on '%s': %s",
|
|
$host,
|
|
implode(',', $subsiteDomains)
|
|
));
|
|
}
|
|
|
|
$subsiteID = $subsiteIDs[0];
|
|
} else {
|
|
if ($default = DataObject::get_one(Subsite::class, '"DefaultSite" = 1')) {
|
|
// Check for a 'default' subsite
|
|
$subsiteID = $default->ID;
|
|
} else {
|
|
// Default subsite id = 0, the main site
|
|
$subsiteID = 0;
|
|
}
|
|
}
|
|
|
|
if ($cacheKey) {
|
|
self::$_cache_subsite_for_domain[$cacheKey] = $subsiteID;
|
|
}
|
|
|
|
return $subsiteID;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param string $className
|
|
* @param string $filter
|
|
* @param string $sort
|
|
* @param string $join
|
|
* @param string $limit
|
|
* @return DataList
|
|
*/
|
|
public static function get_from_all_subsites($className, $filter = '', $sort = '', $join = '', $limit = '')
|
|
{
|
|
$result = DataObject::get($className, $filter, $sort, $join, $limit);
|
|
$result = $result->setDataQueryParam('Subsite.filter', false);
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Disable the sub-site filtering; queries will select from all subsites
|
|
* @param bool $disabled
|
|
*/
|
|
public static function disable_subsite_filter($disabled = true)
|
|
{
|
|
self::$disable_subsite_filter = $disabled;
|
|
}
|
|
|
|
/**
|
|
* Flush caches on database reset
|
|
*/
|
|
public static function on_db_reset()
|
|
{
|
|
self::$_cache_accessible_sites = [];
|
|
self::$_cache_subsite_for_domain = [];
|
|
}
|
|
|
|
/**
|
|
* Return all subsites, regardless of permissions (augmented with main site).
|
|
*
|
|
* @param bool $includeMainSite
|
|
* @param string $mainSiteTitle
|
|
* @return SS_List List of <a href='psi_element://Subsite'>Subsite</a> objects (DataList or ArrayList).
|
|
* objects (DataList or ArrayList).
|
|
*/
|
|
public static function all_sites($includeMainSite = true, $mainSiteTitle = 'Main site')
|
|
{
|
|
$subsites = Subsite::get();
|
|
|
|
if ($includeMainSite) {
|
|
$subsites = $subsites->toArray();
|
|
|
|
$mainSite = new Subsite();
|
|
$mainSite->Title = $mainSiteTitle;
|
|
array_unshift($subsites, $mainSite);
|
|
|
|
$subsites = ArrayList::create($subsites);
|
|
}
|
|
|
|
return $subsites;
|
|
}
|
|
|
|
/*
|
|
* Returns an ArrayList of the subsites accessible to the current user.
|
|
* It's enough for any section to be accessible for the site to be included.
|
|
*
|
|
* @return ArrayList of {@link Subsite} instances.
|
|
*/
|
|
public static function all_accessible_sites($includeMainSite = true, $mainSiteTitle = 'Main site', $member = null)
|
|
{
|
|
// Rationalise member arguments
|
|
if (!$member) {
|
|
$member = Security::getCurrentUser();
|
|
}
|
|
if (!$member) {
|
|
return ArrayList::create();
|
|
}
|
|
if (!is_object($member)) {
|
|
$member = DataObject::get_by_id(Member::class, $member);
|
|
}
|
|
|
|
$subsites = ArrayList::create();
|
|
|
|
// Collect subsites for all sections.
|
|
$menu = CMSMenu::get_viewable_menu_items();
|
|
foreach ($menu as $candidate) {
|
|
if ($candidate->controller) {
|
|
$accessibleSites = singleton($candidate->controller)->sectionSites(
|
|
$includeMainSite,
|
|
$mainSiteTitle,
|
|
$member
|
|
);
|
|
|
|
// Replace existing keys so no one site appears twice.
|
|
$subsites->merge($accessibleSites);
|
|
}
|
|
}
|
|
|
|
$subsites->removeDuplicates();
|
|
|
|
return $subsites;
|
|
}
|
|
|
|
/**
|
|
* Return the subsites that the current user can access by given permission.
|
|
* Sites will only be included if they have a Title.
|
|
*
|
|
* @param $permCode array|string Either a single permission code or an array of permission codes.
|
|
* @param $includeMainSite bool If true, the main site will be included if appropriate.
|
|
* @param $mainSiteTitle string The label to give to the main site
|
|
* @param $member int|Member The member attempting to access the sites
|
|
* @return DataList|ArrayList of {@link Subsite} instances
|
|
*/
|
|
public static function accessible_sites(
|
|
$permCode,
|
|
$includeMainSite = true,
|
|
$mainSiteTitle = 'Main site',
|
|
$member = null
|
|
) {
|
|
|
|
// Rationalise member arguments
|
|
if (!$member) {
|
|
$member = Member::currentUser();
|
|
}
|
|
if (!$member) {
|
|
return new ArrayList();
|
|
}
|
|
if (!is_object($member)) {
|
|
$member = DataObject::get_by_id(Member::class, $member);
|
|
}
|
|
|
|
// Rationalise permCode argument
|
|
if (is_array($permCode)) {
|
|
$SQL_codes = "'" . implode("', '", Convert::raw2sql($permCode)) . "'";
|
|
} else {
|
|
$SQL_codes = "'" . Convert::raw2sql($permCode) . "'";
|
|
}
|
|
|
|
// Cache handling
|
|
$cacheKey = $SQL_codes . '-' . $member->ID . '-' . $includeMainSite . '-' . $mainSiteTitle;
|
|
if (isset(self::$_cache_accessible_sites[$cacheKey])) {
|
|
return self::$_cache_accessible_sites[$cacheKey];
|
|
}
|
|
|
|
$subsites = DataList::create(Subsite::class)
|
|
->where("\"Subsite\".\"Title\" != ''")
|
|
->leftJoin('Group_Subsites', '"Group_Subsites"."SubsiteID" = "Subsite"."ID"')
|
|
->innerJoin(
|
|
'Group',
|
|
'"Group"."ID" = "Group_Subsites"."GroupID" OR "Group"."AccessAllSubsites" = 1'
|
|
)
|
|
->innerJoin(
|
|
'Group_Members',
|
|
"\"Group_Members\".\"GroupID\"=\"Group\".\"ID\" AND \"Group_Members\".\"MemberID\" = $member->ID"
|
|
)
|
|
->innerJoin(
|
|
'Permission',
|
|
"\"Group\".\"ID\"=\"Permission\".\"GroupID\" AND \"Permission\".\"Code\" IN ($SQL_codes, 'CMS_ACCESS_LeftAndMain', 'ADMIN')"
|
|
);
|
|
|
|
if (!$subsites) {
|
|
$subsites = new ArrayList();
|
|
}
|
|
|
|
/** @var DataList $rolesSubsites */
|
|
$rolesSubsites = DataList::create(Subsite::class)
|
|
->where("\"Subsite\".\"Title\" != ''")
|
|
->leftJoin('Group_Subsites', '"Group_Subsites"."SubsiteID" = "Subsite"."ID"')
|
|
->innerJoin(
|
|
'Group',
|
|
'"Group"."ID" = "Group_Subsites"."GroupID" OR "Group"."AccessAllSubsites" = 1'
|
|
)
|
|
->innerJoin(
|
|
'Group_Members',
|
|
"\"Group_Members\".\"GroupID\"=\"Group\".\"ID\" AND \"Group_Members\".\"MemberID\" = $member->ID"
|
|
)
|
|
->innerJoin('Group_Roles', '"Group_Roles"."GroupID"="Group"."ID"')
|
|
->innerJoin('PermissionRole', '"Group_Roles"."PermissionRoleID"="PermissionRole"."ID"')
|
|
->innerJoin(
|
|
'PermissionRoleCode',
|
|
"\"PermissionRole\".\"ID\"=\"PermissionRoleCode\".\"RoleID\" AND \"PermissionRoleCode\".\"Code\" IN ($SQL_codes, 'CMS_ACCESS_LeftAndMain', 'ADMIN')"
|
|
);
|
|
|
|
if (!$subsites && $rolesSubsites) {
|
|
return $rolesSubsites;
|
|
}
|
|
|
|
$subsites = new ArrayList($subsites->toArray());
|
|
|
|
if ($rolesSubsites) {
|
|
foreach ($rolesSubsites as $subsite) {
|
|
if (!$subsites->find('ID', $subsite->ID)) {
|
|
$subsites->push($subsite);
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($includeMainSite) {
|
|
if (!is_array($permCode)) {
|
|
$permCode = [$permCode];
|
|
}
|
|
if (self::hasMainSitePermission($member, $permCode)) {
|
|
$subsites = $subsites->toArray();
|
|
|
|
$mainSite = new Subsite();
|
|
$mainSite->Title = $mainSiteTitle;
|
|
array_unshift($subsites, $mainSite);
|
|
$subsites = ArrayList::create($subsites);
|
|
}
|
|
}
|
|
|
|
self::$_cache_accessible_sites[$cacheKey] = $subsites;
|
|
|
|
return $subsites;
|
|
}
|
|
|
|
/**
|
|
* Write a host->domain map to subsites/host-map.php
|
|
*
|
|
* This is used primarily when using subsites in conjunction with StaticPublisher
|
|
*
|
|
* @param string $file - filepath of the host map to be written
|
|
* @return void
|
|
*/
|
|
public static function writeHostMap($file = null)
|
|
{
|
|
if (!self::$write_hostmap) {
|
|
return;
|
|
}
|
|
|
|
if (!$file) {
|
|
$file = Director::baseFolder() . '/subsites/host-map.php';
|
|
}
|
|
$hostmap = [];
|
|
|
|
$subsites = DataObject::get(Subsite::class);
|
|
|
|
if ($subsites) {
|
|
foreach ($subsites as $subsite) {
|
|
$domains = $subsite->Domains();
|
|
if ($domains) {
|
|
foreach ($domains as $domain) {
|
|
$domainStr = $domain->Domain;
|
|
if (!self::$strict_subdomain_matching) {
|
|
$domainStr = preg_replace('/^www\./', '', $domainStr);
|
|
}
|
|
$hostmap[$domainStr] = $subsite->domain();
|
|
}
|
|
}
|
|
if ($subsite->DefaultSite) {
|
|
$hostmap['default'] = $subsite->domain();
|
|
}
|
|
}
|
|
}
|
|
|
|
$data = "<?php \n";
|
|
$data .= "// Generated by Subsite::writeHostMap() on " . date('d/M/y') . "\n";
|
|
$data .= '$subsiteHostmap = ' . var_export($hostmap, true) . ';';
|
|
|
|
if (is_writable(dirname($file)) || is_writable($file)) {
|
|
file_put_contents($file, $data);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Checks if a member can be granted certain permissions, regardless of the subsite context.
|
|
* Similar logic to {@link Permission::checkMember()}, but only returns TRUE
|
|
* if the member is part of a group with the "AccessAllSubsites" flag set.
|
|
* If more than one permission is passed to the method, at least one of them must
|
|
* be granted for if to return TRUE.
|
|
*
|
|
* @todo Allow permission inheritance through group hierarchy.
|
|
*
|
|
* @param Member Member to check against. Defaults to currently logged in member
|
|
* @param array $permissionCodes
|
|
* @return bool
|
|
*/
|
|
public static function hasMainSitePermission($member = null, $permissionCodes = ['ADMIN'])
|
|
{
|
|
if (!is_array($permissionCodes)) {
|
|
user_error('Permissions must be passed to Subsite::hasMainSitePermission as an array', E_USER_ERROR);
|
|
}
|
|
|
|
if (!$member && $member !== false) {
|
|
$member = Security::getCurrentUser();
|
|
}
|
|
|
|
if (!$member) {
|
|
return false;
|
|
}
|
|
|
|
if (!in_array('ADMIN', $permissionCodes)) {
|
|
$permissionCodes[] = 'ADMIN';
|
|
}
|
|
|
|
$SQLa_perm = Convert::raw2sql($permissionCodes);
|
|
$SQL_perms = join("','", $SQLa_perm);
|
|
$memberID = (int)$member->ID;
|
|
|
|
// Count this user's groups which can access the main site
|
|
$groupCount = DB::query("
|
|
SELECT COUNT(\"Permission\".\"ID\")
|
|
FROM \"Permission\"
|
|
INNER JOIN \"Group\" ON \"Group\".\"ID\" = \"Permission\".\"GroupID\" AND \"Group\".\"AccessAllSubsites\" = 1
|
|
INNER JOIN \"Group_Members\" ON \"Group_Members\".\"GroupID\" = \"Permission\".\"GroupID\"
|
|
WHERE \"Permission\".\"Code\" IN ('$SQL_perms')
|
|
AND \"Group_Members\".\"MemberID\" = {$memberID}
|
|
")->value();
|
|
|
|
// Count this user's groups which have a role that can access the main site
|
|
$roleCount = DB::query("
|
|
SELECT COUNT(\"PermissionRoleCode\".\"ID\")
|
|
FROM \"Group\"
|
|
INNER JOIN \"Group_Members\" ON \"Group_Members\".\"GroupID\" = \"Group\".\"ID\"
|
|
INNER JOIN \"Group_Roles\" ON \"Group_Roles\".\"GroupID\"=\"Group\".\"ID\"
|
|
INNER JOIN \"PermissionRole\" ON \"Group_Roles\".\"PermissionRoleID\"=\"PermissionRole\".\"ID\"
|
|
INNER JOIN \"PermissionRoleCode\" ON \"PermissionRole\".\"ID\"=\"PermissionRoleCode\".\"RoleID\"
|
|
WHERE \"PermissionRoleCode\".\"Code\" IN ('$SQL_perms')
|
|
AND \"Group\".\"AccessAllSubsites\" = 1
|
|
AND \"Group_Members\".\"MemberID\" = {$memberID}
|
|
")->value();
|
|
|
|
// There has to be at least one that allows access.
|
|
return ($groupCount + $roleCount > 0);
|
|
}
|
|
|
|
/**
|
|
* @var array
|
|
*/
|
|
private static $db = [
|
|
'Title' => 'Varchar(255)',
|
|
'RedirectURL' => 'Varchar(255)',
|
|
'DefaultSite' => 'Boolean',
|
|
'Theme' => 'Varchar',
|
|
'Language' => 'Varchar(6)',
|
|
|
|
// Used to hide unfinished/private subsites from public view.
|
|
// If unset, will default to true
|
|
'IsPublic' => 'Boolean',
|
|
|
|
// Comma-separated list of disallowed page types
|
|
'PageTypeBlacklist' => 'Text',
|
|
];
|
|
|
|
/**
|
|
* @var array
|
|
*/
|
|
private static $has_many = [
|
|
'Domains' => SubsiteDomain::class,
|
|
];
|
|
|
|
/**
|
|
* @var array
|
|
*/
|
|
private static $belongs_many_many = [
|
|
'Groups' => Group::class,
|
|
];
|
|
|
|
/**
|
|
* @var array
|
|
*/
|
|
private static $defaults = [
|
|
'IsPublic' => 1
|
|
];
|
|
|
|
/**
|
|
* @var array
|
|
*/
|
|
private static $searchable_fields = [
|
|
'Title',
|
|
'Domains.Domain',
|
|
'IsPublic',
|
|
];
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
private static $default_sort = '"Title" ASC';
|
|
|
|
/**
|
|
* @todo Possible security issue, don't grant edit permissions to everybody.
|
|
* @param bool $member
|
|
* @return bool
|
|
*/
|
|
public function canEdit($member = false)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Show the configuration fields for each subsite
|
|
*
|
|
* @return FieldList
|
|
*/
|
|
public function getCMSFields()
|
|
{
|
|
if ($this->ID != 0) {
|
|
$domainTable = new GridField(
|
|
'Domains',
|
|
_t('Subsite.DomainsListTitle', 'Domains'),
|
|
$this->Domains(),
|
|
GridFieldConfig_RecordEditor::create(10)
|
|
);
|
|
} else {
|
|
$domainTable = new LiteralField(
|
|
'Domains',
|
|
'<p>' . _t(
|
|
'Subsite.DOMAINSAVEFIRST',
|
|
'You can only add domains after saving for the first time'
|
|
) . '</p>'
|
|
);
|
|
}
|
|
|
|
$languageSelector = new DropdownField(
|
|
'Language',
|
|
$this->fieldLabel('Language'),
|
|
Injector::inst()->get(IntlLocales::class)->getLocales()
|
|
);
|
|
|
|
$pageTypeMap = [];
|
|
$pageTypes = SiteTree::page_type_classes();
|
|
foreach ($pageTypes as $pageType) {
|
|
$pageTypeMap[$pageType] = singleton($pageType)->i18n_singular_name();
|
|
}
|
|
asort($pageTypeMap);
|
|
|
|
$fields = new FieldList(
|
|
$subsiteTabs = new TabSet(
|
|
'Root',
|
|
new Tab(
|
|
'Configuration',
|
|
_t('Subsite.TabTitleConfig', 'Configuration'),
|
|
HeaderField::create('ConfigForSubsiteHeaderField', 'Subsite Configuration'),
|
|
TextField::create('Title', $this->fieldLabel('Title'), $this->Title),
|
|
HeaderField::create(
|
|
'DomainsForSubsiteHeaderField',
|
|
_t('Subsite.DomainsHeadline', 'Domains for this subsite')
|
|
),
|
|
$domainTable,
|
|
$languageSelector,
|
|
// new TextField('RedirectURL', 'Redirect to URL', $this->RedirectURL),
|
|
CheckboxField::create('DefaultSite', $this->fieldLabel('DefaultSite'), $this->DefaultSite),
|
|
CheckboxField::create('IsPublic', $this->fieldLabel('IsPublic'), $this->IsPublic),
|
|
LiteralField::create(
|
|
'PageTypeBlacklistToggle',
|
|
sprintf(
|
|
'<div class="field"><a href="#" id="PageTypeBlacklistToggle">%s</a></div>',
|
|
_t('Subsite.PageTypeBlacklistField', 'Disallow page types?')
|
|
)
|
|
),
|
|
CheckboxSetField::create(
|
|
'PageTypeBlacklist',
|
|
false,
|
|
$pageTypeMap
|
|
)
|
|
)
|
|
),
|
|
HiddenField::create('ID', '', $this->ID),
|
|
HiddenField::create('IsSubsite', '', 1)
|
|
);
|
|
|
|
// If there are any themes available, add the dropdown
|
|
$themes = $this->allowedThemes();
|
|
if (!empty($themes)) {
|
|
$fields->addFieldToTab(
|
|
'Root.Configuration',
|
|
DropdownField::create('Theme', $this->fieldLabel('Theme'), $this->allowedThemes(), $this->Theme)
|
|
->setEmptyString(_t('Subsite.ThemeFieldEmptyString', '-')),
|
|
'PageTypeBlacklistToggle'
|
|
);
|
|
}
|
|
|
|
$subsiteTabs->addExtraClass('subsite-model');
|
|
|
|
$this->extend('updateCMSFields', $fields);
|
|
return $fields;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param boolean $includerelations
|
|
* @return array
|
|
*/
|
|
public function fieldLabels($includerelations = true)
|
|
{
|
|
$labels = parent::fieldLabels($includerelations);
|
|
$labels['Title'] = _t('Subsites.TitleFieldLabel', 'Subsite Name');
|
|
$labels['RedirectURL'] = _t('Subsites.RedirectURLFieldLabel', 'Redirect URL');
|
|
$labels['DefaultSite'] = _t('Subsites.DefaultSiteFieldLabel', 'Default site');
|
|
$labels['Theme'] = _t('Subsites.ThemeFieldLabel', 'Theme');
|
|
$labels['Language'] = _t('Subsites.LanguageFieldLabel', 'Language');
|
|
$labels['IsPublic'] = _t('Subsites.IsPublicFieldLabel', 'Enable public access');
|
|
$labels['PageTypeBlacklist'] = _t('Subsites.PageTypeBlacklistFieldLabel', 'Page Type Blacklist');
|
|
$labels['Domains.Domain'] = _t('Subsites.DomainFieldLabel', 'Domain');
|
|
$labels['PrimaryDomain'] = _t('Subsites.PrimaryDomainFieldLabel', 'Primary Domain');
|
|
|
|
return $labels;
|
|
}
|
|
|
|
/**
|
|
* Return the themes that can be used with this subsite, as an array of themecode => description
|
|
*
|
|
* @return array
|
|
*/
|
|
public function allowedThemes()
|
|
{
|
|
if ($themes = $this->stat('allowed_themes')) {
|
|
return ArrayLib::valuekey($themes);
|
|
}
|
|
|
|
$themes = [];
|
|
if (is_dir(THEMES_PATH)) {
|
|
foreach (scandir(THEMES_PATH) as $theme) {
|
|
if ($theme[0] == '.') {
|
|
continue;
|
|
}
|
|
$theme = strtok($theme, '_');
|
|
$themes[$theme] = $theme;
|
|
}
|
|
ksort($themes);
|
|
}
|
|
return $themes;
|
|
}
|
|
|
|
/**
|
|
* @return string Current locale of the subsite
|
|
*/
|
|
public function getLanguage()
|
|
{
|
|
if ($this->getField('Language')) {
|
|
return $this->getField('Language');
|
|
}
|
|
|
|
return i18n::get_locale();
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @return \SilverStripe\ORM\ValidationResult
|
|
*/
|
|
public function validate()
|
|
{
|
|
$result = parent::validate();
|
|
if (!$this->Title) {
|
|
$result->addError(_t('Subsite.ValidateTitle', 'Please add a "Title"'));
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Whenever a Subsite is written, rewrite the hostmap
|
|
*
|
|
* @return void
|
|
*/
|
|
public function onAfterWrite()
|
|
{
|
|
Subsite::writeHostMap();
|
|
parent::onAfterWrite();
|
|
}
|
|
|
|
/**
|
|
* Return the primary domain of this site. Tries to "normalize" the domain name,
|
|
* by replacing potential wildcards.
|
|
*
|
|
* @return string The full domain name of this subsite (without protocol prefix)
|
|
*/
|
|
public function domain()
|
|
{
|
|
// Get best SubsiteDomain object
|
|
$domainObject = $this->getPrimarySubsiteDomain();
|
|
if ($domainObject) {
|
|
return $domainObject->SubstitutedDomain;
|
|
}
|
|
|
|
// If there are no objects, default to the current hostname
|
|
return $_SERVER['HTTP_HOST'];
|
|
}
|
|
|
|
/**
|
|
* Finds the primary {@see SubsiteDomain} object for this subsite
|
|
*
|
|
* @return SubsiteDomain
|
|
*/
|
|
public function getPrimarySubsiteDomain()
|
|
{
|
|
return $this
|
|
->Domains()
|
|
->sort('"IsPrimary" DESC')
|
|
->first();
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @return string - The full domain name of this subsite (without protocol prefix)
|
|
*/
|
|
public function getPrimaryDomain()
|
|
{
|
|
return $this->domain();
|
|
}
|
|
|
|
/**
|
|
* Get the absolute URL for this subsite
|
|
* @return string
|
|
*/
|
|
public function absoluteBaseURL()
|
|
{
|
|
// Get best SubsiteDomain object
|
|
$domainObject = $this->getPrimarySubsiteDomain();
|
|
if ($domainObject) {
|
|
return $domainObject->absoluteBaseURL();
|
|
}
|
|
|
|
// Fall back to the current base url
|
|
return Director::absoluteBaseURL();
|
|
}
|
|
|
|
/**
|
|
* Javascript admin action to duplicate this subsite
|
|
*
|
|
* @return string - javascript
|
|
*/
|
|
public function adminDuplicate()
|
|
{
|
|
$newItem = $this->duplicate();
|
|
$message = _t(
|
|
'Subsite.CopyMessage',
|
|
'Created a copy of {title}',
|
|
['title' => Convert::raw2js($this->Title)]
|
|
);
|
|
|
|
return <<<JS
|
|
statusMessage($message, 'good');
|
|
$('Form_EditForm').loadURLFromServer('admin/subsites/show/$newItem->ID');
|
|
JS;
|
|
}
|
|
|
|
/**
|
|
* Make this subsite the current one
|
|
*/
|
|
public function activate()
|
|
{
|
|
Subsite::changeSubsite($this);
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param array $permissionCodes
|
|
* @return DataList
|
|
*/
|
|
public function getMembersByPermission($permissionCodes = ['ADMIN'])
|
|
{
|
|
if (!is_array($permissionCodes)) {
|
|
user_error('Permissions must be passed to Subsite::getMembersByPermission as an array', E_USER_ERROR);
|
|
}
|
|
$SQL_permissionCodes = Convert::raw2sql($permissionCodes);
|
|
|
|
$SQL_permissionCodes = join("','", $SQL_permissionCodes);
|
|
|
|
return DataObject::get(
|
|
Member::class,
|
|
"\"Group\".\"SubsiteID\" = $this->ID AND \"Permission\".\"Code\" IN ('$SQL_permissionCodes')",
|
|
'',
|
|
'LEFT JOIN "Group_Members" ON "Member"."ID" = "Group_Members"."MemberID"
|
|
LEFT JOIN "Group" ON "Group"."ID" = "Group_Members"."GroupID"
|
|
LEFT JOIN "Permission" ON "Permission"."GroupID" = "Group"."ID"'
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Duplicate this subsite
|
|
* @param bool $doWrite
|
|
* @param string $manyMany
|
|
* @return DataObject
|
|
*/
|
|
public function duplicate($doWrite = true, $manyMany = 'many_many')
|
|
{
|
|
$duplicate = parent::duplicate($doWrite);
|
|
|
|
$oldSubsiteID = SubsiteState::singleton()->getSubsiteId();
|
|
self::changeSubsite($this->ID);
|
|
|
|
/*
|
|
* Copy data from this object to the given subsite. Does this using an iterative depth-first search.
|
|
* This will make sure that the new parents on the new subsite are correct, and there are no funny
|
|
* issues with having to check whether or not the new parents have been added to the site tree
|
|
* when a page, etc, is duplicated
|
|
*/
|
|
$stack = [[0, 0]];
|
|
while (count($stack) > 0) {
|
|
list($sourceParentID, $destParentID) = array_pop($stack);
|
|
$children = Versioned::get_by_stage('Page', 'Live', "\"ParentID\" = $sourceParentID", '');
|
|
|
|
if ($children) {
|
|
foreach ($children as $child) {
|
|
self::changeSubsite($duplicate->ID); //Change to destination subsite
|
|
|
|
$childClone = $child->duplicateToSubsite($duplicate, false);
|
|
$childClone->ParentID = $destParentID;
|
|
$childClone->writeToStage('Stage');
|
|
$childClone->copyVersionToStage('Stage', 'Live');
|
|
|
|
self::changeSubsite($this->ID); //Change Back to this subsite
|
|
|
|
array_push($stack, [$child->ID, $childClone->ID]);
|
|
}
|
|
}
|
|
}
|
|
|
|
self::changeSubsite($oldSubsiteID);
|
|
|
|
return $duplicate;
|
|
}
|
|
}
|