Differences Between: [Versions 402 and 403]
1 <?php 2 // This file is part of Moodle - http://moodle.org/ 3 // 4 // Moodle is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // Moodle is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU General Public License for more details. 13 // 14 // You should have received a copy of the GNU General Public License 15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>. 16 17 namespace mod_quiz; 18 19 use action_link; 20 use block_contents; 21 use cm_info; 22 use coding_exception; 23 use context_module; 24 use Exception; 25 use html_writer; 26 use mod_quiz\output\links_to_other_attempts; 27 use mod_quiz\output\renderer; 28 use mod_quiz\question\bank\qbank_helper; 29 use mod_quiz\question\display_options; 30 use moodle_exception; 31 use moodle_url; 32 use popup_action; 33 use qtype_description_question; 34 use question_attempt; 35 use question_bank; 36 use question_display_options; 37 use question_engine; 38 use question_out_of_sequence_exception; 39 use question_state; 40 use question_usage_by_activity; 41 use stdClass; 42 43 /** 44 * This class represents one user's attempt at a particular quiz. 45 * 46 * @package mod_quiz 47 * @copyright 2008 Tim Hunt 48 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later 49 */ 50 class quiz_attempt { 51 52 /** @var string to identify the in progress state. */ 53 const IN_PROGRESS = 'inprogress'; 54 /** @var string to identify the overdue state. */ 55 const OVERDUE = 'overdue'; 56 /** @var string to identify the finished state. */ 57 const FINISHED = 'finished'; 58 /** @var string to identify the abandoned state. */ 59 const ABANDONED = 'abandoned'; 60 61 /** @var int maximum number of slots in the quiz for the review page to default to show all. */ 62 const MAX_SLOTS_FOR_DEFAULT_REVIEW_SHOW_ALL = 50; 63 64 /** @var quiz_settings object containing the quiz settings. */ 65 protected $quizobj; 66 67 /** @var stdClass the quiz_attempts row. */ 68 protected $attempt; 69 70 /** @var question_usage_by_activity the question usage for this quiz attempt. */ 71 protected $quba; 72 73 /** 74 * @var array of slot information. These objects contain ->slot (int), 75 * ->requireprevious (bool), ->questionids (int) the original question for random questions, 76 * ->firstinsection (bool), ->section (stdClass from $this->sections). 77 * This does not contain page - get that from {@see get_question_page()} - 78 * or maxmark - get that from $this->quba. 79 */ 80 protected $slots; 81 82 /** @var array of quiz_sections rows, with a ->lastslot field added. */ 83 protected $sections; 84 85 /** @var array page no => array of slot numbers on the page in order. */ 86 protected $pagelayout; 87 88 /** @var array slot => displayed question number for this slot. (E.g. 1, 2, 3 or 'i'.) */ 89 protected $questionnumbers; 90 91 /** @var array slot => page number for this slot. */ 92 protected $questionpages; 93 94 /** @var display_options cache for the appropriate review options. */ 95 protected $reviewoptions = null; 96 97 // Constructor =============================================================. 98 /** 99 * Constructor assuming we already have the necessary data loaded. 100 * 101 * @param stdClass $attempt the row of the quiz_attempts table. 102 * @param stdClass $quiz the quiz object for this attempt and user. 103 * @param stdClass|cm_info $cm the course_module object for this quiz. 104 * @param stdClass $course the row from the course table for the course we belong to. 105 * @param bool $loadquestions (optional) if true, the default, load all the details 106 * of the state of each question. Else just set up the basic details of the attempt. 107 */ 108 public function __construct($attempt, $quiz, $cm, $course, $loadquestions = true) { 109 $this->attempt = $attempt; 110 $this->quizobj = new quiz_settings($quiz, $cm, $course); 111 112 if ($loadquestions) { 113 $this->load_questions(); 114 } 115 } 116 117 /** 118 * Used by {create()} and {create_from_usage_id()}. 119 * 120 * @param array $conditions passed to $DB->get_record('quiz_attempts', $conditions). 121 * @return quiz_attempt the desired instance of this class. 122 */ 123 protected static function create_helper($conditions) { 124 global $DB; 125 126 $attempt = $DB->get_record('quiz_attempts', $conditions, '*', MUST_EXIST); 127 $quiz = access_manager::load_quiz_and_settings($attempt->quiz); 128 $course = get_course($quiz->course); 129 $cm = get_coursemodule_from_instance('quiz', $quiz->id, $course->id, false, MUST_EXIST); 130 131 // Update quiz with override information. 132 $quiz = quiz_update_effective_access($quiz, $attempt->userid); 133 134 return new quiz_attempt($attempt, $quiz, $cm, $course); 135 } 136 137 /** 138 * Static function to create a new quiz_attempt object given an attemptid. 139 * 140 * @param int $attemptid the attempt id. 141 * @return quiz_attempt the new quiz_attempt object 142 */ 143 public static function create($attemptid) { 144 return self::create_helper(['id' => $attemptid]); 145 } 146 147 /** 148 * Static function to create a new quiz_attempt object given a usage id. 149 * 150 * @param int $usageid the attempt usage id. 151 * @return quiz_attempt the new quiz_attempt object 152 */ 153 public static function create_from_usage_id($usageid) { 154 return self::create_helper(['uniqueid' => $usageid]); 155 } 156 157 /** 158 * Get a human-readable name for one of the quiz attempt states. 159 * 160 * @param string $state one of the state constants like IN_PROGRESS. 161 * @return string the human-readable state name. 162 */ 163 public static function state_name($state) { 164 return quiz_attempt_state_name($state); 165 } 166 167 /** 168 * This method can be called later if the object was constructed with $loadquestions = false. 169 */ 170 public function load_questions() { 171 global $DB; 172 173 if (isset($this->quba)) { 174 throw new coding_exception('This quiz attempt has already had the questions loaded.'); 175 } 176 177 $this->quba = question_engine::load_questions_usage_by_activity($this->attempt->uniqueid); 178 $this->slots = $DB->get_records('quiz_slots', 179 ['quizid' => $this->get_quizid()], 'slot', 'slot, id, requireprevious, displaynumber'); 180 $this->sections = array_values($DB->get_records('quiz_sections', 181 ['quizid' => $this->get_quizid()], 'firstslot')); 182 183 $this->link_sections_and_slots(); 184 $this->determine_layout(); 185 $this->number_questions(); 186 } 187 188 /** 189 * Preload all attempt step users to show in Response history. 190 */ 191 public function preload_all_attempt_step_users(): void { 192 $this->quba->preload_all_step_users(); 193 } 194 195 /** 196 * Let each slot know which section it is part of. 197 */ 198 protected function link_sections_and_slots() { 199 foreach ($this->sections as $i => $section) { 200 if (isset($this->sections[$i + 1])) { 201 $section->lastslot = $this->sections[$i + 1]->firstslot - 1; 202 } else { 203 $section->lastslot = count($this->slots); 204 } 205 for ($slot = $section->firstslot; $slot <= $section->lastslot; $slot += 1) { 206 $this->slots[$slot]->section = $section; 207 } 208 } 209 } 210 211 /** 212 * Parse attempt->layout to populate the other arrays that represent the layout. 213 */ 214 protected function determine_layout() { 215 216 // Break up the layout string into pages. 217 $pagelayouts = explode(',0', $this->attempt->layout); 218 219 // Strip off any empty last page (normally there is one). 220 if (end($pagelayouts) == '') { 221 array_pop($pagelayouts); 222 } 223 224 // File the ids into the arrays. 225 // Tracking which is the first slot in each section in this attempt is 226 // trickier than you might guess, since the slots in this section 227 // may be shuffled, so $section->firstslot (the lowest numbered slot in 228 // the section) may not be the first one. 229 $unseensections = $this->sections; 230 $this->pagelayout = []; 231 foreach ($pagelayouts as $page => $pagelayout) { 232 $pagelayout = trim($pagelayout, ','); 233 if ($pagelayout == '') { 234 continue; 235 } 236 $this->pagelayout[$page] = explode(',', $pagelayout); 237 foreach ($this->pagelayout[$page] as $slot) { 238 $sectionkey = array_search($this->slots[$slot]->section, $unseensections); 239 if ($sectionkey !== false) { 240 $this->slots[$slot]->firstinsection = true; 241 unset($unseensections[$sectionkey]); 242 } else { 243 $this->slots[$slot]->firstinsection = false; 244 } 245 } 246 } 247 } 248 249 /** 250 * Work out the number to display for each question/slot. 251 */ 252 protected function number_questions() { 253 $number = 1; 254 foreach ($this->pagelayout as $page => $slots) { 255 foreach ($slots as $slot) { 256 if ($length = $this->is_real_question($slot)) { 257 // Whether question numbering is customised or is numeric and automatically incremented. 258 if ($this->slots[$slot]->displaynumber !== null && $this->slots[$slot]->displaynumber !== '' && 259 !$this->slots[$slot]->section->shufflequestions) { 260 $this->questionnumbers[$slot] = $this->slots[$slot]->displaynumber; 261 } else { 262 $this->questionnumbers[$slot] = (string) $number; 263 } 264 $number += $length; 265 } else { 266 $this->questionnumbers[$slot] = get_string('infoshort', 'quiz'); 267 } 268 $this->questionpages[$slot] = $page; 269 } 270 } 271 } 272 273 /** 274 * If the given page number is out of range (before the first page, or after 275 * the last page, change it to be within range). 276 * 277 * @param int $page the requested page number. 278 * @return int a safe page number to use. 279 */ 280 public function force_page_number_into_range($page) { 281 return min(max($page, 0), count($this->pagelayout) - 1); 282 } 283 284 // Simple getters ==========================================================. 285 286 /** 287 * Get the raw quiz settings object. 288 * 289 * @return stdClass 290 */ 291 public function get_quiz() { 292 return $this->quizobj->get_quiz(); 293 } 294 295 /** 296 * Get the {@see seb_quiz_settings} object for this quiz. 297 * 298 * @return quiz_settings 299 */ 300 public function get_quizobj() { 301 return $this->quizobj; 302 } 303 304 /** 305 * Git the id of the course this quiz belongs to. 306 * 307 * @return int the course id. 308 */ 309 public function get_courseid() { 310 return $this->quizobj->get_courseid(); 311 } 312 313 /** 314 * Get the course settings object. 315 * 316 * @return stdClass the course settings object. 317 */ 318 public function get_course() { 319 return $this->quizobj->get_course(); 320 } 321 322 /** 323 * Get the quiz id. 324 * 325 * @return int the quiz id. 326 */ 327 public function get_quizid() { 328 return $this->quizobj->get_quizid(); 329 } 330 331 /** 332 * Get the name of this quiz. 333 * 334 * @return string Quiz name, directly from the database (format_string must be called before output). 335 */ 336 public function get_quiz_name() { 337 return $this->quizobj->get_quiz_name(); 338 } 339 340 /** 341 * Get the quiz navigation method. 342 * 343 * @return int QUIZ_NAVMETHOD_FREE or QUIZ_NAVMETHOD_SEQ. 344 */ 345 public function get_navigation_method() { 346 return $this->quizobj->get_navigation_method(); 347 } 348 349 /** 350 * Get the course_module for this quiz. 351 * 352 * @return stdClass|cm_info the course_module object. 353 */ 354 public function get_cm() { 355 return $this->quizobj->get_cm(); 356 } 357 358 /** 359 * Get the course-module id. 360 * 361 * @return int the course_module id. 362 */ 363 public function get_cmid() { 364 return $this->quizobj->get_cmid(); 365 } 366 367 /** 368 * Get the quiz context. 369 * 370 * @return context_module the context of the quiz this attempt belongs to. 371 */ 372 public function get_context(): context_module { 373 return $this->quizobj->get_context(); 374 } 375 376 /** 377 * Is the current user is someone who previews the quiz, rather than attempting it? 378 * 379 * @return bool true user is a preview user. False, if they can do real attempts. 380 */ 381 public function is_preview_user() { 382 return $this->quizobj->is_preview_user(); 383 } 384 385 /** 386 * Get the number of attempts the user is allowed at this quiz. 387 * 388 * @return int the number of attempts allowed at this quiz (0 = infinite). 389 */ 390 public function get_num_attempts_allowed() { 391 return $this->quizobj->get_num_attempts_allowed(); 392 } 393 394 /** 395 * Get the number of quizzes in the quiz attempt. 396 * 397 * @return int number pages. 398 */ 399 public function get_num_pages() { 400 return count($this->pagelayout); 401 } 402 403 /** 404 * Get the access_manager for this quiz attempt. 405 * 406 * @param int $timenow the current time as a unix timestamp. 407 * @return access_manager and instance of the access_manager class 408 * for this quiz at this time. 409 */ 410 public function get_access_manager($timenow) { 411 return $this->quizobj->get_access_manager($timenow); 412 } 413 414 /** 415 * Get the id of this attempt. 416 * 417 * @return int the attempt id. 418 */ 419 public function get_attemptid() { 420 return $this->attempt->id; 421 } 422 423 /** 424 * Get the question-usage id corresponding to this quiz attempt. 425 * 426 * @return int the attempt unique id. 427 */ 428 public function get_uniqueid() { 429 return $this->attempt->uniqueid; 430 } 431 432 /** 433 * Get the raw quiz attempt object. 434 * 435 * @return stdClass the row from the quiz_attempts table. 436 */ 437 public function get_attempt() { 438 return $this->attempt; 439 } 440 441 /** 442 * Get the attempt number. 443 * 444 * @return int the number of this attempt (is it this user's first, second, ... attempt). 445 */ 446 public function get_attempt_number() { 447 return $this->attempt->attempt; 448 } 449 450 /** 451 * Get the state of this attempt. 452 * 453 * @return string {@see IN_PROGRESS}, {@see FINISHED}, {@see OVERDUE} or {@see ABANDONED}. 454 */ 455 public function get_state() { 456 return $this->attempt->state; 457 } 458 459 /** 460 * Get the id of the user this attempt belongs to. 461 * @return int user id. 462 */ 463 public function get_userid() { 464 return $this->attempt->userid; 465 } 466 467 /** 468 * Get the current page of the attempt 469 * @return int page number. 470 */ 471 public function get_currentpage() { 472 return $this->attempt->currentpage; 473 } 474 475 /** 476 * Get the total number of marks that the user had scored on all the questions. 477 * 478 * @return float 479 */ 480 public function get_sum_marks() { 481 return $this->attempt->sumgrades; 482 } 483 484 /** 485 * Has this attempt been finished? 486 * 487 * States {@see FINISHED} and {@see ABANDONED} are both considered finished in this state. 488 * Other states are not. 489 * 490 * @return bool 491 */ 492 public function is_finished() { 493 return $this->attempt->state == self::FINISHED || $this->attempt->state == self::ABANDONED; 494 } 495 496 /** 497 * Is this attempt a preview? 498 * 499 * @return bool true if it is. 500 */ 501 public function is_preview() { 502 return $this->attempt->preview; 503 } 504 505 /** 506 * Does this attempt belong to the current user? 507 * 508 * @return bool true => own attempt/preview. false => reviewing someone else's. 509 */ 510 public function is_own_attempt() { 511 global $USER; 512 return $this->attempt->userid == $USER->id; 513 } 514 515 /** 516 * Is this attempt is a preview belonging to the current user. 517 * 518 * @return bool true if it is. 519 */ 520 public function is_own_preview() { 521 return $this->is_own_attempt() && 522 $this->is_preview_user() && $this->attempt->preview; 523 } 524 525 /** 526 * Is the current user allowed to review this attempt. This applies when 527 * {@see is_own_attempt()} returns false. 528 * 529 * @return bool whether the review should be allowed. 530 */ 531 public function is_review_allowed() { 532 if (!$this->has_capability('mod/quiz:viewreports')) { 533 return false; 534 } 535 536 $cm = $this->get_cm(); 537 if ($this->has_capability('moodle/site:accessallgroups') || 538 groups_get_activity_groupmode($cm) != SEPARATEGROUPS) { 539 return true; 540 } 541 542 // Check the users have at least one group in common. 543 $teachersgroups = groups_get_activity_allowed_groups($cm); 544 $studentsgroups = groups_get_all_groups( 545 $cm->course, $this->attempt->userid, $cm->groupingid); 546 return $teachersgroups && $studentsgroups && 547 array_intersect(array_keys($teachersgroups), array_keys($studentsgroups)); 548 } 549 550 /** 551 * Has the student, in this attempt, engaged with the quiz in a non-trivial way? 552 * 553 * That is, is there any question worth a non-zero number of marks, where 554 * the student has made some response that we have saved? 555 * 556 * @return bool true if we have saved a response for at least one graded question. 557 */ 558 public function has_response_to_at_least_one_graded_question() { 559 foreach ($this->quba->get_attempt_iterator() as $qa) { 560 if ($qa->get_max_mark() == 0) { 561 continue; 562 } 563 if ($qa->get_num_steps() > 1) { 564 return true; 565 } 566 } 567 return false; 568 } 569 570 /** 571 * Do any questions in this attempt need to be graded manually? 572 * 573 * @return bool True if we have at least one question still needs manual grading. 574 */ 575 public function requires_manual_grading(): bool { 576 return $this->quba->get_total_mark() === null; 577 } 578 579 /** 580 * Get extra summary information about this attempt. 581 * 582 * Some behaviours may be able to provide interesting summary information 583 * about the attempt as a whole, and this method provides access to that data. 584 * To see how this works, try setting a quiz to one of the CBM behaviours, 585 * and then look at the extra information displayed at the top of the quiz 586 * review page once you have submitted an attempt. 587 * 588 * In the return value, the array keys are identifiers of the form 589 * qbehaviour_behaviourname_meaningfullkey. For qbehaviour_deferredcbm_highsummary. 590 * The values are arrays with two items, title and content. Each of these 591 * will be either a string, or a renderable. 592 * 593 * @param question_display_options $options the display options for this quiz attempt at this time. 594 * @return array as described above. 595 */ 596 public function get_additional_summary_data(question_display_options $options) { 597 return $this->quba->get_summary_information($options); 598 } 599 600 /** 601 * Get the overall feedback corresponding to a particular mark. 602 * 603 * @param number $grade a particular grade. 604 * @return string the feedback. 605 */ 606 public function get_overall_feedback($grade) { 607 return quiz_feedback_for_grade($grade, $this->get_quiz(), 608 $this->quizobj->get_context()); 609 } 610 611 /** 612 * Wrapper round the has_capability function that automatically passes in the quiz context. 613 * 614 * @param string $capability the name of the capability to check. For example mod/forum:view. 615 * @param int|null $userid A user id. If null checks the permissions of the current user. 616 * @param bool $doanything If false, ignore effect of admin role assignment. 617 * @return boolean true if the user has this capability, otherwise false. 618 */ 619 public function has_capability($capability, $userid = null, $doanything = true) { 620 return $this->quizobj->has_capability($capability, $userid, $doanything); 621 } 622 623 /** 624 * Wrapper round the require_capability function that automatically passes in the quiz context. 625 * 626 * @param string $capability the name of the capability to check. For example mod/forum:view. 627 * @param int|null $userid A user id. If null checks the permissions of the current user. 628 * @param bool $doanything If false, ignore effect of admin role assignment. 629 */ 630 public function require_capability($capability, $userid = null, $doanything = true) { 631 $this->quizobj->require_capability($capability, $userid, $doanything); 632 } 633 634 /** 635 * Check the appropriate capability to see whether this user may review their own attempt. 636 * If not, prints an error. 637 */ 638 public function check_review_capability() { 639 if ($this->get_attempt_state() == display_options::IMMEDIATELY_AFTER) { 640 $capability = 'mod/quiz:attempt'; 641 } else { 642 $capability = 'mod/quiz:reviewmyattempts'; 643 } 644 645 // These next tests are in a slightly funny order. The point is that the 646 // common and most performance-critical case is students attempting a quiz, 647 // so we want to check that permission first. 648 649 if ($this->has_capability($capability)) { 650 // User has the permission that lets you do the quiz as a student. Fine. 651 return; 652 } 653 654 if ($this->has_capability('mod/quiz:viewreports') || 655 $this->has_capability('mod/quiz:preview')) { 656 // User has the permission that lets teachers review. Fine. 657 return; 658 } 659 660 // They should not be here. Trigger the standard no-permission error 661 // but using the name of the student capability. 662 // We know this will fail. We just want the standard exception thrown. 663 $this->require_capability($capability); 664 } 665 666 /** 667 * Checks whether a user may navigate to a particular slot. 668 * 669 * @param int $slot the target slot (currently does not affect the answer). 670 * @return bool true if the navigation should be allowed. 671 */ 672 public function can_navigate_to($slot) { 673 if ($this->attempt->state == self::OVERDUE) { 674 // When the attempt is overdue, students can only see the 675 // attempt summary page and cannot navigate anywhere else. 676 return false; 677 } 678 679 return $this->get_navigation_method() == QUIZ_NAVMETHOD_FREE; 680 } 681 682 /** 683 * Get where we are time-wise in relation to this attempt and the quiz settings. 684 * 685 * @return int one of {@see display_options::DURING}, {@see display_options::IMMEDIATELY_AFTER}, 686 * {@see display_options::LATER_WHILE_OPEN} or {@see display_options::AFTER_CLOSE}. 687 */ 688 public function get_attempt_state() { 689 return quiz_attempt_state($this->get_quiz(), $this->attempt); 690 } 691 692 /** 693 * Wrapper that the correct display_options for this quiz at the 694 * moment. 695 * 696 * @param bool $reviewing true for options when reviewing, false for when attempting. 697 * @return question_display_options the render options for this user on this attempt. 698 */ 699 public function get_display_options($reviewing) { 700 if ($reviewing) { 701 if (is_null($this->reviewoptions)) { 702 $this->reviewoptions = quiz_get_review_options($this->get_quiz(), 703 $this->attempt, $this->quizobj->get_context()); 704 if ($this->is_own_preview()) { 705 // It should always be possible for a teacher to review their 706 // own preview irrespective of the review options settings. 707 $this->reviewoptions->attempt = true; 708 } 709 } 710 return $this->reviewoptions; 711 712 } else { 713 $options = display_options::make_from_quiz($this->get_quiz(), 714 display_options::DURING); 715 $options->flags = quiz_get_flag_option($this->attempt, $this->quizobj->get_context()); 716 return $options; 717 } 718 } 719 720 /** 721 * Wrapper that the correct display_options for this quiz at the 722 * moment. 723 * 724 * @param bool $reviewing true for review page, else attempt page. 725 * @param int $slot which question is being displayed. 726 * @param moodle_url $thispageurl to return to after the editing form is 727 * submitted or cancelled. If null, no edit link will be generated. 728 * 729 * @return question_display_options the render options for this user on this 730 * attempt, with extra info to generate an edit link, if applicable. 731 */ 732 public function get_display_options_with_edit_link($reviewing, $slot, $thispageurl) { 733 $options = clone($this->get_display_options($reviewing)); 734 735 if (!$thispageurl) { 736 return $options; 737 } 738 739 if (!($reviewing || $this->is_preview())) { 740 return $options; 741 } 742 743 $question = $this->quba->get_question($slot, false); 744 if (!question_has_capability_on($question, 'edit', $question->category)) { 745 return $options; 746 } 747 748 $options->editquestionparams['cmid'] = $this->get_cmid(); 749 $options->editquestionparams['returnurl'] = $thispageurl; 750 751 return $options; 752 } 753 754 /** 755 * Is a particular page the last one in the quiz? 756 * 757 * @param int $page a page number 758 * @return bool true if that is the last page of the quiz. 759 */ 760 public function is_last_page($page) { 761 return $page == count($this->pagelayout) - 1; 762 } 763 764 /** 765 * Return the list of slot numbers for either a given page of the quiz, or for the 766 * whole quiz. 767 * 768 * @param mixed $page string 'all' or integer page number. 769 * @return array the requested list of slot numbers. 770 */ 771 public function get_slots($page = 'all') { 772 if ($page === 'all') { 773 $numbers = []; 774 foreach ($this->pagelayout as $numbersonpage) { 775 $numbers = array_merge($numbers, $numbersonpage); 776 } 777 return $numbers; 778 } else { 779 return $this->pagelayout[$page]; 780 } 781 } 782 783 /** 784 * Return the list of slot numbers for either a given page of the quiz, or for the 785 * whole quiz. 786 * 787 * @param mixed $page string 'all' or integer page number. 788 * @return array the requested list of slot numbers. 789 */ 790 public function get_active_slots($page = 'all') { 791 $activeslots = []; 792 foreach ($this->get_slots($page) as $slot) { 793 if (!$this->is_blocked_by_previous_question($slot)) { 794 $activeslots[] = $slot; 795 } 796 } 797 return $activeslots; 798 } 799 800 /** 801 * Helper method for unit tests. Get the underlying question usage object. 802 * 803 * @return question_usage_by_activity the usage. 804 */ 805 public function get_question_usage() { 806 if (!(PHPUNIT_TEST || defined('BEHAT_TEST'))) { 807 throw new coding_exception('get_question_usage is only for use in unit tests. ' . 808 'For other operations, use the quiz_attempt api, or extend it properly.'); 809 } 810 return $this->quba; 811 } 812 813 /** 814 * Get the question_attempt object for a particular question in this attempt. 815 * 816 * @param int $slot the number used to identify this question within this attempt. 817 * @return question_attempt the requested question_attempt. 818 */ 819 public function get_question_attempt($slot) { 820 return $this->quba->get_question_attempt($slot); 821 } 822 823 /** 824 * Get all the question_attempt objects that have ever appeared in a given slot. 825 * 826 * This relates to the 'Try another question like this one' feature. 827 * 828 * @param int $slot the number used to identify this question within this attempt. 829 * @return question_attempt[] the attempts. 830 */ 831 public function all_question_attempts_originally_in_slot($slot) { 832 $qas = []; 833 foreach ($this->quba->get_attempt_iterator() as $qa) { 834 if ($qa->get_metadata('originalslot') == $slot) { 835 $qas[] = $qa; 836 } 837 } 838 $qas[] = $this->quba->get_question_attempt($slot); 839 return $qas; 840 } 841 842 /** 843 * Is a particular question in this attempt a real question, or something like a description. 844 * 845 * @param int $slot the number used to identify this question within this attempt. 846 * @return int whether that question is a real question. Actually returns the 847 * question length, which could theoretically be greater than one. 848 */ 849 public function is_real_question($slot) { 850 return $this->quba->get_question($slot, false)->length; 851 } 852 853 /** 854 * Is a particular question in this attempt a real question, or something like a description. 855 * 856 * @param int $slot the number used to identify this question within this attempt. 857 * @return bool whether that question is a real question. 858 */ 859 public function is_question_flagged($slot) { 860 return $this->quba->get_question_attempt($slot)->is_flagged(); 861 } 862 863 /** 864 * Checks whether the question in this slot requires the previous 865 * question to have been completed. 866 * 867 * @param int $slot the number used to identify this question within this attempt. 868 * @return bool whether the previous question must have been completed before 869 * this one can be seen. 870 */ 871 public function is_blocked_by_previous_question($slot) { 872 return $slot > 1 && isset($this->slots[$slot]) && $this->slots[$slot]->requireprevious && 873 !$this->slots[$slot]->section->shufflequestions && 874 !$this->slots[$slot - 1]->section->shufflequestions && 875 $this->get_navigation_method() != QUIZ_NAVMETHOD_SEQ && 876 !$this->get_question_state($slot - 1)->is_finished() && 877 $this->quba->can_question_finish_during_attempt($slot - 1); 878 } 879 880 /** 881 * Is it possible for this question to be re-started within this attempt? 882 * 883 * @param int $slot the number used to identify this question within this attempt. 884 * @return bool whether the student should be given the option to restart this question now. 885 */ 886 public function can_question_be_redone_now($slot) { 887 return $this->get_quiz()->canredoquestions && !$this->is_finished() && 888 $this->get_question_state($slot)->is_finished(); 889 } 890 891 /** 892 * Given a slot in this attempt, which may or not be a redone question, return the original slot. 893 * 894 * @param int $slot identifies a particular question in this attempt. 895 * @return int the slot where this question was originally. 896 */ 897 public function get_original_slot($slot) { 898 $originalslot = $this->quba->get_question_attempt_metadata($slot, 'originalslot'); 899 if ($originalslot) { 900 return $originalslot; 901 } else { 902 return $slot; 903 } 904 } 905 906 /** 907 * Get the displayed question number for a slot. 908 * 909 * @param int $slot the number used to identify this question within this attempt. 910 * @return string the displayed question number for the question in this slot. 911 * For example '1', '2', '3' or 'i'. 912 */ 913 public function get_question_number($slot): string { 914 return $this->questionnumbers[$slot]; 915 } 916 917 /** 918 * If the section heading, if any, that should come just before this slot. 919 * 920 * @param int $slot identifies a particular question in this attempt. 921 * @return string|null the required heading, or null if there is not one here. 922 */ 923 public function get_heading_before_slot($slot) { 924 if ($this->slots[$slot]->firstinsection) { 925 return $this->slots[$slot]->section->heading; 926 } else { 927 return null; 928 } 929 } 930 931 /** 932 * Return the page of the quiz where this question appears. 933 * 934 * @param int $slot the number used to identify this question within this attempt. 935 * @return int the page of the quiz this question appears on. 936 */ 937 public function get_question_page($slot) { 938 return $this->questionpages[$slot]; 939 } 940 941 /** 942 * Return the grade obtained on a particular question, if the user is permitted 943 * to see it. You must previously have called load_question_states to load the 944 * state data about this question. 945 * 946 * @param int $slot the number used to identify this question within this attempt. 947 * @return string the formatted grade, to the number of decimal places specified 948 * by the quiz. 949 */ 950 public function get_question_name($slot) { 951 return $this->quba->get_question($slot, false)->name; 952 } 953 954 /** 955 * Return the {@see question_state} that this question is in. 956 * 957 * @param int $slot the number used to identify this question within this attempt. 958 * @return question_state the state this question is in. 959 */ 960 public function get_question_state($slot) { 961 return $this->quba->get_question_state($slot); 962 } 963 964 /** 965 * Return the grade obtained on a particular question, if the user is permitted 966 * to see it. You must previously have called load_question_states to load the 967 * state data about this question. 968 * 969 * @param int $slot the number used to identify this question within this attempt. 970 * @param bool $showcorrectness Whether right/partial/wrong states should 971 * be distinguished. 972 * @return string the formatted grade, to the number of decimal places specified 973 * by the quiz. 974 */ 975 public function get_question_status($slot, $showcorrectness) { 976 return $this->quba->get_question_state_string($slot, $showcorrectness); 977 } 978 979 /** 980 * Return the grade obtained on a particular question, if the user is permitted 981 * to see it. You must previously have called load_question_states to load the 982 * state data about this question. 983 * 984 * @param int $slot the number used to identify this question within this attempt. 985 * @param bool $showcorrectness Whether right/partial/wrong states should 986 * be distinguished. 987 * @return string class name for this state. 988 */ 989 public function get_question_state_class($slot, $showcorrectness) { 990 return $this->quba->get_question_state_class($slot, $showcorrectness); 991 } 992 993 /** 994 * Return the grade obtained on a particular question. 995 * 996 * You must previously have called load_question_states to load the state 997 * data about this question. 998 * 999 * @param int $slot the number used to identify this question within this attempt. 1000 * @return string the formatted grade, to the number of decimal places specified by the quiz. 1001 */ 1002 public function get_question_mark($slot) { 1003 return quiz_format_question_grade($this->get_quiz(), $this->quba->get_question_mark($slot)); 1004 } 1005 1006 /** 1007 * Get the time of the most recent action performed on a question. 1008 * 1009 * @param int $slot the number used to identify this question within this usage. 1010 * @return int timestamp. 1011 */ 1012 public function get_question_action_time($slot) { 1013 return $this->quba->get_question_action_time($slot); 1014 } 1015 1016 /** 1017 * Return the question type name for a given slot within the current attempt. 1018 * 1019 * @param int $slot the number used to identify this question within this attempt. 1020 * @return string the question type name. 1021 */ 1022 public function get_question_type_name($slot) { 1023 return $this->quba->get_question($slot, false)->get_type_name(); 1024 } 1025 1026 /** 1027 * Get the time remaining for an in-progress attempt, if the time is short 1028 * enough that it would be worth showing a timer. 1029 * 1030 * @param int $timenow the time to consider as 'now'. 1031 * @return int|false the number of seconds remaining for this attempt. 1032 * False if there is no limit. 1033 */ 1034 public function get_time_left_display($timenow) { 1035 if ($this->attempt->state != self::IN_PROGRESS) { 1036 return false; 1037 } 1038 return $this->get_access_manager($timenow)->get_time_left_display($this->attempt, $timenow); 1039 } 1040 1041 1042 /** 1043 * Get the time when this attempt was submitted. 1044 * 1045 * @return int timestamp, or 0 if it has not been submitted yet. 1046 */ 1047 public function get_submitted_date() { 1048 return $this->attempt->timefinish; 1049 } 1050 1051 /** 1052 * If the attempt is in an applicable state, work out the time by which the 1053 * student should next do something. 1054 * 1055 * @return int timestamp by which the student needs to do something. 1056 */ 1057 public function get_due_date() { 1058 $deadlines = []; 1059 if ($this->quizobj->get_quiz()->timelimit) { 1060 $deadlines[] = $this->attempt->timestart + $this->quizobj->get_quiz()->timelimit; 1061 } 1062 if ($this->quizobj->get_quiz()->timeclose) { 1063 $deadlines[] = $this->quizobj->get_quiz()->timeclose; 1064 } 1065 if ($deadlines) { 1066 $duedate = min($deadlines); 1067 } else { 1068 return false; 1069 } 1070 1071 switch ($this->attempt->state) { 1072 case self::IN_PROGRESS: 1073 return $duedate; 1074 1075 case self::OVERDUE: 1076 return $duedate + $this->quizobj->get_quiz()->graceperiod; 1077 1078 default: 1079 throw new coding_exception('Unexpected state: ' . $this->attempt->state); 1080 } 1081 } 1082 1083 // URLs related to this attempt ============================================. 1084 1085 /** 1086 * Get the URL of this quiz's view.php page. 1087 * 1088 * @return moodle_url quiz view url. 1089 */ 1090 public function view_url() { 1091 return $this->quizobj->view_url(); 1092 } 1093 1094 /** 1095 * Get the URL to start or continue an attempt. 1096 * 1097 * @param int|null $slot which question in the attempt to go to after starting (optional). 1098 * @param int $page which page in the attempt to go to after starting. 1099 * @return moodle_url the URL of this quiz's edit page. Needs to be POSTed to with a cmid parameter. 1100 */ 1101 public function start_attempt_url($slot = null, $page = -1) { 1102 if ($page == -1 && !is_null($slot)) { 1103 $page = $this->get_question_page($slot); 1104 } else { 1105 $page = 0; 1106 } 1107 return $this->quizobj->start_attempt_url($page); 1108 } 1109 1110 /** 1111 * Generates the title of the attempt page. 1112 * 1113 * @param int $page the page number (starting with 0) in the attempt. 1114 * @return string attempt page title. 1115 */ 1116 public function attempt_page_title(int $page) : string { 1117 if ($this->get_num_pages() > 1) { 1118 $a = new stdClass(); 1119 $a->name = $this->get_quiz_name(); 1120 $a->currentpage = $page + 1; 1121 $a->totalpages = $this->get_num_pages(); 1122 $title = get_string('attempttitlepaged', 'quiz', $a); 1123 } else { 1124 $title = get_string('attempttitle', 'quiz', $this->get_quiz_name()); 1125 } 1126 1127 return $title; 1128 } 1129 1130 /** 1131 * Get the URL of a particular page within this attempt. 1132 * 1133 * @param int|null $slot if specified, the slot number of a specific question to link to. 1134 * @param int $page if specified, a particular page to link to. If not given deduced 1135 * from $slot, or goes to the first page. 1136 * @param int $thispage if not -1, the current page. Will cause links to other things on 1137 * this page to be output as only a fragment. 1138 * @return moodle_url the URL to continue this attempt. 1139 */ 1140 public function attempt_url($slot = null, $page = -1, $thispage = -1) { 1141 return $this->page_and_question_url('attempt', $slot, $page, false, $thispage); 1142 } 1143 1144 /** 1145 * Generates the title of the summary page. 1146 * 1147 * @return string summary page title. 1148 */ 1149 public function summary_page_title() : string { 1150 return get_string('attemptsummarytitle', 'quiz', $this->get_quiz_name()); 1151 } 1152 1153 /** 1154 * Get the URL of the summary page of this attempt. 1155 * 1156 * @return moodle_url the URL of this quiz's summary page. 1157 */ 1158 public function summary_url() { 1159 return new moodle_url('/mod/quiz/summary.php', ['attempt' => $this->attempt->id, 'cmid' => $this->get_cmid()]); 1160 } 1161 1162 /** 1163 * Get the URL to which the attempt data should be submitted. 1164 * 1165 * @return moodle_url the URL of this quiz's summary page. 1166 */ 1167 public function processattempt_url() { 1168 return new moodle_url('/mod/quiz/processattempt.php'); 1169 } 1170 1171 /** 1172 * Generates the title of the review page. 1173 * 1174 * @param int $page the page number (starting with 0) in the attempt. 1175 * @param bool $showall whether the review page contains the entire attempt on one page. 1176 * @return string title of the review page. 1177 */ 1178 public function review_page_title(int $page, bool $showall = false) : string { 1179 if (!$showall && $this->get_num_pages() > 1) { 1180 $a = new stdClass(); 1181 $a->name = $this->get_quiz_name(); 1182 $a->currentpage = $page + 1; 1183 $a->totalpages = $this->get_num_pages(); 1184 $title = get_string('attemptreviewtitlepaged', 'quiz', $a); 1185 } else { 1186 $title = get_string('attemptreviewtitle', 'quiz', $this->get_quiz_name()); 1187 } 1188 1189 return $title; 1190 } 1191 1192 /** 1193 * Get the URL of a particular page in the review of this attempt. 1194 * 1195 * @param int|null $slot indicates which question to link to. 1196 * @param int $page if specified, the URL of this particular page of the attempt, otherwise 1197 * the URL will go to the first page. If -1, deduce $page from $slot. 1198 * @param bool|null $showall if true, the URL will be to review the entire attempt on one page, 1199 * and $page will be ignored. If null, a sensible default will be chosen. 1200 * @param int $thispage if not -1, the current page. Will cause links to other things on 1201 * this page to be output as only a fragment. 1202 * @return moodle_url the URL to review this attempt. 1203 */ 1204 public function review_url($slot = null, $page = -1, $showall = null, $thispage = -1) { 1205 return $this->page_and_question_url('review', $slot, $page, $showall, $thispage); 1206 } 1207 1208 /** 1209 * By default, should this script show all questions on one page for this attempt? 1210 * 1211 * @param string $script the script name, e.g. 'attempt', 'summary', 'review'. 1212 * @return bool whether show all on one page should be on by default. 1213 */ 1214 public function get_default_show_all($script) { 1215 return $script === 'review' && count($this->questionpages) < self::MAX_SLOTS_FOR_DEFAULT_REVIEW_SHOW_ALL; 1216 } 1217 1218 // Bits of content =========================================================. 1219 1220 /** 1221 * If $reviewoptions->attempt is false, meaning that students can't review this 1222 * attempt at the moment, return an appropriate string explaining why. 1223 * 1224 * @param bool $short if true, return a shorter string. 1225 * @return string an appropriate message. 1226 */ 1227 public function cannot_review_message($short = false) { 1228 return $this->quizobj->cannot_review_message( 1229 $this->get_attempt_state(), $short); 1230 } 1231 1232 /** 1233 * Initialise the JS etc. required all the questions on a page. 1234 * 1235 * @param int|string $page a page number, or 'all'. 1236 * @param bool $showall if true, forces page number to all. 1237 * @return string HTML to output - mostly obsolete, will probably be an empty string. 1238 */ 1239 public function get_html_head_contributions($page = 'all', $showall = false) { 1240 if ($showall) { 1241 $page = 'all'; 1242 } 1243 $result = ''; 1244 foreach ($this->get_slots($page) as $slot) { 1245 $result .= $this->quba->render_question_head_html($slot); 1246 } 1247 $result .= question_engine::initialise_js(); 1248 return $result; 1249 } 1250 1251 /** 1252 * Initialise the JS etc. required by one question. 1253 * 1254 * @param int $slot the question slot number. 1255 * @return string HTML to output - but this is mostly obsolete. Will probably be an empty string. 1256 */ 1257 public function get_question_html_head_contributions($slot) { 1258 return $this->quba->render_question_head_html($slot) . 1259 question_engine::initialise_js(); 1260 } 1261 1262 /** 1263 * Print the HTML for the start new preview button, if the current user 1264 * is allowed to see one. 1265 * 1266 * @return string HTML for the button. 1267 */ 1268 public function restart_preview_button() { 1269 global $OUTPUT; 1270 if ($this->is_preview() && $this->is_preview_user()) { 1271 return $OUTPUT->single_button(new moodle_url( 1272 $this->start_attempt_url(), ['forcenew' => true]), 1273 get_string('startnewpreview', 'quiz')); 1274 } else { 1275 return ''; 1276 } 1277 } 1278 1279 /** 1280 * Generate the HTML that displays the question in its current state, with 1281 * the appropriate display options. 1282 * 1283 * @param int $slot identifies the question in the attempt. 1284 * @param bool $reviewing is the being printed on an attempt or a review page. 1285 * @param renderer $renderer the quiz renderer. 1286 * @param moodle_url $thispageurl the URL of the page this question is being printed on. 1287 * @return string HTML for the question in its current state. 1288 */ 1289 public function render_question($slot, $reviewing, renderer $renderer, $thispageurl = null) { 1290 if ($this->is_blocked_by_previous_question($slot)) { 1291 $placeholderqa = $this->make_blocked_question_placeholder($slot); 1292 1293 $displayoptions = $this->get_display_options($reviewing); 1294 $displayoptions->manualcomment = question_display_options::HIDDEN; 1295 $displayoptions->history = question_display_options::HIDDEN; 1296 $displayoptions->readonly = true; 1297 1298 return html_writer::div($placeholderqa->render($displayoptions, 1299 $this->get_question_number($this->get_original_slot($slot))), 1300 'mod_quiz-blocked_question_warning'); 1301 } 1302 1303 return $this->render_question_helper($slot, $reviewing, $thispageurl, $renderer, null); 1304 } 1305 1306 /** 1307 * Helper used by {@see render_question()} and {@see render_question_at_step()}. 1308 * 1309 * @param int $slot identifies the question in the attempt. 1310 * @param bool $reviewing is the being printed on an attempt or a review page. 1311 * @param moodle_url $thispageurl the URL of the page this question is being printed on. 1312 * @param renderer $renderer the quiz renderer. 1313 * @param int|null $seq the seq number of the past state to display. 1314 * @return string HTML fragment. 1315 */ 1316 protected function render_question_helper($slot, $reviewing, $thispageurl, 1317 renderer $renderer, $seq) { 1318 $originalslot = $this->get_original_slot($slot); 1319 $number = $this->get_question_number($originalslot); 1320 $displayoptions = $this->get_display_options_with_edit_link($reviewing, $slot, $thispageurl); 1321 1322 if ($slot != $originalslot) { 1323 $originalmaxmark = $this->get_question_attempt($slot)->get_max_mark(); 1324 $this->get_question_attempt($slot)->set_max_mark($this->get_question_attempt($originalslot)->get_max_mark()); 1325 } 1326 1327 if ($this->can_question_be_redone_now($slot)) { 1328 $displayoptions->extrainfocontent = $renderer->redo_question_button( 1329 $slot, $displayoptions->readonly); 1330 } 1331 1332 if ($displayoptions->history && $displayoptions->questionreviewlink) { 1333 $links = $this->links_to_other_redos($slot, $displayoptions->questionreviewlink); 1334 if ($links) { 1335 $displayoptions->extrahistorycontent = html_writer::tag('p', 1336 get_string('redoesofthisquestion', 'quiz', $renderer->render($links))); 1337 } 1338 } 1339 1340 if ($seq === null) { 1341 $output = $this->quba->render_question($slot, $displayoptions, $number); 1342 } else { 1343 $output = $this->quba->render_question_at_step($slot, $seq, $displayoptions, $number); 1344 } 1345 1346 if ($slot != $originalslot) { 1347 $this->get_question_attempt($slot)->set_max_mark($originalmaxmark); 1348 } 1349 1350 return $output; 1351 } 1352 1353 /** 1354 * Create a fake question to be displayed in place of a question that is blocked 1355 * until the previous question has been answered. 1356 * 1357 * @param int $slot int slot number of the question to replace. 1358 * @return question_attempt the placeholder question attempt. 1359 */ 1360 protected function make_blocked_question_placeholder($slot) { 1361 $replacedquestion = $this->get_question_attempt($slot)->get_question(false); 1362 1363 question_bank::load_question_definition_classes('description'); 1364 $question = new qtype_description_question(); 1365 $question->id = $replacedquestion->id; 1366 $question->category = null; 1367 $question->parent = 0; 1368 $question->qtype = question_bank::get_qtype('description'); 1369 $question->name = ''; 1370 $question->questiontext = get_string('questiondependsonprevious', 'quiz'); 1371 $question->questiontextformat = FORMAT_HTML; 1372 $question->generalfeedback = ''; 1373 $question->defaultmark = $this->quba->get_question_max_mark($slot); 1374 $question->length = $replacedquestion->length; 1375 $question->penalty = 0; 1376 $question->stamp = ''; 1377 $question->status = \core_question\local\bank\question_version_status::QUESTION_STATUS_READY; 1378 $question->timecreated = null; 1379 $question->timemodified = null; 1380 $question->createdby = null; 1381 $question->modifiedby = null; 1382 1383 $placeholderqa = new question_attempt($question, $this->quba->get_id(), 1384 null, $this->quba->get_question_max_mark($slot)); 1385 $placeholderqa->set_slot($slot); 1386 $placeholderqa->start($this->get_quiz()->preferredbehaviour, 1); 1387 $placeholderqa->set_flagged($this->is_question_flagged($slot)); 1388 return $placeholderqa; 1389 } 1390 1391 /** 1392 * Like {@see render_question()} but displays the question at the past step 1393 * indicated by $seq, rather than showing the latest step. 1394 * 1395 * @param int $slot the slot number of a question in this quiz attempt. 1396 * @param int $seq the seq number of the past state to display. 1397 * @param bool $reviewing is the being printed on an attempt or a review page. 1398 * @param renderer $renderer the quiz renderer. 1399 * @param moodle_url $thispageurl the URL of the page this question is being printed on. 1400 * @return string HTML for the question in its current state. 1401 */ 1402 public function render_question_at_step($slot, $seq, $reviewing, 1403 renderer $renderer, $thispageurl = null) { 1404 return $this->render_question_helper($slot, $reviewing, $thispageurl, $renderer, $seq); 1405 } 1406 1407 /** 1408 * Wrapper round print_question from lib/questionlib.php. 1409 * 1410 * @param int $slot the id of a question in this quiz attempt. 1411 * @return string HTML of the question. 1412 */ 1413 public function render_question_for_commenting($slot) { 1414 $options = $this->get_display_options(true); 1415 $options->generalfeedback = question_display_options::HIDDEN; 1416 $options->manualcomment = question_display_options::EDITABLE; 1417 return $this->quba->render_question($slot, $options, 1418 $this->get_question_number($slot)); 1419 } 1420 1421 /** 1422 * Check whether access should be allowed to a particular file. 1423 * 1424 * @param int $slot the slot of a question in this quiz attempt. 1425 * @param bool $reviewing is the being printed on an attempt or a review page. 1426 * @param int $contextid the file context id from the request. 1427 * @param string $component the file component from the request. 1428 * @param string $filearea the file area from the request. 1429 * @param array $args extra part components from the request. 1430 * @param bool $forcedownload whether to force download. 1431 * @return bool true if the file can be accessed. 1432 */ 1433 public function check_file_access($slot, $reviewing, $contextid, $component, 1434 $filearea, $args, $forcedownload) { 1435 $options = $this->get_display_options($reviewing); 1436 1437 // Check permissions - warning there is similar code in review.php and 1438 // reviewquestion.php. If you change on, change them all. 1439 if ($reviewing && $this->is_own_attempt() && !$options->attempt) { 1440 return false; 1441 } 1442 1443 if ($reviewing && !$this->is_own_attempt() && !$this->is_review_allowed()) { 1444 return false; 1445 } 1446 1447 return $this->quba->check_file_access($slot, $options, 1448 $component, $filearea, $args, $forcedownload); 1449 } 1450 1451 /** 1452 * Get the navigation panel object for this attempt. 1453 * 1454 * @param renderer $output the quiz renderer to use to output things. 1455 * @param string $panelclass The type of panel, navigation_panel_attempt::class or navigation_panel_review::class 1456 * @param int $page the current page number. 1457 * @param bool $showall whether we are showing the whole quiz on one page. (Used by review.php.) 1458 * @return block_contents the requested object. 1459 */ 1460 public function get_navigation_panel(renderer $output, 1461 $panelclass, $page, $showall = false) { 1462 $panel = new $panelclass($this, $this->get_display_options(true), $page, $showall); 1463 1464 $bc = new block_contents(); 1465 $bc->attributes['id'] = 'mod_quiz_navblock'; 1466 $bc->attributes['role'] = 'navigation'; 1467 $bc->title = get_string('quiznavigation', 'quiz'); 1468 $bc->content = $output->navigation_panel($panel); 1469 return $bc; 1470 } 1471 1472 /** 1473 * Return an array of variant URLs to other attempts at this quiz. 1474 * 1475 * The $url passed in must contain an attempt parameter. 1476 * 1477 * The {@see links_to_other_attempts} object returned contains an 1478 * array with keys that are the attempt number, 1, 2, 3. 1479 * The array values are either a {@see moodle_url} with the attempt parameter 1480 * updated to point to the attempt id of the other attempt, or null corresponding 1481 * to the current attempt number. 1482 * 1483 * @param moodle_url $url a URL. 1484 * @return links_to_other_attempts|bool containing array int => null|moodle_url. 1485 * False if none. 1486 */ 1487 public function links_to_other_attempts(moodle_url $url) { 1488 $attempts = quiz_get_user_attempts($this->get_quiz()->id, $this->attempt->userid, 'all'); 1489 if (count($attempts) <= 1) { 1490 return false; 1491 } 1492 1493 $links = new links_to_other_attempts(); 1494 foreach ($attempts as $at) { 1495 if ($at->id == $this->attempt->id) { 1496 $links->links[$at->attempt] = null; 1497 } else { 1498 $links->links[$at->attempt] = new moodle_url($url, ['attempt' => $at->id]); 1499 } 1500 } 1501 return $links; 1502 } 1503 1504 /** 1505 * Return an array of variant URLs to other redos of the question in a particular slot. 1506 * 1507 * The $url passed in must contain a slot parameter. 1508 * 1509 * The {@see links_to_other_attempts} object returned contains an 1510 * array with keys that are the redo number, 1, 2, 3. 1511 * The array values are either a {@see moodle_url} with the slot parameter 1512 * updated to point to the slot that has that redo of this question; or null 1513 * corresponding to the redo identified by $slot. 1514 * 1515 * @param int $slot identifies a question in this attempt. 1516 * @param moodle_url $baseurl the base URL to modify to generate each link. 1517 * @return links_to_other_attempts|null containing array int => null|moodle_url, 1518 * or null if the question in this slot has not been redone. 1519 */ 1520 public function links_to_other_redos($slot, moodle_url $baseurl) { 1521 $originalslot = $this->get_original_slot($slot); 1522 1523 $qas = $this->all_question_attempts_originally_in_slot($originalslot); 1524 if (count($qas) <= 1) { 1525 return null; 1526 } 1527 1528 $links = new links_to_other_attempts(); 1529 $index = 1; 1530 foreach ($qas as $qa) { 1531 if ($qa->get_slot() == $slot) { 1532 $links->links[$index] = null; 1533 } else { 1534 $url = new moodle_url($baseurl, ['slot' => $qa->get_slot()]); 1535 $links->links[$index] = new action_link($url, $index, 1536 new popup_action('click', $url, 'reviewquestion', 1537 ['width' => 450, 'height' => 650]), 1538 ['title' => get_string('reviewresponse', 'question')]); 1539 } 1540 $index++; 1541 } 1542 return $links; 1543 } 1544 1545 // Methods for processing ==================================================. 1546 1547 /** 1548 * Check this attempt, to see if there are any state transitions that should 1549 * happen automatically. This function will update the attempt checkstatetime. 1550 * @param int $timestamp the timestamp that should be stored as the modified 1551 * @param bool $studentisonline is the student currently interacting with Moodle? 1552 */ 1553 public function handle_if_time_expired($timestamp, $studentisonline) { 1554 1555 $timeclose = $this->get_access_manager($timestamp)->get_end_time($this->attempt); 1556 1557 if ($timeclose === false || $this->is_preview()) { 1558 $this->update_timecheckstate(null); 1559 return; // No time limit. 1560 } 1561 if ($timestamp < $timeclose) { 1562 $this->update_timecheckstate($timeclose); 1563 return; // Time has not yet expired. 1564 } 1565 1566 // If the attempt is already overdue, look to see if it should be abandoned ... 1567 if ($this->attempt->state == self::OVERDUE) { 1568 $timeoverdue = $timestamp - $timeclose; 1569 $graceperiod = $this->quizobj->get_quiz()->graceperiod; 1570 if ($timeoverdue >= $graceperiod) { 1571 $this->process_abandon($timestamp, $studentisonline); 1572 } else { 1573 // Overdue time has not yet expired. 1574 $this->update_timecheckstate($timeclose + $graceperiod); 1575 } 1576 return; // ... and we are done. 1577 } 1578 1579 if ($this->attempt->state != self::IN_PROGRESS) { 1580 $this->update_timecheckstate(null); 1581 return; // Attempt is already in a final state. 1582 } 1583 1584 // Otherwise, we were in quiz_attempt::IN_PROGRESS, and time has now expired. 1585 // Transition to the appropriate state. 1586 switch ($this->quizobj->get_quiz()->overduehandling) { 1587 case 'autosubmit': 1588 $this->process_finish($timestamp, false, $studentisonline ? $timestamp : $timeclose, $studentisonline); 1589 return; 1590 1591 case 'graceperiod': 1592 $this->process_going_overdue($timestamp, $studentisonline); 1593 return; 1594 1595 case 'autoabandon': 1596 $this->process_abandon($timestamp, $studentisonline); 1597 return; 1598 } 1599 1600 // This is an overdue attempt with no overdue handling defined, so just abandon. 1601 $this->process_abandon($timestamp, $studentisonline); 1602 } 1603 1604 /** 1605 * Process all the actions that were submitted as part of the current request. 1606 * 1607 * @param int $timestamp the timestamp that should be stored as the modified. 1608 * time in the database for these actions. If null, will use the current time. 1609 * @param bool $becomingoverdue 1610 * @param array|null $simulatedresponses If not null, then we are testing, and this is an array of simulated data. 1611 * There are two formats supported here, for historical reasons. The newer approach is to pass an array created by 1612 * {@see core_question_generator::get_simulated_post_data_for_questions_in_usage()}. 1613 * the second is to pass an array slot no => contains arrays representing student 1614 * responses which will be passed to {@see question_definition::prepare_simulated_post_data()}. 1615 * This second method will probably get deprecated one day. 1616 */ 1617 public function process_submitted_actions($timestamp, $becomingoverdue = false, $simulatedresponses = null) { 1618 global $DB; 1619 1620 $transaction = $DB->start_delegated_transaction(); 1621 1622 if ($simulatedresponses !== null) { 1623 if (is_int(key($simulatedresponses))) { 1624 // Legacy approach. Should be removed one day. 1625 $simulatedpostdata = $this->quba->prepare_simulated_post_data($simulatedresponses); 1626 } else { 1627 $simulatedpostdata = $simulatedresponses; 1628 } 1629 } else { 1630 $simulatedpostdata = null; 1631 } 1632 1633 $this->quba->process_all_actions($timestamp, $simulatedpostdata); 1634 question_engine::save_questions_usage_by_activity($this->quba); 1635 1636 $this->attempt->timemodified = $timestamp; 1637 if ($this->attempt->state == self::FINISHED) { 1638 $this->attempt->sumgrades = $this->quba->get_total_mark(); 1639 } 1640 if ($becomingoverdue) { 1641 $this->process_going_overdue($timestamp, true); 1642 } else { 1643 $DB->update_record('quiz_attempts', $this->attempt); 1644 } 1645 1646 if (!$this->is_preview() && $this->attempt->state == self::FINISHED) { 1647 $this->recompute_final_grade(); 1648 } 1649 1650 $transaction->allow_commit(); 1651 } 1652 1653 /** 1654 * Replace a question in an attempt with a new attempt at the same question. 1655 * 1656 * Well, for randomised questions, it won't be the same question, it will be 1657 * a different randomly selected pick from the available question. 1658 * 1659 * @param int $slot the question to restart. 1660 * @param int $timestamp the timestamp to record for this action. 1661 */ 1662 public function process_redo_question($slot, $timestamp) { 1663 global $DB; 1664 1665 if (!$this->can_question_be_redone_now($slot)) { 1666 throw new coding_exception('Attempt to restart the question in slot ' . $slot . 1667 ' when it is not in a state to be restarted.'); 1668 } 1669 1670 $qubaids = new \mod_quiz\question\qubaids_for_users_attempts( 1671 $this->get_quizid(), $this->get_userid(), 'all', true); 1672 1673 $transaction = $DB->start_delegated_transaction(); 1674 1675 // Add the question to the usage. It is important we do this before we choose a variant. 1676 $newquestionid = qbank_helper::choose_question_for_redo($this->get_quizid(), 1677 $this->get_quizobj()->get_context(), $this->slots[$slot]->id, $qubaids); 1678 $newquestion = question_bank::load_question($newquestionid, $this->get_quiz()->shuffleanswers); 1679 $newslot = $this->quba->add_question_in_place_of_other($slot, $newquestion); 1680 1681 // Choose the variant. 1682 if ($newquestion->get_num_variants() == 1) { 1683 $variant = 1; 1684 } else { 1685 $variantstrategy = new \core_question\engine\variants\least_used_strategy( 1686 $this->quba, $qubaids); 1687 $variant = $variantstrategy->choose_variant($newquestion->get_num_variants(), 1688 $newquestion->get_variants_selection_seed()); 1689 } 1690 1691 // Start the question. 1692 $this->quba->start_question($slot, $variant); 1693 $this->quba->set_max_mark($newslot, 0); 1694 $this->quba->set_question_attempt_metadata($newslot, 'originalslot', $slot); 1695 question_engine::save_questions_usage_by_activity($this->quba); 1696 $this->fire_attempt_question_restarted_event($slot, $newquestion->id); 1697 1698 $transaction->allow_commit(); 1699 } 1700 1701 /** 1702 * Process all the autosaved data that was part of the current request. 1703 * 1704 * @param int $timestamp the timestamp that should be stored as the modified. 1705 * time in the database for these actions. If null, will use the current time. 1706 */ 1707 public function process_auto_save($timestamp) { 1708 global $DB; 1709 1710 $transaction = $DB->start_delegated_transaction(); 1711 1712 $this->quba->process_all_autosaves($timestamp); 1713 question_engine::save_questions_usage_by_activity($this->quba); 1714 $this->fire_attempt_autosaved_event(); 1715 1716 $transaction->allow_commit(); 1717 } 1718 1719 /** 1720 * Update the flagged state for all question_attempts in this usage, if their 1721 * flagged state was changed in the request. 1722 */ 1723 public function save_question_flags() { 1724 global $DB; 1725 1726 $transaction = $DB->start_delegated_transaction(); 1727 $this->quba->update_question_flags(); 1728 question_engine::save_questions_usage_by_activity($this->quba); 1729 $transaction->allow_commit(); 1730 } 1731 1732 /** 1733 * Submit the attempt. 1734 * 1735 * The separate $timefinish argument should be used when the quiz attempt 1736 * is being processed asynchronously (for example when cron is submitting 1737 * attempts where the time has expired). 1738 * 1739 * @param int $timestamp the time to record as last modified time. 1740 * @param bool $processsubmitted if true, and question responses in the current 1741 * POST request are stored to be graded, before the attempt is finished. 1742 * @param ?int $timefinish if set, use this as the finish time for the attempt. 1743 * (otherwise use $timestamp as the finish time as well). 1744 * @param bool $studentisonline is the student currently interacting with Moodle? 1745 */ 1746 public function process_finish($timestamp, $processsubmitted, $timefinish = null, $studentisonline = false) { 1747 global $DB; 1748 1749 $transaction = $DB->start_delegated_transaction(); 1750 1751 if ($processsubmitted) { 1752 $this->quba->process_all_actions($timestamp); 1753 } 1754 $this->quba->finish_all_questions($timestamp); 1755 1756 question_engine::save_questions_usage_by_activity($this->quba); 1757 1758 $this->attempt->timemodified = $timestamp; 1759 $this->attempt->timefinish = $timefinish ?? $timestamp; 1760 $this->attempt->sumgrades = $this->quba->get_total_mark(); 1761 $this->attempt->state = self::FINISHED; 1762 $this->attempt->timecheckstate = null; 1763 $this->attempt->gradednotificationsenttime = null; 1764 1765 if (!$this->requires_manual_grading() || 1766 !has_capability('mod/quiz:emailnotifyattemptgraded', $this->get_quizobj()->get_context(), 1767 $this->get_userid())) { 1768 $this->attempt->gradednotificationsenttime = $this->attempt->timefinish; 1769 } 1770 1771 $DB->update_record('quiz_attempts', $this->attempt); 1772 1773 if (!$this->is_preview()) { 1774 $this->recompute_final_grade(); 1775 1776 // Trigger event. 1777 $this->fire_state_transition_event('\mod_quiz\event\attempt_submitted', $timestamp, $studentisonline); 1778 1779 // Tell any access rules that care that the attempt is over. 1780 $this->get_access_manager($timestamp)->current_attempt_finished(); 1781 } 1782 1783 $transaction->allow_commit(); 1784 } 1785 1786 /** 1787 * Update this attempt timecheckstate if necessary. 1788 * 1789 * @param int|null $time the timestamp to set. 1790 */ 1791 public function update_timecheckstate($time) { 1792 global $DB; 1793 if ($this->attempt->timecheckstate !== $time) { 1794 $this->attempt->timecheckstate = $time; 1795 $DB->set_field('quiz_attempts', 'timecheckstate', $time, ['id' => $this->attempt->id]); 1796 } 1797 } 1798 1799 /** 1800 * Needs to be called after this attempt's grade is changed, to update the overall quiz grade. 1801 */ 1802 protected function recompute_final_grade(): void { 1803 $this->quizobj->get_grade_calculator()->recompute_final_grade($this->get_userid()); 1804 } 1805 1806 /** 1807 * Mark this attempt as now overdue. 1808 * 1809 * @param int $timestamp the time to deem as now. 1810 * @param bool $studentisonline is the student currently interacting with Moodle? 1811 */ 1812 public function process_going_overdue($timestamp, $studentisonline) { 1813 global $DB; 1814 1815 $transaction = $DB->start_delegated_transaction(); 1816 $this->attempt->timemodified = $timestamp; 1817 $this->attempt->state = self::OVERDUE; 1818 // If we knew the attempt close time, we could compute when the graceperiod ends. 1819 // Instead, we'll just fix it up through cron. 1820 $this->attempt->timecheckstate = $timestamp; 1821 $DB->update_record('quiz_attempts', $this->attempt); 1822 1823 $this->fire_state_transition_event('\mod_quiz\event\attempt_becameoverdue', $timestamp, $studentisonline); 1824 1825 $transaction->allow_commit(); 1826 1827 quiz_send_overdue_message($this); 1828 } 1829 1830 /** 1831 * Mark this attempt as abandoned. 1832 * 1833 * @param int $timestamp the time to deem as now. 1834 * @param bool $studentisonline is the student currently interacting with Moodle? 1835 */ 1836 public function process_abandon($timestamp, $studentisonline) { 1837 global $DB; 1838 1839 $transaction = $DB->start_delegated_transaction(); 1840 $this->attempt->timemodified = $timestamp; 1841 $this->attempt->state = self::ABANDONED; 1842 $this->attempt->timecheckstate = null; 1843 $DB->update_record('quiz_attempts', $this->attempt); 1844 1845 $this->fire_state_transition_event('\mod_quiz\event\attempt_abandoned', $timestamp, $studentisonline); 1846 1847 $transaction->allow_commit(); 1848 } 1849 1850 /** 1851 * This method takes an attempt in the 'Never submitted' state, and reopens it. 1852 * 1853 * If, for this student, time has not expired (perhaps, because an override has 1854 * been added, then the attempt is left open. Otherwise, it is immediately submitted 1855 * for grading. 1856 * 1857 * @param int $timestamp the time to deem as now. 1858 */ 1859 public function process_reopen_abandoned($timestamp) { 1860 global $DB; 1861 1862 // Verify that things are as we expect. 1863 if ($this->get_state() != self::ABANDONED) { 1864 throw new coding_exception('Can only reopen an attempt that was never submitted.'); 1865 } 1866 1867 $transaction = $DB->start_delegated_transaction(); 1868 $this->attempt->timemodified = $timestamp; 1869 $this->attempt->state = self::IN_PROGRESS; 1870 $this->attempt->timecheckstate = null; 1871 $DB->update_record('quiz_attempts', $this->attempt); 1872 1873 $this->fire_state_transition_event('\mod_quiz\event\attempt_reopened', $timestamp, false); 1874 1875 $timeclose = $this->get_access_manager($timestamp)->get_end_time($this->attempt); 1876 if ($timeclose && $timestamp > $timeclose) { 1877 $this->process_finish($timestamp, false, $timeclose); 1878 } 1879 1880 $transaction->allow_commit(); 1881 } 1882 1883 /** 1884 * Fire a state transition event. 1885 * 1886 * @param string $eventclass the event class name. 1887 * @param int $timestamp the timestamp to include in the event. 1888 * @param bool $studentisonline is the student currently interacting with Moodle? 1889 */ 1890 protected function fire_state_transition_event($eventclass, $timestamp, $studentisonline) { 1891 global $USER; 1892 $quizrecord = $this->get_quiz(); 1893 $params = [ 1894 'context' => $this->get_quizobj()->get_context(), 1895 'courseid' => $this->get_courseid(), 1896 'objectid' => $this->attempt->id, 1897 'relateduserid' => $this->attempt->userid, 1898 'other' => [ 1899 'submitterid' => CLI_SCRIPT ? null : $USER->id, 1900 'quizid' => $quizrecord->id, 1901 'studentisonline' => $studentisonline 1902 ] 1903 ]; 1904 $event = $eventclass::create($params); 1905 $event->add_record_snapshot('quiz', $this->get_quiz()); 1906 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 1907 $event->trigger(); 1908 } 1909 1910 // Private methods =========================================================. 1911 1912 /** 1913 * Get a URL for a particular question on a particular page of the quiz. 1914 * Used by {@see attempt_url()} and {@see review_url()}. 1915 * 1916 * @param string $script e.g. 'attempt' or 'review'. Used in the URL like /mod/quiz/$script.php. 1917 * @param int $slot identifies the specific question on the page to jump to. 1918 * 0 to just use the $page parameter. 1919 * @param int $page -1 to look up the page number from the slot, otherwise 1920 * the page number to go to. 1921 * @param bool|null $showall if true, return a URL with showall=1, and not page number. 1922 * if null, then an intelligent default will be chosen. 1923 * @param int $thispage the page we are currently on. Links to questions on this 1924 * page will just be a fragment #q123. -1 to disable this. 1925 * @return moodle_url The requested URL. 1926 */ 1927 protected function page_and_question_url($script, $slot, $page, $showall, $thispage) { 1928 1929 $defaultshowall = $this->get_default_show_all($script); 1930 if ($showall === null && ($page == 0 || $page == -1)) { 1931 $showall = $defaultshowall; 1932 } 1933 1934 // Fix up $page. 1935 if ($page == -1) { 1936 if ($slot !== null && !$showall) { 1937 $page = $this->get_question_page($slot); 1938 } else { 1939 $page = 0; 1940 } 1941 } 1942 1943 if ($showall) { 1944 $page = 0; 1945 } 1946 1947 // Add a fragment to scroll down to the question. 1948 $fragment = ''; 1949 if ($slot !== null) { 1950 if ($slot == reset($this->pagelayout[$page]) && $thispage != $page) { 1951 // Changing the page, go to top. 1952 $fragment = '#'; 1953 } else { 1954 // Link to the question container. 1955 $qa = $this->get_question_attempt($slot); 1956 $fragment = '#' . $qa->get_outer_question_div_unique_id(); 1957 } 1958 } 1959 1960 // Work out the correct start to the URL. 1961 if ($thispage == $page) { 1962 return new moodle_url($fragment); 1963 1964 } else { 1965 $url = new moodle_url('/mod/quiz/' . $script . '.php' . $fragment, 1966 ['attempt' => $this->attempt->id, 'cmid' => $this->get_cmid()]); 1967 if ($page == 0 && $showall != $defaultshowall) { 1968 $url->param('showall', (int) $showall); 1969 } else if ($page > 0) { 1970 $url->param('page', $page); 1971 } 1972 return $url; 1973 } 1974 } 1975 1976 /** 1977 * Process responses during an attempt at a quiz. 1978 * 1979 * @param int $timenow time when the processing started. 1980 * @param bool $finishattempt whether to finish the attempt or not. 1981 * @param bool $timeup true if form was submitted by timer. 1982 * @param int $thispage current page number. 1983 * @return string the attempt state once the data has been processed. 1984 * @since Moodle 3.1 1985 */ 1986 public function process_attempt($timenow, $finishattempt, $timeup, $thispage) { 1987 global $DB; 1988 1989 $transaction = $DB->start_delegated_transaction(); 1990 1991 // Get key times. 1992 $accessmanager = $this->get_access_manager($timenow); 1993 $timeclose = $accessmanager->get_end_time($this->get_attempt()); 1994 $graceperiodmin = get_config('quiz', 'graceperiodmin'); 1995 1996 // Don't enforce timeclose for previews. 1997 if ($this->is_preview()) { 1998 $timeclose = false; 1999 } 2000 2001 // Check where we are in relation to the end time, if there is one. 2002 $toolate = false; 2003 if ($timeclose !== false) { 2004 if ($timenow > $timeclose - QUIZ_MIN_TIME_TO_CONTINUE) { 2005 // If there is only a very small amount of time left, there is no point trying 2006 // to show the student another page of the quiz. Just finish now. 2007 $timeup = true; 2008 if ($timenow > $timeclose + $graceperiodmin) { 2009 $toolate = true; 2010 } 2011 } else { 2012 // If time is not close to expiring, then ignore the client-side timer's opinion 2013 // about whether time has expired. This can happen if the time limit has changed 2014 // since the student's previous interaction. 2015 $timeup = false; 2016 } 2017 } 2018 2019 // If time is running out, trigger the appropriate action. 2020 $becomingoverdue = false; 2021 $becomingabandoned = false; 2022 if ($timeup) { 2023 if ($this->get_quiz()->overduehandling === 'graceperiod') { 2024 if ($timenow > $timeclose + $this->get_quiz()->graceperiod + $graceperiodmin) { 2025 // Grace period has run out. 2026 $finishattempt = true; 2027 $becomingabandoned = true; 2028 } else { 2029 $becomingoverdue = true; 2030 } 2031 } else { 2032 $finishattempt = true; 2033 } 2034 } 2035 2036 if (!$finishattempt) { 2037 // Just process the responses for this page and go to the next page. 2038 if (!$toolate) { 2039 try { 2040 $this->process_submitted_actions($timenow, $becomingoverdue); 2041 $this->fire_attempt_updated_event(); 2042 } catch (question_out_of_sequence_exception $e) { 2043 throw new moodle_exception('submissionoutofsequencefriendlymessage', 'question', 2044 $this->attempt_url(null, $thispage)); 2045 2046 } catch (Exception $e) { 2047 // This sucks, if we display our own custom error message, there is no way 2048 // to display the original stack trace. 2049 $debuginfo = ''; 2050 if (!empty($e->debuginfo)) { 2051 $debuginfo = $e->debuginfo; 2052 } 2053 throw new moodle_exception('errorprocessingresponses', 'question', 2054 $this->attempt_url(null, $thispage), $e->getMessage(), $debuginfo); 2055 } 2056 2057 if (!$becomingoverdue) { 2058 foreach ($this->get_slots() as $slot) { 2059 if (optional_param('redoslot' . $slot, false, PARAM_BOOL)) { 2060 $this->process_redo_question($slot, $timenow); 2061 } 2062 } 2063 } 2064 2065 } else { 2066 // The student is too late. 2067 $this->process_going_overdue($timenow, true); 2068 } 2069 2070 $transaction->allow_commit(); 2071 2072 return $becomingoverdue ? self::OVERDUE : self::IN_PROGRESS; 2073 } 2074 2075 // Update the quiz attempt record. 2076 try { 2077 if ($becomingabandoned) { 2078 $this->process_abandon($timenow, true); 2079 } else { 2080 if (!$toolate || $this->get_quiz()->overduehandling === 'graceperiod') { 2081 // Normally, we record the accurate finish time when the student is online. 2082 $finishtime = $timenow; 2083 } else { 2084 // But, if there is no grade period, and the final responses were too 2085 // late to be processed, record the close time, to reduce confusion. 2086 $finishtime = $timeclose; 2087 } 2088 $this->process_finish($timenow, !$toolate, $finishtime, true); 2089 } 2090 2091 } catch (question_out_of_sequence_exception $e) { 2092 throw new moodle_exception('submissionoutofsequencefriendlymessage', 'question', 2093 $this->attempt_url(null, $thispage)); 2094 2095 } catch (Exception $e) { 2096 // This sucks, if we display our own custom error message, there is no way 2097 // to display the original stack trace. 2098 $debuginfo = ''; 2099 if (!empty($e->debuginfo)) { 2100 $debuginfo = $e->debuginfo; 2101 } 2102 throw new moodle_exception('errorprocessingresponses', 'question', 2103 $this->attempt_url(null, $thispage), $e->getMessage(), $debuginfo); 2104 } 2105 2106 // Send the user to the review page. 2107 $transaction->allow_commit(); 2108 2109 return $becomingabandoned ? self::ABANDONED : self::FINISHED; 2110 } 2111 2112 /** 2113 * Check a page read access to see if is an out of sequence access. 2114 * 2115 * If allownext is set then we also check whether access to the page 2116 * after the current one should be permitted. 2117 * 2118 * @param int $page page number. 2119 * @param bool $allownext in case of a sequential navigation, can we go to next page ? 2120 * @return boolean false is an out of sequence access, true otherwise. 2121 * @since Moodle 3.1 2122 */ 2123 public function check_page_access(int $page, bool $allownext = true): bool { 2124 if ($this->get_navigation_method() != QUIZ_NAVMETHOD_SEQ) { 2125 return true; 2126 } 2127 // Sequential access: allow access to the summary, current page or next page. 2128 // Or if the user review his/her attempt, see MDLQA-1523. 2129 return $page == -1 2130 || $page == $this->get_currentpage() 2131 || $allownext && ($page == $this->get_currentpage() + 1); 2132 } 2133 2134 /** 2135 * Update attempt page. 2136 * 2137 * @param int $page page number. 2138 * @return boolean true if everything was ok, false otherwise (out of sequence access). 2139 * @since Moodle 3.1 2140 */ 2141 public function set_currentpage($page) { 2142 global $DB; 2143 2144 if ($this->check_page_access($page)) { 2145 $DB->set_field('quiz_attempts', 'currentpage', $page, ['id' => $this->get_attemptid()]); 2146 return true; 2147 } 2148 return false; 2149 } 2150 2151 /** 2152 * Trigger the attempt_viewed event. 2153 * 2154 * @since Moodle 3.1 2155 */ 2156 public function fire_attempt_viewed_event() { 2157 $params = [ 2158 'objectid' => $this->get_attemptid(), 2159 'relateduserid' => $this->get_userid(), 2160 'courseid' => $this->get_courseid(), 2161 'context' => $this->get_context(), 2162 'other' => [ 2163 'quizid' => $this->get_quizid(), 2164 'page' => $this->get_currentpage() 2165 ] 2166 ]; 2167 $event = \mod_quiz\event\attempt_viewed::create($params); 2168 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 2169 $event->trigger(); 2170 } 2171 2172 /** 2173 * Trigger the attempt_updated event. 2174 * 2175 * @return void 2176 */ 2177 public function fire_attempt_updated_event(): void { 2178 $params = [ 2179 'objectid' => $this->get_attemptid(), 2180 'relateduserid' => $this->get_userid(), 2181 'courseid' => $this->get_courseid(), 2182 'context' => $this->get_context(), 2183 'other' => [ 2184 'quizid' => $this->get_quizid(), 2185 'page' => $this->get_currentpage() 2186 ] 2187 ]; 2188 $event = \mod_quiz\event\attempt_updated::create($params); 2189 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 2190 $event->trigger(); 2191 } 2192 2193 /** 2194 * Trigger the attempt_autosaved event. 2195 * 2196 * @return void 2197 */ 2198 public function fire_attempt_autosaved_event(): void { 2199 $params = [ 2200 'objectid' => $this->get_attemptid(), 2201 'relateduserid' => $this->get_userid(), 2202 'courseid' => $this->get_courseid(), 2203 'context' => $this->get_context(), 2204 'other' => [ 2205 'quizid' => $this->get_quizid(), 2206 'page' => $this->get_currentpage() 2207 ] 2208 ]; 2209 $event = \mod_quiz\event\attempt_autosaved::create($params); 2210 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 2211 $event->trigger(); 2212 } 2213 2214 /** 2215 * Trigger the attempt_question_restarted event. 2216 * 2217 * @param int $slot Slot number 2218 * @param int $newquestionid New question id. 2219 * @return void 2220 */ 2221 public function fire_attempt_question_restarted_event(int $slot, int $newquestionid): void { 2222 $params = [ 2223 'objectid' => $this->get_attemptid(), 2224 'relateduserid' => $this->get_userid(), 2225 'courseid' => $this->get_courseid(), 2226 'context' => $this->get_context(), 2227 'other' => [ 2228 'quizid' => $this->get_quizid(), 2229 'page' => $this->get_currentpage(), 2230 'slot' => $slot, 2231 'newquestionid' => $newquestionid 2232 ] 2233 ]; 2234 $event = \mod_quiz\event\attempt_question_restarted::create($params); 2235 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 2236 $event->trigger(); 2237 } 2238 2239 /** 2240 * Trigger the attempt_summary_viewed event. 2241 * 2242 * @since Moodle 3.1 2243 */ 2244 public function fire_attempt_summary_viewed_event() { 2245 2246 $params = [ 2247 'objectid' => $this->get_attemptid(), 2248 'relateduserid' => $this->get_userid(), 2249 'courseid' => $this->get_courseid(), 2250 'context' => $this->get_context(), 2251 'other' => [ 2252 'quizid' => $this->get_quizid() 2253 ] 2254 ]; 2255 $event = \mod_quiz\event\attempt_summary_viewed::create($params); 2256 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 2257 $event->trigger(); 2258 } 2259 2260 /** 2261 * Trigger the attempt_reviewed event. 2262 * 2263 * @since Moodle 3.1 2264 */ 2265 public function fire_attempt_reviewed_event() { 2266 2267 $params = [ 2268 'objectid' => $this->get_attemptid(), 2269 'relateduserid' => $this->get_userid(), 2270 'courseid' => $this->get_courseid(), 2271 'context' => $this->get_context(), 2272 'other' => [ 2273 'quizid' => $this->get_quizid() 2274 ] 2275 ]; 2276 $event = \mod_quiz\event\attempt_reviewed::create($params); 2277 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 2278 $event->trigger(); 2279 } 2280 2281 /** 2282 * Trigger the attempt manual grading completed event. 2283 */ 2284 public function fire_attempt_manual_grading_completed_event() { 2285 $params = [ 2286 'objectid' => $this->get_attemptid(), 2287 'relateduserid' => $this->get_userid(), 2288 'courseid' => $this->get_courseid(), 2289 'context' => $this->get_context(), 2290 'other' => [ 2291 'quizid' => $this->get_quizid() 2292 ] 2293 ]; 2294 2295 $event = \mod_quiz\event\attempt_manual_grading_completed::create($params); 2296 $event->add_record_snapshot('quiz_attempts', $this->get_attempt()); 2297 $event->trigger(); 2298 } 2299 2300 /** 2301 * Update the timemodifiedoffline attempt field. 2302 * 2303 * This function should be used only when web services are being used. 2304 * 2305 * @param int $time time stamp. 2306 * @return boolean false if the field is not updated because web services aren't being used. 2307 * @since Moodle 3.2 2308 */ 2309 public function set_offline_modified_time($time) { 2310 // Update the timemodifiedoffline field only if web services are being used. 2311 if (WS_SERVER) { 2312 $this->attempt->timemodifiedoffline = $time; 2313 return true; 2314 } 2315 return false; 2316 } 2317 2318 /** 2319 * Get the total number of unanswered questions in the attempt. 2320 * 2321 * @return int 2322 */ 2323 public function get_number_of_unanswered_questions(): int { 2324 $totalunanswered = 0; 2325 foreach ($this->get_slots() as $slot) { 2326 if (!$this->is_real_question($slot)) { 2327 continue; 2328 } 2329 $questionstate = $this->get_question_state($slot); 2330 if ($questionstate == question_state::$todo || $questionstate == question_state::$invalid) { 2331 $totalunanswered++; 2332 } 2333 } 2334 return $totalunanswered; 2335 } 2336 }
title
Description
Body
title
Description
Body
title
Description
Body
title
Body