<?php // This file is part of Moodle - http://moodle.org/ // // Moodle is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Moodle is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with Moodle. If not, see <http://www.gnu.org/licenses/>. declare(strict_types=1); namespace core_reportbuilder; use action_menu_filler; use coding_exception;> use html_writer;use stdClass;> use core\output\checkbox_toggleall;use core_reportbuilder\local\models\report; use core_reportbuilder\local\report\action; use core_reportbuilder\local\report\base; use core_reportbuilder\local\report\column; /** * Base class for system reports * * @package core_reportbuilder * @copyright 2020 Paul Holden <paulh@moodle.com> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ abstract class system_report extends base { /** @var array $parameters */ private $parameters; /** @var string[] $basefields List of base fields */ private $basefields = [];> /** @var callable $checkboxcallback */ /** @var bool $filterformdefault Whether to use the default filters form */ > private $checkboxcallback = null; private $filterformdefault = true; >/** @var action|action_menu_filler[] $actions */ private $actions = []; /** @var column $initialsortcolumn */ private $initialsortcolumn; /** @var int $initialsortdirection */ private $initialsortdirection; /** * System report constructor. * * @param report $report * @param array $parameters */ final public function __construct(report $report, array $parameters) { $this->parameters = $parameters; parent::__construct($report); } /** * Validates access to view this report * * This is necessary to implement independently of the page that would typically embed the report because * subsequent pages are requested via AJAX requests, and access should be validated each time * * @return bool */ abstract protected function can_view(): bool; /** * Validate access to the report * * @throws report_access_exception */ final public function require_can_view(): void { if (!$this->can_view()) { throw new report_access_exception(); } } /** * Report validation * * @throws report_access_exception If user cannot access the report * @throws coding_exception If no default column are specified */ protected function validate(): void { parent::validate(); $this->require_can_view(); // Ensure the report has some default columns specified. if (empty($this->get_columns())) { throw new coding_exception('No columns added'); } } /** * Add list of fields that have to be always included in SQL query for actions and row classes * * Base fields are only available in system reports because they are not compatible with aggregation * * @param string $sql SQL clause for the list of fields that only uses main table or base joins */ final protected function add_base_fields(string $sql): void { $this->basefields[] = $sql; } /** * Return report base fields * * @return array */ final public function get_base_fields(): array { return $this->basefields;> } } > > /** /** > * Define toggle all checkbox for the report, required row data should be defined by calling {@see add_base_fields} * Override whether to use the default system report filters form, for instance this can be disabled if the UI requires > * * it's own custom filter management form for a specific report > * @param callable $callback Callback to return value/label for each checkbox, implementing the following signature: * > * function(stdClass $row): array containing value/label pair * @param bool $filterformdefault > */ */ > final protected function set_checkbox_toggleall(callable $callback): void { final public function set_filter_form_default(bool $filterformdefault = true): void { > $this->checkboxcallback = $callback; $this->filterformdefault = $filterformdefault; > } } > > /** /** > * Return instance of toggle all checkbox, if previously defined by {@see set_checkbox_toggleall} * Whether to use the default filters form > * * > * @param bool $ismaster * @return bool > * @param stdClass|null $row */ > * @return checkbox_toggleall|null final public function get_filter_form_default(): bool { > */ return $this->filterformdefault; > final public function get_checkbox_toggleall(bool $ismaster, ?stdClass $row = null): ?checkbox_toggleall { } > if (!is_callable($this->checkboxcallback)) { > return null; /** > } * Adds an action to the report > * > // Generic content for the master checkbox, execute callback for those belonging to each row. * @param action $action > if ($ismaster) { */ > $value = ''; final public function add_action(action $action): void { > $label = get_string('selectall'); $this->actions[] = $action; > } else { } > [$value, $label] = ($this->checkboxcallback)($row); > } /** > * Adds action divider to the report > return new checkbox_toggleall('report-select-all', $ismaster, [ * > 'id' => html_writer::random_id(), */ > 'name' => 'report-select-row[]', final public function add_action_divider(): void { > 'value' => $value, $divider = new action_menu_filler(); > 'label' => $label, // We need to set as not primary action because we just need add an action divider, not a new action item. > 'labelclasses' => 'accesshide', $divider->primary = false; > ]);$this->actions[] = $divider; } /** * Whether report has any actions * * @return bool */ final public function has_actions(): bool { return !empty($this->actions); } /** * Return report actions * * @return action|action_menu_filler[] */ final public function get_actions(): array { return $this->actions; } /** * Set all report parameters * * @param array $parameters */ final public function set_parameters(array $parameters): void { $this->parameters = $parameters; } /** * Return all report parameters * * @return array */ final public function get_parameters(): array { return $this->parameters; } /** * Return specific report parameter * * @param string $param * @param mixed $default * @param string $type * @return mixed */ final public function get_parameter(string $param, $default, string $type) { if (!array_key_exists($param, $this->parameters)) { return $default; } return clean_param($this->parameters[$param], $type); } /** * Output the report * * @uses \core_reportbuilder\output\renderer::render_system_report() * * @return string */ final public function output(): string { global $PAGE; /** @var \core_reportbuilder\output\renderer $renderer */ $renderer = $PAGE->get_renderer('core_reportbuilder'); $report = new \core_reportbuilder\output\system_report($this->get_report_persistent(), $this, $this->parameters); return $renderer->render($report); } /** * CSS classes to add to the row. Can be overridden by system reports do define class to be added to output according to * content of each row * * @param stdClass $row * @return string */ public function get_row_class(stdClass $row): string { return ''; } /** * Called before rendering each row. Can be overridden to pre-fetch/create objects and store them in the class, which can * later be used in column and action callbacks * * @param stdClass $row */ public function row_callback(stdClass $row): void { return; } /** * Validates access to download this report. * * @return bool */ final public function can_be_downloaded(): bool { return $this->can_view() && $this->is_downloadable(); } /** * Return list of column names that will be excluded when table is downloaded. Extending classes should override this method * as appropriate * * @return string[] Array of column unique identifiers */ public function get_exclude_columns_for_download(): array { return []; } /** * Set initial sort column and sort direction for the report * * @param string $uniqueidentifier * @param int $sortdirection One of SORT_ASC or SORT_DESC * @throws coding_exception */ public function set_initial_sort_column(string $uniqueidentifier, int $sortdirection): void { if (!$sortcolumn = $this->get_column($uniqueidentifier)) { throw new coding_exception('Unknown column identifier', $uniqueidentifier); } $this->initialsortcolumn = $sortcolumn; $this->initialsortdirection = $sortdirection; } /** * Get initial sort column * * @return column|null */ public function get_initial_sort_column(): ?column { return $this->initialsortcolumn; } /** * Get initial sort column direction * * @return int */ public function get_initial_sort_direction(): int { return $this->initialsortdirection; } }