mirror of
https://github.com/moodle/moodle.git
synced 2025-04-21 00:12:56 +02:00
MDL-70935 mod_quiz: Custom completion implementation
This commit is contained in:
parent
7a9f13e37f
commit
a75861f78e
97
mod/quiz/classes/completion/custom_completion.php
Normal file
97
mod/quiz/classes/completion/custom_completion.php
Normal file
@ -0,0 +1,97 @@
|
||||
<?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 core_completion\activity_custom_completion;
|
||||
|
||||
/**
|
||||
* 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 {
|
||||
|
||||
/**
|
||||
* 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 {
|
||||
global $DB;
|
||||
|
||||
$this->validate_rule($rule);
|
||||
|
||||
$quiz = $DB->get_record('quiz', ['id' => $this->cm->instance], '*', MUST_EXIST);
|
||||
|
||||
switch ($rule) {
|
||||
case 'completionpassorattemptsexhausted':
|
||||
$status = quiz_completion_check_passing_grade_or_all_attempts(
|
||||
$this->cm->get_course(),
|
||||
$this->cm,
|
||||
$this->userid,
|
||||
$quiz
|
||||
);
|
||||
break;
|
||||
case 'completionminattempts':
|
||||
$status = quiz_completion_check_min_attempts($this->userid, $quiz);
|
||||
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'];
|
||||
|
||||
$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');
|
||||
}
|
||||
|
||||
return [
|
||||
'completionpassorattemptsexhausted' => $passorallattemptslabel,
|
||||
'completionminattempts' => get_string('completiondetail:minattempts', 'mod_quiz', $minattempts),
|
||||
];
|
||||
}
|
||||
}
|
@ -179,6 +179,9 @@ $string['comment'] = 'Comment';
|
||||
$string['commentorgrade'] = 'Make comment or override grade';
|
||||
$string['comments'] = 'Comments';
|
||||
$string['completedon'] = 'Completed on';
|
||||
$string['completiondetail:minattempts'] = 'Make attempts: {$a}';
|
||||
$string['completiondetail:passgrade'] = 'Receive a pass grade';
|
||||
$string['completiondetail:passorexhaust'] = 'Receive a pass grade or complete all available attempts';
|
||||
$string['completionminattempts'] = 'Minimum number of attempts:';
|
||||
$string['completionminattemptsdesc'] = 'Minimum number of attempts required: {$a}';
|
||||
$string['completionminattemptsgroup'] = 'Require attempts';
|
||||
|
324
mod/quiz/tests/custom_completion_test.php
Normal file
324
mod/quiz/tests/custom_completion_test.php
Normal file
@ -0,0 +1,324 @@
|
||||
<?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;
|
||||
|
||||
use advanced_testcase;
|
||||
use cm_info;
|
||||
use grade_item;
|
||||
use mod_quiz\completion\custom_completion;
|
||||
use question_engine;
|
||||
use quiz;
|
||||
use quiz_attempt;
|
||||
|
||||
defined('MOODLE_INTERNAL') || die();
|
||||
|
||||
global $CFG;
|
||||
require_once($CFG->libdir . '/completionlib.php');
|
||||
|
||||
/**
|
||||
* Class for unit testing mod_quiz/custom_completion.
|
||||
*
|
||||
* @package mod_quiz
|
||||
* @copyright 2021 Shamim Rezaie <shamim@moodle.com>
|
||||
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||||
* @coversDefaultClass \mod_quiz\completion\custom_completion
|
||||
*/
|
||||
class custom_completion_test extends advanced_testcase {
|
||||
|
||||
/**
|
||||
* Setup function for all tests.
|
||||
*
|
||||
* @param array $completionoptions ['nbstudents'] => int, ['qtype'] => string, ['quizoptions'] => array
|
||||
* @return array [$students, $quiz, $cm]
|
||||
*/
|
||||
private function setup_quiz_for_testing_completion(array $completionoptions): array {
|
||||
global $CFG, $DB;
|
||||
|
||||
$this->resetAfterTest(true);
|
||||
|
||||
// Enable completion before creating modules, otherwise the completion data is not written in DB.
|
||||
$CFG->enablecompletion = true;
|
||||
|
||||
// Create a course and students.
|
||||
$studentrole = $DB->get_record('role', ['shortname' => 'student']);
|
||||
$course = $this->getDataGenerator()->create_course(['enablecompletion' => true]);
|
||||
$students = [];
|
||||
for ($i = 0; $i < $completionoptions['nbstudents']; $i++) {
|
||||
$students[$i] = $this->getDataGenerator()->create_user();
|
||||
$this->assertTrue($this->getDataGenerator()->enrol_user($students[$i]->id, $course->id, $studentrole->id));
|
||||
}
|
||||
|
||||
// Make a quiz.
|
||||
$quizgenerator = $this->getDataGenerator()->get_plugin_generator('mod_quiz');
|
||||
$data = array_merge([
|
||||
'course' => $course->id,
|
||||
'grade' => 100.0,
|
||||
'questionsperpage' => 0,
|
||||
'sumgrades' => 1,
|
||||
'completion' => COMPLETION_TRACKING_AUTOMATIC
|
||||
], $completionoptions['quizoptions']);
|
||||
$quiz = $quizgenerator->create_instance($data);
|
||||
$cm = get_coursemodule_from_id('quiz', $quiz->cmid);
|
||||
$cm = cm_info::create($cm);
|
||||
|
||||
// Create a question.
|
||||
$questiongenerator = $this->getDataGenerator()->get_plugin_generator('core_question');
|
||||
|
||||
$cat = $questiongenerator->create_question_category();
|
||||
$question = $questiongenerator->create_question($completionoptions['qtype'], null, ['category' => $cat->id]);
|
||||
quiz_add_quiz_question($question->id, $quiz);
|
||||
|
||||
// Set grade to pass.
|
||||
$item = grade_item::fetch(['courseid' => $course->id, 'itemtype' => 'mod', 'itemmodule' => 'quiz',
|
||||
'iteminstance' => $quiz->id, 'outcomeid' => null]);
|
||||
$item->gradepass = 80;
|
||||
$item->update();
|
||||
|
||||
return [
|
||||
$students,
|
||||
$quiz,
|
||||
$cm
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper function for tests.
|
||||
* Starts an attempt, processes responses and finishes the attempt.
|
||||
*
|
||||
* @param array $attemptoptions ['quiz'] => object, ['student'] => object, ['tosubmit'] => array, ['attemptnumber'] => int
|
||||
*/
|
||||
private function do_attempt_quiz(array $attemptoptions) {
|
||||
$quizobj = quiz::create($attemptoptions['quiz']->id);
|
||||
|
||||
// Start the passing attempt.
|
||||
$quba = question_engine::make_questions_usage_by_activity('mod_quiz', $quizobj->get_context());
|
||||
$quba->set_preferred_behaviour($quizobj->get_quiz()->preferredbehaviour);
|
||||
|
||||
$timenow = time();
|
||||
$attempt = quiz_create_attempt($quizobj, $attemptoptions['attemptnumber'], false, $timenow, false,
|
||||
$attemptoptions['student']->id);
|
||||
quiz_start_new_attempt($quizobj, $quba, $attempt, $attemptoptions['attemptnumber'], $timenow);
|
||||
quiz_attempt_save_started($quizobj, $quba, $attempt);
|
||||
|
||||
// Process responses from the student.
|
||||
$attemptobj = quiz_attempt::create($attempt->id);
|
||||
$attemptobj->process_submitted_actions($timenow, false, $attemptoptions['tosubmit']);
|
||||
|
||||
// Finish the attempt.
|
||||
$attemptobj = quiz_attempt::create($attempt->id);
|
||||
$this->assertTrue($attemptobj->has_response_to_at_least_one_graded_question());
|
||||
$attemptobj->process_finish($timenow, false);
|
||||
}
|
||||
|
||||
/**
|
||||
* Test checking the completion state of a quiz.
|
||||
* The quiz requires a passing grade to be completed.
|
||||
*
|
||||
* @covers ::get_state
|
||||
* @covers ::get_custom_rule_descriptions
|
||||
*/
|
||||
public function test_completionpass() {
|
||||
list($students, $quiz, $cm) = $this->setup_quiz_for_testing_completion([
|
||||
'nbstudents' => 2,
|
||||
'qtype' => 'numerical',
|
||||
'quizoptions' => [
|
||||
'completionusegrade' => 1,
|
||||
'completionpass' => 1
|
||||
]
|
||||
]);
|
||||
|
||||
list($passstudent, $failstudent) = $students;
|
||||
|
||||
// Do a passing attempt.
|
||||
$this->do_attempt_quiz([
|
||||
'quiz' => $quiz,
|
||||
'student' => $passstudent,
|
||||
'attemptnumber' => 1,
|
||||
'tosubmit' => [1 => ['answer' => '3.14']]
|
||||
]);
|
||||
|
||||
// Check the results.
|
||||
$customcompletion = new custom_completion($cm, (int) $passstudent->id);
|
||||
$this->assertArrayHasKey('completionpassorattemptsexhausted', $cm->customdata['customcompletionrules']);
|
||||
$this->assertEquals(COMPLETION_COMPLETE, $customcompletion->get_state('completionpassorattemptsexhausted'));
|
||||
$this->assertEquals(
|
||||
'Receive a pass grade',
|
||||
$customcompletion->get_custom_rule_descriptions()['completionpassorattemptsexhausted']
|
||||
);
|
||||
|
||||
// Do a failing attempt.
|
||||
$this->do_attempt_quiz([
|
||||
'quiz' => $quiz,
|
||||
'student' => $failstudent,
|
||||
'attemptnumber' => 1,
|
||||
'tosubmit' => [1 => ['answer' => '0']]
|
||||
]);
|
||||
|
||||
// Check the results.
|
||||
$customcompletion = new custom_completion($cm, (int) $failstudent->id);
|
||||
$this->assertArrayHasKey('completionpassorattemptsexhausted', $cm->customdata['customcompletionrules']);
|
||||
$this->assertEquals(COMPLETION_INCOMPLETE, $customcompletion->get_state('completionpassorattemptsexhausted'));
|
||||
$this->assertEquals(
|
||||
'Receive a pass grade',
|
||||
$customcompletion->get_custom_rule_descriptions()['completionpassorattemptsexhausted']
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Test checking the completion state of a quiz.
|
||||
* To be completed, this quiz requires either a passing grade or for all attempts to be used up.
|
||||
*
|
||||
* @covers ::get_state
|
||||
* @covers ::get_custom_rule_descriptions
|
||||
*/
|
||||
public function test_completionexhausted() {
|
||||
list($students, $quiz, $cm) = $this->setup_quiz_for_testing_completion([
|
||||
'nbstudents' => 2,
|
||||
'qtype' => 'numerical',
|
||||
'quizoptions' => [
|
||||
'attempts' => 2,
|
||||
'completionusegrade' => 1,
|
||||
'completionpass' => 1,
|
||||
'completionattemptsexhausted' => 1
|
||||
]
|
||||
]);
|
||||
|
||||
list($passstudent, $exhauststudent) = $students;
|
||||
|
||||
// Start a passing attempt.
|
||||
$this->do_attempt_quiz([
|
||||
'quiz' => $quiz,
|
||||
'student' => $passstudent,
|
||||
'attemptnumber' => 1,
|
||||
'tosubmit' => [1 => ['answer' => '3.14']]
|
||||
]);
|
||||
|
||||
// Check the results. Quiz is completed by $passstudent because of passing grade.
|
||||
$customcompletion = new custom_completion($cm, (int) $passstudent->id);
|
||||
$this->assertArrayHasKey('completionpassorattemptsexhausted', $cm->customdata['customcompletionrules']);
|
||||
$this->assertEquals(COMPLETION_COMPLETE, $customcompletion->get_state('completionpassorattemptsexhausted'));
|
||||
$this->assertEquals(
|
||||
'Receive a pass grade or complete all available attempts',
|
||||
$customcompletion->get_custom_rule_descriptions()['completionpassorattemptsexhausted']
|
||||
);
|
||||
|
||||
// Do a failing attempt.
|
||||
$this->do_attempt_quiz([
|
||||
'quiz' => $quiz,
|
||||
'student' => $exhauststudent,
|
||||
'attemptnumber' => 1,
|
||||
'tosubmit' => [1 => ['answer' => '0']]
|
||||
]);
|
||||
|
||||
// Check the results. Quiz is not completed by $exhauststudent yet because of failing grade and of remaining attempts.
|
||||
$customcompletion = new custom_completion($cm, (int) $exhauststudent->id);
|
||||
$this->assertArrayHasKey('completionpassorattemptsexhausted', $cm->customdata['customcompletionrules']);
|
||||
$this->assertEquals(COMPLETION_INCOMPLETE, $customcompletion->get_state('completionpassorattemptsexhausted'));
|
||||
$this->assertEquals(
|
||||
'Receive a pass grade or complete all available attempts',
|
||||
$customcompletion->get_custom_rule_descriptions()['completionpassorattemptsexhausted']
|
||||
);
|
||||
|
||||
// Do a second failing attempt.
|
||||
$this->do_attempt_quiz([
|
||||
'quiz' => $quiz,
|
||||
'student' => $exhauststudent,
|
||||
'attemptnumber' => 2,
|
||||
'tosubmit' => [1 => ['answer' => '0']]
|
||||
]);
|
||||
|
||||
// Check the results. Quiz is completed by $exhauststudent because there are no remaining attempts.
|
||||
$customcompletion = new custom_completion($cm, (int) $exhauststudent->id);
|
||||
$this->assertArrayHasKey('completionpassorattemptsexhausted', $cm->customdata['customcompletionrules']);
|
||||
$this->assertEquals(COMPLETION_COMPLETE, $customcompletion->get_state('completionpassorattemptsexhausted'));
|
||||
$this->assertEquals(
|
||||
'Receive a pass grade or complete all available attempts',
|
||||
$customcompletion->get_custom_rule_descriptions()['completionpassorattemptsexhausted']
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Test checking the completion state of a quiz.
|
||||
* To be completed, this quiz requires a minimum number of attempts.
|
||||
*
|
||||
* @covers ::get_state
|
||||
* @covers ::get_custom_rule_descriptions
|
||||
*/
|
||||
public function test_completionminattempts() {
|
||||
list($students, $quiz, $cm) = $this->setup_quiz_for_testing_completion([
|
||||
'nbstudents' => 1,
|
||||
'qtype' => 'essay',
|
||||
'quizoptions' => [
|
||||
'completionminattemptsenabled' => 1,
|
||||
'completionminattempts' => 2
|
||||
]
|
||||
]);
|
||||
|
||||
list($student) = $students;
|
||||
|
||||
// Do a first attempt.
|
||||
$this->do_attempt_quiz([
|
||||
'quiz' => $quiz,
|
||||
'student' => $student,
|
||||
'attemptnumber' => 1,
|
||||
'tosubmit' => [1 => ['answer' => 'Lorem ipsum.', 'answerformat' => '1']]
|
||||
]);
|
||||
|
||||
// Check the results. Quiz is not completed yet because only one attempt was done.
|
||||
$customcompletion = new custom_completion($cm, (int) $student->id);
|
||||
$this->assertArrayHasKey('completionminattempts', $cm->customdata['customcompletionrules']);
|
||||
$this->assertEquals(COMPLETION_INCOMPLETE, $customcompletion->get_state('completionminattempts'));
|
||||
$this->assertEquals(
|
||||
'Make attempts: 2',
|
||||
$customcompletion->get_custom_rule_descriptions()['completionminattempts']
|
||||
);
|
||||
|
||||
// Do a second attempt.
|
||||
$this->do_attempt_quiz([
|
||||
'quiz' => $quiz,
|
||||
'student' => $student,
|
||||
'attemptnumber' => 2,
|
||||
'tosubmit' => [1 => ['answer' => 'Lorem ipsum.', 'answerformat' => '1']]
|
||||
]);
|
||||
|
||||
// Check the results. Quiz is completed by $student because two attempts were done.
|
||||
$customcompletion = new custom_completion($cm, (int) $student->id);
|
||||
$this->assertArrayHasKey('completionminattempts', $cm->customdata['customcompletionrules']);
|
||||
$this->assertEquals(COMPLETION_COMPLETE, $customcompletion->get_state('completionminattempts'));
|
||||
$this->assertEquals(
|
||||
'Make attempts: 2',
|
||||
$customcompletion->get_custom_rule_descriptions()['completionminattempts']
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Test for get_defined_custom_rules().
|
||||
*
|
||||
* @covers ::get_defined_custom_rules
|
||||
*/
|
||||
public function test_get_defined_custom_rules() {
|
||||
$rules = custom_completion::get_defined_custom_rules();
|
||||
$this->assertCount(2, $rules);
|
||||
$this->assertEquals(
|
||||
['completionpassorattemptsexhausted', 'completionminattempts'],
|
||||
$rules
|
||||
);
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user