See Release Notes
Long Term Support Release
<?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 mod_quiz\completion; use context_module; use core_completion\activity_custom_completion; use grade_grade; use grade_item; use quiz; use quiz_access_manager; /** * Activity custom completion subclass for the quiz activity. * * Class for defining mod_quiz's custom completion rules and fetching the completion statuses * of the custom completion rules for a given quiz instance and a user. * * @package mod_quiz * @copyright 2021 Shamim Rezaie <shamim@moodle.com> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class custom_completion extends activity_custom_completion { /** * Check passing grade (or no attempts left) requirement for completion. * * @return bool True if the passing grade (or no attempts left) requirement is disabled or met. */ protected function check_passing_grade_or_all_attempts(): bool { global $CFG; require_once($CFG->libdir . '/gradelib.php'); $completionpassorattempts = $this->cm->customdata['customcompletionrules']['completionpassorattemptsexhausted'];< if (empty($completionpassorattempts['completionpass'])) {> if (empty($completionpassorattempts['completionpassgrade'])) {return true; }< // Check for passing grade. < $item = grade_item::fetch([ < 'courseid' => $this->cm->get_course()->id, < 'itemtype' => 'mod', < 'itemmodule' => 'quiz', < 'iteminstance' => $this->cm->instance, < 'outcomeid' => null < ]); < if ($item) { < $grades = grade_grade::fetch_users_grades($item, [$this->userid], false); < if (!empty($grades[$this->userid]) && $grades[$this->userid]->is_passed($item)) {> if ($this->completionstate && > isset($this->completionstate['passgrade']) && > $this->completionstate['passgrade'] == COMPLETION_COMPLETE_PASS) {return true; }< }// If a passing grade is required and exhausting all available attempts is not accepted for completion, // then this quiz is not complete. if (empty($completionpassorattempts['completionattemptsexhausted'])) { return false; } // Check if all attempts are used up. $attempts = quiz_get_user_attempts($this->cm->instance, $this->userid, 'finished', true); if (!$attempts) { return false; } $lastfinishedattempt = end($attempts); $context = context_module::instance($this->cm->id); $quizobj = quiz::create($this->cm->instance, $this->userid); $accessmanager = new quiz_access_manager( $quizobj, time(), has_capability('mod/quiz:ignoretimelimits', $context, $this->userid, false) ); return $accessmanager->is_finished(count($attempts), $lastfinishedattempt); } /** * Check minimum attempts requirement for completion. * * @return bool True if minimum attempts requirement is disabled or met. */ protected function check_min_attempts() { $minattempts = $this->cm->customdata['customcompletionrules']['completionminattempts']; if (!$minattempts) { return true; } // Check if the user has done enough attempts. $attempts = quiz_get_user_attempts($this->cm->instance, $this->userid, 'finished', true); return $minattempts <= count($attempts); } /** * Fetches the completion state for a given completion rule. * * @param string $rule The completion rule. * @return int The completion state. */ public function get_state(string $rule): int { $this->validate_rule($rule); switch ($rule) { case 'completionpassorattemptsexhausted': $status = static::check_passing_grade_or_all_attempts(); break; case 'completionminattempts': $status = static::check_min_attempts(); break; } return empty($status) ? COMPLETION_INCOMPLETE : COMPLETION_COMPLETE; } /** * Fetch the list of custom completion rules that this module defines. * * @return array */ public static function get_defined_custom_rules(): array { return [ 'completionpassorattemptsexhausted', 'completionminattempts', ]; } /** * Returns an associative array of the descriptions of custom completion rules. * * @return array */ public function get_custom_rule_descriptions(): array { $minattempts = $this->cm->customdata['customcompletionrules']['completionminattempts'] ?? 0;> $description['completionminattempts'] = get_string('completiondetail:minattempts', 'mod_quiz', $minattempts);> // Completion pass grade is now part of core. Only show the following if it's combined with min attempts.$completionpassorattempts = $this->cm->customdata['customcompletionrules']['completionpassorattemptsexhausted'] ?? []; if (!empty($completionpassorattempts['completionattemptsexhausted'])) {< $passorallattemptslabel = get_string('completiondetail:passorexhaust', 'mod_quiz'); < } else { < $passorallattemptslabel = get_string('completiondetail:passgrade', 'mod_quiz');> $description['completionpassorattemptsexhausted'] = get_string('completiondetail:passorexhaust', 'mod_quiz');}< return [ < 'completionpassorattemptsexhausted' => $passorallattemptslabel, < 'completionminattempts' => get_string('completiondetail:minattempts', 'mod_quiz', $minattempts), < ];> return $description;} /** * Returns an array of all completion rules, in the order they should be displayed to users. * * @return array */ public function get_sort_order(): array { return [ 'completionview', 'completionminattempts', 'completionusegrade',> 'completionpassgrade','completionpassorattemptsexhausted', ]; } }