2009-08-04 03:09:26 +00:00
|
|
|
<?php
|
2016-06-16 16:57:19 +12:00
|
|
|
|
2016-07-22 11:32:32 +12:00
|
|
|
namespace SilverStripe\CMS\Controllers;
|
|
|
|
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Admin\LeftAndMain_SearchFilter;
|
2016-08-10 16:08:39 +12:00
|
|
|
use SilverStripe\CMS\Model\SiteTree;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Core\ClassInfo;
|
2020-10-05 16:17:52 +13:00
|
|
|
use SilverStripe\Core\Convert;
|
2017-03-29 17:19:57 +13:00
|
|
|
use SilverStripe\Core\Injector\Injectable;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Forms\DateField;
|
2016-08-10 16:08:39 +12:00
|
|
|
use SilverStripe\ORM\DataList;
|
2024-09-23 14:37:02 +12:00
|
|
|
use SilverStripe\Model\List\SS_List;
|
2017-03-21 17:26:46 +13:00
|
|
|
use SilverStripe\Versioned\Versioned;
|
2016-06-16 16:57:19 +12:00
|
|
|
|
2009-08-04 03:09:26 +00:00
|
|
|
/**
|
2009-10-21 22:26:52 +00:00
|
|
|
* Base class for filtering the subtree for certain node statuses.
|
2016-03-09 09:50:55 +13:00
|
|
|
*
|
2009-10-21 22:26:52 +00:00
|
|
|
* The simplest way of building a CMSSiteTreeFilter is to create a pagesToBeShown() method that
|
|
|
|
* returns an Iterator of maps, each entry containing the 'ID' and 'ParentID' of the pages to be
|
2014-03-28 08:29:30 +13:00
|
|
|
* included in the tree. The result of a DB::query() can then be returned directly.
|
2009-10-21 22:26:52 +00:00
|
|
|
*
|
|
|
|
* If you wish to make a more complex tree, you can overload includeInTree($page) to return true/
|
2014-03-28 08:29:30 +13:00
|
|
|
* false depending on whether the given page should be included. Note that you will need to include
|
2009-10-21 22:26:52 +00:00
|
|
|
* parent helper pages yourself.
|
2009-08-04 03:09:26 +00:00
|
|
|
*/
|
2017-01-26 09:59:25 +13:00
|
|
|
abstract class CMSSiteTreeFilter implements LeftAndMain_SearchFilter
|
|
|
|
{
|
2017-03-29 17:19:57 +13:00
|
|
|
use Injectable;
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Search parameters, mostly properties on {@link SiteTree}.
|
|
|
|
* Caution: Unescaped data.
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
2020-04-19 16:18:01 +12:00
|
|
|
protected $params = [];
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* List of filtered items and all their parents
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $_cache_ids = null;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Subset of $_cache_ids which include only items that appear directly in search results.
|
|
|
|
* When highlighting these, item IDs in this subset should be visually distinguished from
|
|
|
|
* others in the complete set.
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $_cache_highlight_ids = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
2020-04-19 16:18:01 +12:00
|
|
|
protected $_cache_expanded = [];
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $childrenMethod = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $numChildrenMethod = 'numChildren';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a sorted array of all implementators of CMSSiteTreeFilter, suitable for use in a dropdown.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function get_all_filters()
|
|
|
|
{
|
|
|
|
// get all filter instances
|
2017-09-20 13:51:07 +12:00
|
|
|
$filters = ClassInfo::subclassesFor(CMSSiteTreeFilter::class);
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
// remove abstract CMSSiteTreeFilter class
|
|
|
|
array_shift($filters);
|
|
|
|
|
|
|
|
// add filters to map
|
2020-04-19 16:18:01 +12:00
|
|
|
$filterMap = [];
|
2017-01-26 09:59:25 +13:00
|
|
|
foreach ($filters as $filter) {
|
|
|
|
$filterMap[$filter] = $filter::title();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure that 'all pages' filter is on top position and everything else is sorted alphabetically
|
|
|
|
uasort($filterMap, function ($a, $b) {
|
|
|
|
return ($a === CMSSiteTreeFilter_Search::title())
|
|
|
|
? -1
|
2022-04-13 17:07:59 +12:00
|
|
|
: strcasecmp($a ?? '', $b ?? '');
|
2017-01-26 09:59:25 +13:00
|
|
|
});
|
|
|
|
|
|
|
|
return $filterMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function __construct($params = null)
|
|
|
|
{
|
|
|
|
if ($params) {
|
|
|
|
$this->params = $params;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getChildrenMethod()
|
|
|
|
{
|
|
|
|
return $this->childrenMethod;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getNumChildrenMethod()
|
|
|
|
{
|
|
|
|
return $this->numChildrenMethod;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getPageClasses($page)
|
|
|
|
{
|
|
|
|
if ($this->_cache_ids === null) {
|
|
|
|
$this->populateIDs();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If directly selected via filter, apply highlighting
|
|
|
|
if (!empty($this->_cache_highlight_ids[$page->ID])) {
|
|
|
|
return 'filtered-item';
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the list of filtered pages
|
|
|
|
*
|
|
|
|
* @see {@link SiteTree::getStatusFlags()}
|
|
|
|
* @return SS_List
|
|
|
|
*/
|
|
|
|
abstract public function getFilteredPages();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array Map of Page IDs to their respective ParentID values.
|
|
|
|
*/
|
|
|
|
public function pagesIncluded()
|
|
|
|
{
|
|
|
|
return $this->mapIDs($this->getFilteredPages());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Populate the IDs of the pages returned by pagesIncluded(), also including
|
|
|
|
* the necessary parent helper pages.
|
|
|
|
*/
|
|
|
|
protected function populateIDs()
|
|
|
|
{
|
2020-04-19 16:18:01 +12:00
|
|
|
$parents = [];
|
|
|
|
$this->_cache_ids = [];
|
|
|
|
$this->_cache_highlight_ids = [];
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
if ($pages = $this->pagesIncluded()) {
|
|
|
|
// And keep a record of parents we don't need to get
|
|
|
|
// parents of themselves, as well as IDs to mark
|
|
|
|
foreach ($pages as $pageArr) {
|
|
|
|
$parents[$pageArr['ParentID']] = true;
|
|
|
|
$this->_cache_ids[$pageArr['ID']] = true;
|
|
|
|
$this->_cache_highlight_ids[$pageArr['ID']] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!empty($parents)) {
|
2018-03-14 16:34:46 +13:00
|
|
|
$q = Versioned::get_including_deleted(SiteTree::class)
|
2022-04-13 17:07:59 +12:00
|
|
|
->byIDs(array_keys($parents ?? []));
|
2017-01-26 09:59:25 +13:00
|
|
|
$list = $q->map('ID', 'ParentID');
|
2020-04-19 16:18:01 +12:00
|
|
|
$parents = [];
|
2017-01-26 09:59:25 +13:00
|
|
|
foreach ($list as $id => $parentID) {
|
|
|
|
if ($parentID) {
|
|
|
|
$parents[$parentID] = true;
|
|
|
|
}
|
|
|
|
$this->_cache_ids[$id] = true;
|
|
|
|
$this->_cache_expanded[$id] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function isPageIncluded($page)
|
|
|
|
{
|
|
|
|
if ($this->_cache_ids === null) {
|
|
|
|
$this->populateIDs();
|
|
|
|
}
|
|
|
|
|
|
|
|
return !empty($this->_cache_ids[$page->ID]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Applies the default filters to a specified DataList of pages
|
|
|
|
*
|
|
|
|
* @param DataList $query Unfiltered query
|
|
|
|
* @return DataList Filtered query
|
|
|
|
*/
|
|
|
|
protected function applyDefaultFilters($query)
|
|
|
|
{
|
|
|
|
$sng = SiteTree::singleton();
|
|
|
|
foreach ($this->params as $name => $val) {
|
|
|
|
if (empty($val)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ($name) {
|
|
|
|
case 'Term':
|
2020-10-22 11:52:02 +13:00
|
|
|
$query = $query->filterAny([
|
2020-10-05 16:17:52 +13:00
|
|
|
'URLSegment:PartialMatch' => Convert::raw2url($val),
|
2017-01-26 09:59:25 +13:00
|
|
|
'Title:PartialMatch' => $val,
|
|
|
|
'MenuTitle:PartialMatch' => $val,
|
|
|
|
'Content:PartialMatch' => $val
|
2020-04-19 16:18:01 +12:00
|
|
|
]);
|
2017-01-26 09:59:25 +13:00
|
|
|
break;
|
|
|
|
|
2020-10-05 16:17:52 +13:00
|
|
|
case 'URLSegment':
|
|
|
|
$query = $query->filter([
|
|
|
|
'URLSegment:PartialMatch' => Convert::raw2url($val),
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
case 'LastEditedFrom':
|
|
|
|
$fromDate = new DateField(null, null, $val);
|
|
|
|
$query = $query->filter("LastEdited:GreaterThanOrEqual", $fromDate->dataValue().' 00:00:00');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'LastEditedTo':
|
|
|
|
$toDate = new DateField(null, null, $val);
|
|
|
|
$query = $query->filter("LastEdited:LessThanOrEqual", $toDate->dataValue().' 23:59:59');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'ClassName':
|
|
|
|
if ($val != 'All') {
|
|
|
|
$query = $query->filter('ClassName', $val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
$field = $sng->dbObject($name);
|
|
|
|
if ($field) {
|
|
|
|
$filter = $field->defaultSearchFilter();
|
|
|
|
$filter->setValue($val);
|
2020-04-19 16:18:01 +12:00
|
|
|
$query = $query->alterDataQuery([$filter, 'apply']);
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $query;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps a list of pages to an array of associative arrays with ID and ParentID keys
|
|
|
|
*
|
|
|
|
* @param SS_List $pages
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function mapIDs($pages)
|
|
|
|
{
|
2020-04-19 16:18:01 +12:00
|
|
|
$ids = [];
|
2017-01-26 09:59:25 +13:00
|
|
|
if ($pages) {
|
|
|
|
foreach ($pages as $page) {
|
2020-04-19 16:18:01 +12:00
|
|
|
$ids[] = ['ID' => $page->ID, 'ParentID' => $page->ParentID];
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return $ids;
|
|
|
|
}
|
2009-08-04 03:09:26 +00:00
|
|
|
}
|