2013-01-17 13:22:13 +13:00
|
|
|
<?php
|
2016-06-16 17:06:51 +12:00
|
|
|
|
2016-08-29 13:55:43 +12:00
|
|
|
namespace SilverStripe\Reports;
|
|
|
|
|
|
|
|
use SilverStripe\Admin\LeftAndMain;
|
|
|
|
use SilverStripe\Control\Controller;
|
|
|
|
use SilverStripe\Forms\FieldList;
|
|
|
|
use SilverStripe\Forms\Form;
|
2017-06-21 16:30:14 +12:00
|
|
|
use SilverStripe\Forms\GridField\GridField;
|
2024-10-11 14:45:26 +13:00
|
|
|
use SilverStripe\Forms\GridField\GridFieldButtonRow;
|
2016-08-29 13:55:43 +12:00
|
|
|
use SilverStripe\Forms\GridField\GridFieldConfig;
|
|
|
|
use SilverStripe\Forms\GridField\GridFieldDataColumns;
|
2024-10-11 14:45:26 +13:00
|
|
|
use SilverStripe\Forms\GridField\GridFieldFilterHeader;
|
|
|
|
use SilverStripe\Forms\GridField\GridFieldPageCount;
|
|
|
|
use SilverStripe\Forms\GridField\GridFieldPaginator;
|
2017-06-21 16:30:14 +12:00
|
|
|
use SilverStripe\Forms\GridField\GridFieldSortableHeader;
|
2024-10-11 14:45:26 +13:00
|
|
|
use SilverStripe\Forms\HiddenField;
|
2016-08-29 13:55:43 +12:00
|
|
|
use SilverStripe\Forms\HTMLEditor\HTMLEditorConfig;
|
2024-10-11 14:45:26 +13:00
|
|
|
use SilverStripe\Forms\TextField;
|
2016-06-16 17:06:51 +12:00
|
|
|
use SilverStripe\ORM\ArrayList;
|
2024-10-11 14:45:26 +13:00
|
|
|
use SilverStripe\ORM\Filters\PartialMatchFilter;
|
|
|
|
use SilverStripe\ORM\Search\BasicSearchContext;
|
2016-07-07 13:43:27 +12:00
|
|
|
use SilverStripe\ORM\SS_List;
|
|
|
|
use SilverStripe\Security\Member;
|
|
|
|
use SilverStripe\Security\PermissionProvider;
|
2017-06-09 16:48:30 +12:00
|
|
|
use SilverStripe\Security\Security;
|
2016-08-29 13:55:43 +12:00
|
|
|
use SilverStripe\View\ArrayData;
|
|
|
|
use SilverStripe\View\Requirements;
|
2016-06-16 17:06:51 +12:00
|
|
|
|
2013-01-17 13:22:13 +13:00
|
|
|
/**
|
|
|
|
* Reports section of the CMS.
|
2016-06-16 17:06:51 +12:00
|
|
|
*
|
2013-01-17 13:22:13 +13:00
|
|
|
* All reports that should show in the ReportAdmin section
|
2017-06-27 15:52:13 +12:00
|
|
|
* of the CMS need to subclass {@link SilverStripe\Reports\Report}, and implement
|
2013-01-17 13:22:13 +13:00
|
|
|
* the appropriate methods and variables that are required.
|
|
|
|
*/
|
2015-12-16 11:06:45 +13:00
|
|
|
class ReportAdmin extends LeftAndMain implements PermissionProvider
|
|
|
|
{
|
|
|
|
private static $url_segment = 'reports';
|
2016-06-16 17:06:51 +12:00
|
|
|
|
2015-12-16 11:06:45 +13:00
|
|
|
private static $menu_title = 'Reports';
|
2016-06-16 17:06:51 +12:00
|
|
|
|
2016-12-20 16:13:46 +13:00
|
|
|
private static $menu_icon_class = 'font-icon-chart-line';
|
2016-12-20 13:30:50 +13:00
|
|
|
|
2015-12-16 11:06:45 +13:00
|
|
|
private static $template_path = null; // defaults to (project)/templates/email
|
|
|
|
|
2017-07-16 16:36:12 +12:00
|
|
|
private static $tree_class = Report::class;
|
2015-12-16 11:06:45 +13:00
|
|
|
|
|
|
|
private static $url_handlers = array(
|
2016-08-01 16:35:37 +12:00
|
|
|
'show/$ReportClass/$Action' => 'handleAction'
|
2015-12-16 11:06:45 +13:00
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
2016-06-16 17:06:51 +12:00
|
|
|
* Variable that describes which report we are currently viewing based on
|
2015-12-16 11:06:45 +13:00
|
|
|
* the URL (gets set in init method).
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $reportClass;
|
|
|
|
|
2016-07-14 17:51:20 +12:00
|
|
|
/**
|
2016-09-09 18:11:38 +12:00
|
|
|
* @var Report
|
2016-07-14 17:51:20 +12:00
|
|
|
*/
|
2015-12-16 11:06:45 +13:00
|
|
|
protected $reportObject;
|
2016-06-16 17:06:51 +12:00
|
|
|
|
2016-08-12 15:55:47 +12:00
|
|
|
private static $required_permission_codes = 'CMS_ACCESS_ReportAdmin';
|
|
|
|
|
2015-12-16 11:06:45 +13:00
|
|
|
public function init()
|
|
|
|
{
|
|
|
|
parent::init();
|
|
|
|
|
|
|
|
// Set custom options for TinyMCE specific to ReportAdmin
|
2016-07-07 13:43:27 +12:00
|
|
|
HTMLEditorConfig::get('cms')->setOption('content_css', project() . '/css/editor.css');
|
2015-12-16 11:06:45 +13:00
|
|
|
|
2017-06-28 10:24:31 +12:00
|
|
|
Requirements::javascript('silverstripe/reports: javascript/ReportAdmin.js');
|
2015-12-16 11:06:45 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Does the parent permission checks, but also
|
|
|
|
* makes sure that instantiatable subclasses of
|
2017-06-27 15:52:13 +12:00
|
|
|
* {@link SilverStripe\Reports\Report} exist. By default, the CMS doesn't
|
2015-12-16 11:06:45 +13:00
|
|
|
* include any Reports, so there's no point in showing
|
|
|
|
*
|
|
|
|
* @param Member $member
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function canView($member = null)
|
|
|
|
{
|
|
|
|
if (!$member && $member !== false) {
|
2017-06-09 16:48:30 +12:00
|
|
|
$member = Security::getCurrentUser();
|
2015-12-16 11:06:45 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!parent::canView($member)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($this->Reports() as $report) {
|
|
|
|
if ($report->canView($member)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a SS_List of SS_Report subclasses
|
|
|
|
* that are available for use.
|
|
|
|
*
|
|
|
|
* @return SS_List
|
|
|
|
*/
|
|
|
|
public function Reports()
|
|
|
|
{
|
2024-10-11 14:45:26 +13:00
|
|
|
$output = ArrayList::create();
|
2016-09-09 18:11:38 +12:00
|
|
|
foreach (Report::get_reports() as $report) {
|
2015-12-16 11:06:45 +13:00
|
|
|
if ($report->canView()) {
|
|
|
|
$output->push($report);
|
|
|
|
}
|
|
|
|
}
|
2024-10-11 14:45:26 +13:00
|
|
|
|
|
|
|
return $output
|
|
|
|
->sort('Title', 'ASC')
|
|
|
|
->setDataClass(Report::class);
|
2015-12-16 11:06:45 +13:00
|
|
|
}
|
|
|
|
|
2016-07-29 10:44:00 +12:00
|
|
|
public function handleAction($request, $action)
|
|
|
|
{
|
2016-08-12 15:55:47 +12:00
|
|
|
$this->reportClass = $this->unsanitiseClassName($request->param('ReportClass'));
|
2016-08-01 16:35:37 +12:00
|
|
|
|
|
|
|
// Check report
|
|
|
|
if ($this->reportClass) {
|
2016-09-09 18:11:38 +12:00
|
|
|
$allReports = Report::get_reports();
|
2016-08-01 16:35:37 +12:00
|
|
|
if (empty($allReports[$this->reportClass])) {
|
|
|
|
return $this->httpError(404);
|
|
|
|
}
|
|
|
|
$this->reportObject = $allReports[$this->reportClass];
|
2024-03-27 14:06:39 +13:00
|
|
|
if (!$this->reportObject->canView()) {
|
|
|
|
return Security::permissionFailure($this);
|
|
|
|
}
|
2016-08-01 16:35:37 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delegate to sub-form
|
|
|
|
return parent::handleAction($request, $action);
|
|
|
|
}
|
|
|
|
|
2017-06-21 16:30:14 +12:00
|
|
|
/**
|
|
|
|
* Unsanitise a model class' name from a URL param
|
|
|
|
*
|
|
|
|
* @param string $class
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function unsanitiseClassName($class)
|
|
|
|
{
|
2022-04-13 17:39:46 +12:00
|
|
|
return str_replace('-', '\\', $class ?? '');
|
2017-06-21 16:30:14 +12:00
|
|
|
}
|
2016-08-12 15:55:47 +12:00
|
|
|
|
2015-12-16 11:06:45 +13:00
|
|
|
/**
|
|
|
|
* Determine if we have reports and need
|
|
|
|
* to display the "Reports" main menu item
|
|
|
|
* in the CMS.
|
|
|
|
*
|
|
|
|
* The test for an existance of a report
|
|
|
|
* is done by checking for a subclass of
|
|
|
|
* "SS_Report" that exists.
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public static function has_reports()
|
|
|
|
{
|
2022-04-13 17:39:46 +12:00
|
|
|
return sizeof(Report::get_reports() ?? []) > 0;
|
2015-12-16 11:06:45 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the Breadcrumbs for the ReportAdmin
|
|
|
|
*/
|
|
|
|
public function Breadcrumbs($unlinked = false)
|
|
|
|
{
|
|
|
|
$items = parent::Breadcrumbs($unlinked);
|
2016-06-16 17:06:51 +12:00
|
|
|
|
2015-12-16 11:06:45 +13:00
|
|
|
// The root element should explicitly point to the root node.
|
|
|
|
// Uses session state for current record otherwise.
|
2016-08-29 13:55:43 +12:00
|
|
|
$items[0]->Link = singleton('SilverStripe\\Reports\\ReportAdmin')->Link();
|
2015-12-16 11:06:45 +13:00
|
|
|
|
2018-11-16 15:09:45 +13:00
|
|
|
if ($report = $this->reportObject) {
|
|
|
|
$breadcrumbs = $report->getBreadcrumbs();
|
|
|
|
if (!empty($breadcrumbs)) {
|
|
|
|
foreach ($breadcrumbs as $crumb) {
|
|
|
|
$items->push($crumb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-16 11:06:45 +13:00
|
|
|
//build breadcrumb trail to the current report
|
2019-01-09 20:59:28 +01:00
|
|
|
$items->push(ArrayData::create([
|
2018-11-16 15:09:45 +13:00
|
|
|
'Title' => $report->title(),
|
|
|
|
'Link' => Controller::join_links(
|
|
|
|
$this->Link(),
|
2019-01-09 20:59:28 +01:00
|
|
|
'?' . http_build_query(['q' => $this->request->requestVar('q')])
|
2018-11-16 15:09:45 +13:00
|
|
|
)
|
2019-01-09 20:59:28 +01:00
|
|
|
]));
|
2015-12-16 11:06:45 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
return $items;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the link to the report admin section, or the specific report that is currently displayed
|
2016-07-07 13:43:27 +12:00
|
|
|
*
|
|
|
|
* @param string $action
|
|
|
|
* @return string
|
2015-12-16 11:06:45 +13:00
|
|
|
*/
|
|
|
|
public function Link($action = null)
|
|
|
|
{
|
|
|
|
if ($this->reportObject) {
|
2016-07-14 17:51:20 +12:00
|
|
|
return $this->reportObject->getLink($action);
|
2015-12-16 11:06:45 +13:00
|
|
|
}
|
2016-07-14 17:51:20 +12:00
|
|
|
|
|
|
|
// Basic link to this cms section
|
2016-08-01 16:35:37 +12:00
|
|
|
return parent::Link($action);
|
2015-12-16 11:06:45 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function providePermissions()
|
|
|
|
{
|
|
|
|
return array(
|
|
|
|
"CMS_ACCESS_ReportAdmin" => array(
|
2017-05-08 17:56:43 +12:00
|
|
|
'name' => _t('SilverStripe\\CMS\\Controllers\\CMSMain.ACCESS', "Access to '{title}' section", array(
|
2016-08-11 14:29:53 +12:00
|
|
|
'title' => static::menu_title()
|
|
|
|
)),
|
2017-05-08 17:56:43 +12:00
|
|
|
'category' => _t('SilverStripe\\Security\\Permission.CMS_ACCESS_CATEGORY', 'CMS Access')
|
2015-12-16 11:06:45 +13:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getEditForm($id = null, $fields = null)
|
|
|
|
{
|
|
|
|
$report = $this->reportObject;
|
|
|
|
if ($report) {
|
|
|
|
$fields = $report->getCMSFields();
|
|
|
|
} else {
|
|
|
|
// List all reports
|
2024-10-11 14:45:26 +13:00
|
|
|
$fields = FieldList::create();
|
2015-12-16 11:06:45 +13:00
|
|
|
$gridFieldConfig = GridFieldConfig::create()->addComponents(
|
2024-10-11 14:45:26 +13:00
|
|
|
// This is a container component that is required by filter header component
|
|
|
|
GridFieldButtonRow::create('before'),
|
|
|
|
$filterHeader = GridFieldFilterHeader::create(),
|
2022-03-04 10:13:45 +13:00
|
|
|
GridFieldSortableHeader::create(),
|
2024-10-11 14:45:26 +13:00
|
|
|
$columns = GridFieldDataColumns::create(),
|
|
|
|
GridFieldPageCount::create(),
|
|
|
|
GridFieldPaginator::create()
|
2015-12-16 11:06:45 +13:00
|
|
|
);
|
2024-10-11 14:45:26 +13:00
|
|
|
|
2024-10-14 13:37:13 +13:00
|
|
|
$titleLabel = _t('SilverStripe\\Reports\\ReportAdmin.ReportTitle', 'Title');
|
|
|
|
$descriptionLabel = _t('SilverStripe\\Reports\\ReportAdmin.ReportDescription', 'Description');
|
|
|
|
|
2024-10-11 14:45:26 +13:00
|
|
|
// Configure the filter header filter search form
|
|
|
|
$generalField = BasicSearchContext::config()->get('general_search_field_name');
|
|
|
|
$searchFieldList = FieldList::create([
|
|
|
|
HiddenField::create($generalField),
|
2024-10-14 13:37:13 +13:00
|
|
|
TextField::create('Title', $titleLabel),
|
|
|
|
TextField::create('Description', $descriptionLabel),
|
2024-10-11 14:45:26 +13:00
|
|
|
]);
|
|
|
|
$searchContext = BasicSearchContext::create(Report::class);
|
|
|
|
$searchContext->setFields($searchFieldList);
|
|
|
|
|
|
|
|
// Setup filter configuration - partial match with case-insensitive modifier
|
|
|
|
$filters = [
|
|
|
|
'Title',
|
|
|
|
'Description',
|
|
|
|
];
|
|
|
|
foreach ($filters as $fieldName) {
|
|
|
|
$fieldFilter = PartialMatchFilter::create($fieldName);
|
|
|
|
$searchContext->addFilter($fieldFilter);
|
|
|
|
}
|
|
|
|
$filterHeader->setSearchContext($searchContext);
|
|
|
|
|
2022-03-04 10:13:45 +13:00
|
|
|
$gridField = GridField::create('Reports', false, $this->Reports(), $gridFieldConfig);
|
2015-12-16 11:06:45 +13:00
|
|
|
$columns->setDisplayFields(array(
|
2024-10-14 13:37:13 +13:00
|
|
|
'title' => $titleLabel,
|
|
|
|
'description' => $descriptionLabel,
|
2015-12-16 11:06:45 +13:00
|
|
|
));
|
|
|
|
|
2024-10-11 14:45:26 +13:00
|
|
|
$columns->setFieldFormatting([
|
|
|
|
'title' => '<a href=\"$Link\" class=\"grid-field__link-block\">$value ($CountForOverview)</a>'
|
|
|
|
]);
|
2015-12-16 11:06:45 +13:00
|
|
|
$gridField->addExtraClass('all-reports-gridfield');
|
|
|
|
$fields->push($gridField);
|
|
|
|
}
|
|
|
|
|
2024-10-11 14:45:26 +13:00
|
|
|
$actions = FieldList::create();
|
|
|
|
$form = Form::create($this, "EditForm", $fields, $actions);
|
2019-01-09 20:59:28 +01:00
|
|
|
$form->addExtraClass(
|
|
|
|
'panel panel--padded panel--scrollable cms-edit-form cms-panel-padded' . $this->BaseCSSClasses()
|
|
|
|
);
|
2015-12-16 11:06:45 +13:00
|
|
|
$form->loadDataFrom($this->request->getVars());
|
|
|
|
|
|
|
|
$this->extend('updateEditForm', $form);
|
|
|
|
|
|
|
|
return $form;
|
|
|
|
}
|
2013-02-21 09:04:21 +13:00
|
|
|
}
|