. declare(strict_types=1); namespace core\reportbuilder\local\entities; use core\context_helper; use core_reportbuilder\local\entities\base; use core_reportbuilder\local\filters\{select, text}; use core_reportbuilder\local\report\{column, filter}; use html_writer; use lang_string; use stdClass; /** * Context entity * * @package core * @copyright 2023 Paul Holden * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class context extends base { /** * Database tables that this entity uses * * @return string[] */ protected function get_default_tables(): array { return [ 'context', ]; } /** * The default title for this entity in the list of columns/conditions/filters in the report builder * * @return lang_string */ protected function get_default_entity_title(): lang_string { return new lang_string('context'); } /** * Initialise the entity * * @return base */ public function initialise(): base { $columns = $this->get_all_columns(); foreach ($columns as $column) { $this->add_column($column); } // All the filters defined by the entity can also be used as conditions. $filters = $this->get_all_filters(); foreach ($filters as $filter) { $this ->add_filter($filter) ->add_condition($filter); } return $this; } /** * Returns list of all available columns * * @return column[] */ protected function get_all_columns(): array { global $DB; $contextalias = $this->get_table_alias('context'); // Name. $columns[] = (new column( 'name', new lang_string('contextname'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_TEXT) ->add_fields(context_helper::get_preload_record_columns_sql($contextalias)) // Sorting may not order alphabetically, but will at least group contexts together. ->set_is_sortable(true) ->add_callback(static function($contextid, stdClass $context): string { if ($contextid === null) { return ''; } context_helper::preload_from_record($context); return context_helper::instance_by_id($contextid)->get_context_name(); }); // Link. $columns[] = (new column( 'link', new lang_string('contexturl'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_TEXT) ->add_fields(context_helper::get_preload_record_columns_sql($contextalias)) // Sorting may not order alphabetically, but will at least group contexts together. ->set_is_sortable(true) ->add_callback(static function($contextid, stdClass $context): string { if ($contextid === null) { return ''; } context_helper::preload_from_record($context); $context = context_helper::instance_by_id($contextid); return html_writer::link($context->get_url(), $context->get_context_name()); }); // Level. $columns[] = (new column( 'level', new lang_string('contextlevel'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->add_fields("{$contextalias}.contextlevel") ->set_is_sortable(true) ->add_callback(static function(?string $level): string { if ($level === null) { return ''; } return context_helper::get_level_name((int) $level); }); // Path. $columns[] = (new column( 'path', new lang_string('path'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_TEXT) ->add_field("{$contextalias}.path") ->set_is_sortable(true); // Parent (note we select the parent path in SQL, so that aggregation/grouping is on the parent data itself). $columns[] = (new column( 'parent', new lang_string('contextparent'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_TEXT) // The "path" column looks like "/1/2/3", for context ID 3. In order to select/group by the parent context, we // concatenate a trailing slash (to prevent partial matches, e.g. "/1/2/31"), then replace "/3/" with empty string. ->add_field(" REPLACE( " . $DB->sql_concat("{$contextalias}.path", "'/'") . ", " . $DB->sql_concat("'/'", $DB->sql_cast_to_char("{$contextalias}.id"), "'/'") . ", '' )", 'parent' ) // Sorting may not order alphabetically, but will at least group contexts together. ->set_is_sortable(true) ->add_callback(static function (?string $parent): string { // System level (no parent) or null. if ($parent === '' || $parent === null) { return ''; } $contextids = explode('/', $parent); $contextid = (int) array_pop($contextids); return context_helper::instance_by_id($contextid)->get_context_name(); }); return $columns; } /** * Return list of all available filters * * @return filter[] */ protected function get_all_filters(): array { $contextalias = $this->get_table_alias('context'); // Level. $filters[] = (new filter( select::class, 'level', new lang_string('contextlevel'), $this->get_entity_name(), "{$contextalias}.contextlevel" )) ->add_joins($this->get_joins()) ->set_options_callback(static function(): array { $levels = context_helper::get_all_levels(); return array_map(static function(string $levelclass): string { return $levelclass::get_level_name(); }, $levels); }); // Path. $filters[] = (new filter( text::class, 'path', new lang_string('path'), $this->get_entity_name(), "{$contextalias}.path" )) ->add_joins($this->get_joins()); return $filters; } }