<?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_course\reportbuilder\local\entities; use core_reportbuilder\local\entities\base; use core_course\reportbuilder\local\formatters\completion as completion_formatter; use core_reportbuilder\local\filters\boolean_select; use core_reportbuilder\local\filters\date;> use core_reportbuilder\local\helpers\database;use core_reportbuilder\local\helpers\format; use core_reportbuilder\local\report\column; use core_reportbuilder\local\report\filter;> use completion_criteria_completion; use lang_string; > use completion_info; use stdClass; > use html_writer;/** * Course completion entity implementation * * @package core_course * @copyright 2022 David Matamoros <davidmc@moodle.com> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class completion extends base { /** * Database tables that this entity uses and their default aliases * * @return array */ protected function get_default_table_aliases(): array { return [ 'course_completion' => 'ccomp', 'course' => 'c', 'grade_grades' => 'gg', 'grade_items' => 'gi', 'user' => 'u', ]; } /** * 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('coursecompletion', 'completion'); } /** * Initialise the entity * * @return base */ public function initialise(): base { foreach ($this->get_all_columns() as $column) { $this->add_column($column); } // All the filters defined by the entity can also be used as conditions. foreach ($this->get_all_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 { $coursecompletion = $this->get_table_alias('course_completion'); $course = $this->get_table_alias('course'); $grade = $this->get_table_alias('grade_grades'); $gradeitem = $this->get_table_alias('grade_items'); $user = $this->get_table_alias('user'); // Completed column. $columns[] = (new column( 'completed', new lang_string('completed', 'completion'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_BOOLEAN) ->add_field("CASE WHEN {$coursecompletion}.timecompleted > 0 THEN 1 ELSE 0 END", 'completed') ->add_field("{$user}.id", 'userid') ->set_is_sortable(true) ->add_callback(static function(bool $value, stdClass $row): string { if (!$row->userid) { return ''; } return format::boolean_as_text($value);> }); }); > > // Completion criteria column. // Progress percentage column. > $criterias = database::generate_alias(); $columns[] = (new column( > $columns[] = (new column( 'progresspercent', > 'criteria', new lang_string('progress', 'completion'), > new lang_string('criteria', 'core_completion'), $this->get_entity_name() > $this->get_entity_name() )) > )) ->add_joins($this->get_joins()) > ->add_joins($this->get_joins()) ->set_type(column::TYPE_TEXT) > // Determine whether any criteria exist for the course. We also group per course, rather than report each separately. ->add_field("{$course}.id", 'courseid') > ->add_join("LEFT JOIN ( ->add_field("{$user}.id", 'userid') > SELECT DISTINCT course FROM {course_completion_criteria} ->set_is_sortable(false) > ) {$criterias} ON {$criterias}.course = {$course}.id") ->add_callback([completion_formatter::class, 'completion_progress']); > ->set_type(column::TYPE_TEXT) > // Select enough fields to determine user criteria for the course. // Time enrolled. > ->add_field("{$criterias}.course", 'courseid') $columns[] = (new column( > ->add_field("{$course}.enablecompletion") 'timeenrolled', > ->add_field("{$user}.id", 'userid') new lang_string('timeenrolled', 'enrol'), > ->set_disabled_aggregation_all() $this->get_entity_name() > ->add_callback(static function($id, stdClass $record): string { )) > if (!$record->courseid) { ->add_joins($this->get_joins()) > return ''; ->set_type(column::TYPE_TIMESTAMP) > } ->add_field("{$coursecompletion}.timeenrolled") > ->set_is_sortable(true) > $info = new completion_info((object) ['id' => $record->courseid, 'enablecompletion' => $record->enablecompletion]); ->add_callback([format::class, 'userdate']); > if ($info->get_aggregation_method() == COMPLETION_AGGREGATION_ALL) { > $title = get_string('criteriarequiredall', 'core_completion'); // Time started. > } else { $columns[] = (new column( > $title = get_string('criteriarequiredany', 'core_completion'); 'timestarted', > } new lang_string('timestarted', 'enrol'), > $this->get_entity_name() > // Map all completion data to their criteria summaries. )) > $items = array_map(static function(completion_criteria_completion $completion): string { ->add_joins($this->get_joins()) > $criteria = $completion->get_criteria(); ->set_type(column::TYPE_TIMESTAMP) > ->add_field("{$coursecompletion}.timestarted") > return get_string('criteriasummary', 'core_completion', [ ->set_is_sortable(true) > 'type' => $criteria->get_details($completion)['type'], ->add_callback([format::class, 'userdate']); > 'summary' => $criteria->get_title_detailed(), > ]); // Time completed. > }, $info->get_completions((int) $record->userid)); $columns[] = (new column( > 'timecompleted', > return $title . html_writer::alist($items);new lang_string('timecompleted', 'completion'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_TIMESTAMP) ->add_field("{$coursecompletion}.timecompleted") ->set_is_sortable(true) ->add_callback([format::class, 'userdate']); // Time reaggregated. $columns[] = (new column( 'reaggregate', new lang_string('timereaggregated', 'enrol'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_TIMESTAMP) ->add_field("{$coursecompletion}.reaggregate") ->set_is_sortable(true) ->add_callback([format::class, 'userdate']); // Days taking course (days since course start date until completion or until current date if not completed). $currenttime = time(); $columns[] = (new column( 'dayscourse', new lang_string('daystakingcourse', 'course'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_INTEGER) ->add_field("( CASE WHEN {$coursecompletion}.timecompleted > 0 THEN {$coursecompletion}.timecompleted ELSE {$currenttime} END - {$course}.startdate) / " . DAYSECS, 'dayscourse') ->add_field("{$user}.id", 'userid') ->set_is_sortable(true) ->add_callback([completion_formatter::class, 'get_days']); // Days since last completion (days since last enrolment date until completion or until current date if not completed). $columns[] = (new column( 'daysuntilcompletion', new lang_string('daysuntilcompletion', 'completion'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->set_type(column::TYPE_INTEGER) ->add_field("( CASE WHEN {$coursecompletion}.timecompleted > 0 THEN {$coursecompletion}.timecompleted ELSE {$currenttime} END - {$coursecompletion}.timeenrolled) / " . DAYSECS, 'daysuntilcompletion') ->add_field("{$user}.id", 'userid') ->set_is_sortable(true) ->add_callback([completion_formatter::class, 'get_days']); // Student course grade. $columns[] = (new column( 'grade', new lang_string('gradenoun'), $this->get_entity_name() )) ->add_joins($this->get_joins()) ->add_join(" LEFT JOIN {grade_items} {$gradeitem} ON ({$gradeitem}.itemtype = 'course' AND {$course}.id = {$gradeitem}.courseid) ") ->add_join(" LEFT JOIN {grade_grades} {$grade} ON ({$user}.id = {$grade}.userid AND {$gradeitem}.id = {$grade}.itemid) ") ->set_type(column::TYPE_FLOAT) ->add_fields("{$grade}.finalgrade") ->set_is_sortable(true) ->add_callback(function(?float $value): string { if ($value === null) { return ''; } return format_float($value, 2); }); return $columns; } /** * Return list of all available filters * * @return filter[] */ protected function get_all_filters(): array { $coursecompletion = $this->get_table_alias('course_completion'); // Completed status filter. $filters[] = (new filter( boolean_select::class, 'completed', new lang_string('completed', 'completion'), $this->get_entity_name(), "CASE WHEN {$coursecompletion}.timecompleted > 0 THEN 1 ELSE 0 END" )) ->add_joins($this->get_joins()); // Time completed filter. $filters[] = (new filter( date::class, 'timecompleted', new lang_string('timecompleted', 'completion'), $this->get_entity_name(), "{$coursecompletion}.timecompleted" )) ->add_joins($this->get_joins()) ->set_limited_operators([ date::DATE_ANY, date::DATE_NOT_EMPTY, date::DATE_EMPTY, date::DATE_RANGE, date::DATE_LAST, date::DATE_CURRENT, ]); // Time enrolled/started filter and condition. $fields = ['timeenrolled', 'timestarted']; foreach ($fields as $field) { $filters[] = (new filter( date::class, $field, new lang_string($field, 'enrol'), $this->get_entity_name(), "{$coursecompletion}.{$field}" )) ->add_joins($this->get_joins()) ->set_limited_operators([ date::DATE_ANY, date::DATE_NOT_EMPTY, date::DATE_EMPTY, date::DATE_RANGE, date::DATE_LAST, date::DATE_CURRENT, ]); } // Time reaggregated filter and condition. $filters[] = (new filter( date::class, 'reaggregate', new lang_string('timereaggregated', 'enrol'), $this->get_entity_name(), "{$coursecompletion}.reaggregate" )) ->add_joins($this->get_joins()) ->set_limited_operators([ date::DATE_ANY, date::DATE_NOT_EMPTY, date::DATE_EMPTY, date::DATE_RANGE, date::DATE_LAST, date::DATE_CURRENT, ]); return $filters; } }