dd1bc955211d162588524e3c48e28983d872aa50
[moodle.git] / question / engine / simpletest / helpers.php
1 <?php
3 // This file is part of Moodle - http://moodle.org/
4 //
5 // Moodle is free software: you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation, either version 3 of the License, or
8 // (at your option) any later version.
9 //
10 // Moodle is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 // GNU General Public License for more details.
14 //
15 // You should have received a copy of the GNU General Public License
16 // along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
18 /**
19  * This file contains helper classes for testing the question engine.
20  *
21  * @package    moodlecore
22  * @subpackage questionengine
23  * @copyright  2009 The Open University
24  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
25  */
28 defined('MOODLE_INTERNAL') || die();
30 require_once(dirname(__FILE__) . '/../lib.php');
33 /**
34  * Makes some protected methods of question_attempt public to facilitate testing.
35  *
36  * @copyright  2009 The Open University
37  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
38  */
39 class testable_question_attempt extends question_attempt {
40     public function add_step($step) {#
41         parent::add_step($step);
42     }
43     public function set_min_fraction($fraction) {
44         $this->minfraction = $fraction;
45     }
46     public function set_behaviour(question_behaviour $behaviour) {
47         $this->behaviour = $behaviour;
48     }
49 }
52 /**
53  * This class creates questions of various types, which can then be used when
54  * testing.
55  *
56  * @copyright  2009 The Open University
57  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
58  */
59 class test_question_maker {
60     const STANDARD_OVERALL_CORRECT_FEEDBACK = 'Well done!';
61     const STANDARD_OVERALL_PARTIALLYCORRECT_FEEDBACK = 'Parts, but only parts, of your response are correct.';
62     const STANDARD_OVERALL_INCORRECT_FEEDBACK = 'That is not right at all.';
64     /**
65      * Just make a question_attempt at a question. Useful for unit tests that
66      * need to pass a $qa to methods that call format_text. Probably not safe
67      * to use for anything beyond that.
68      * @param question_definition $question a question.
69      * @param number $maxmark the max mark to set.
70      * @return question_attempt the question attempt.
71      */
72     public function get_a_qa($question, $maxmark = 3) {
73         return new question_attempt($question, 13, null, $maxmark);
74     }
76     /**
77      * Initialise the common fields of a question of any type.
78      */
79     public static function initialise_a_question($q) {
80         global $USER;
82         $q->id = 0;
83         $q->category = 0;
84         $q->parent = 0;
85         $q->questiontextformat = FORMAT_HTML;
86         $q->generalfeedbackformat = FORMAT_HTML;
87         $q->defaultmark = 1;
88         $q->penalty = 0.3333333;
89         $q->length = 1;
90         $q->stamp = make_unique_id_code();
91         $q->version = make_unique_id_code();
92         $q->hidden = 0;
93         $q->timecreated = time();
94         $q->timemodified = time();
95         $q->createdby = $USER->id;
96         $q->modifiedby = $USER->id;
97     }
99     /**
100      * Makes a truefalse question with correct answer true, defaultmark 1.
101      * @return qtype_truefalse_question
102      */
103     public static function make_a_truefalse_question() {
104         question_bank::load_question_definition_classes('truefalse');
105         $tf = new qtype_truefalse_question();
106         self::initialise_a_question($tf);
107         $tf->name = 'True/false question';
108         $tf->questiontext = 'The answer is true.';
109         $tf->generalfeedback = 'You should have selected true.';
110         $tf->penalty = 1;
111         $tf->qtype = question_bank::get_qtype('truefalse');
113         $tf->rightanswer = true;
114         $tf->truefeedback = 'This is the right answer.';
115         $tf->falsefeedback = 'This is the wrong answer.';
116         $tf->truefeedbackformat = FORMAT_HTML;
117         $tf->falsefeedbackformat = FORMAT_HTML;
118         $tf->trueanswerid = 13;
119         $tf->falseanswerid = 14;
121         return $tf;
122     }
124     /**
125      * Makes a multichoice question with choices 'A', 'B' and 'C' shuffled. 'A'
126      * is correct, defaultmark 1.
127      * @return qtype_multichoice_single_question
128      */
129     public static function make_a_multichoice_single_question() {
130         question_bank::load_question_definition_classes('multichoice');
131         $mc = new qtype_multichoice_single_question();
132         self::initialise_a_question($mc);
133         $mc->name = 'Multi-choice question, single response';
134         $mc->questiontext = 'The answer is A.';
135         $mc->generalfeedback = 'You should have selected A.';
136         $mc->qtype = question_bank::get_qtype('multichoice');
138         $mc->shuffleanswers = 1;
139         $mc->answernumbering = 'abc';
141         $mc->answers = array(
142             13 => new question_answer(13, 'A', 1, 'A is right', FORMAT_HTML),
143             14 => new question_answer(14, 'B', -0.3333333, 'B is wrong', FORMAT_HTML),
144             15 => new question_answer(15, 'C', -0.3333333, 'C is wrong', FORMAT_HTML),
145         );
147         return $mc;
148     }
150     /**
151      * Makes a multichoice question with choices 'A', 'B', 'C' and 'D' shuffled.
152      * 'A' and 'C' is correct, defaultmark 1.
153      * @return qtype_multichoice_multi_question
154      */
155     public static function make_a_multichoice_multi_question() {
156         question_bank::load_question_definition_classes('multichoice');
157         $mc = new qtype_multichoice_multi_question();
158         self::initialise_a_question($mc);
159         $mc->name = 'Multi-choice question, multiple response';
160         $mc->questiontext = 'The answer is A and C.';
161         $mc->generalfeedback = 'You should have selected A and C.';
162         $mc->qtype = question_bank::get_qtype('multichoice');
164         $mc->shuffleanswers = 1;
165         $mc->answernumbering = 'abc';
167         self::set_standard_combined_feedback_fields($mc);
169         $mc->answers = array(
170             13 => new question_answer(13, 'A', 0.5, 'A is part of the right answer', FORMAT_HTML),
171             14 => new question_answer(14, 'B', -1, 'B is wrong', FORMAT_HTML),
172             15 => new question_answer(15, 'C', 0.5, 'C is part of the right answer', FORMAT_HTML),
173             16 => new question_answer(16, 'D', -1, 'D is wrong', FORMAT_HTML),
174         );
176         return $mc;
177     }
179     /**
180      * Makes a matching question to classify 'Dog', 'Frog', 'Toad' and 'Cat' as
181      * 'Mammal', 'Amphibian' or 'Insect'.
182      * defaultmark 1. Stems are shuffled by default.
183      * @return qtype_match_question
184      */
185     public static function make_a_matching_question() {
186         question_bank::load_question_definition_classes('match');
187         $match = new qtype_match_question();
188         self::initialise_a_question($match);
189         $match->name = 'Matching question';
190         $match->questiontext = 'Classify the animals.';
191         $match->generalfeedback = 'Frogs and toads are amphibians, the others are mammals.';
192         $match->qtype = question_bank::get_qtype('match');
194         $match->shufflestems = 1;
196         self::set_standard_combined_feedback_fields($match);
198         // Using unset to get 1-based arrays.
199         $match->stems = array('', 'Dog', 'Frog', 'Toad', 'Cat');
200         $match->stemformat = array('', FORMAT_HTML, FORMAT_HTML, FORMAT_HTML, FORMAT_HTML);
201         $match->choices = array('', 'Mammal', 'Amphibian', 'Insect');
202         $match->right = array('', 1, 2, 2, 1);
203         unset($match->stems[0]);
204         unset($match->stemformat[0]);
205         unset($match->choices[0]);
206         unset($match->right[0]);
208         return $match;
209     }
211     /**
212      * Makes a shortanswer question with correct ansewer 'frog', partially
213      * correct answer 'toad' and defaultmark 1.
214      * @return qtype_shortanswer_question
215      */
216     public static function make_a_shortanswer_question() {
217         question_bank::load_question_definition_classes('shortanswer');
218         $sa = new qtype_shortanswer_question();
219         self::initialise_a_question($sa);
220         $sa->name = 'Short answer question';
221         $sa->questiontext = 'Name an amphibian: __________';
222         $sa->generalfeedback = 'Generalfeedback: frog or toad would have been OK.';
223         $sa->usecase = false;
224         $sa->answers = array(
225             13 => new question_answer(13, 'frog', 1.0, 'Frog is a very good answer.', FORMAT_HTML),
226             14 => new question_answer(14, 'toad', 0.8, 'Toad is an OK good answer.', FORMAT_HTML),
227             15 => new question_answer(15, '*', 0.0, 'That is a bad answer.', FORMAT_HTML),
228         );
229         $sa->qtype = question_bank::get_qtype('shortanswer');
231         return $sa;
232     }
234     /**
235      * Makes a numerical question with correct ansewer 3.14, and various incorrect
236      * answers with different feedback.
237      * @return qtype_numerical_question
238      */
239     public static function make_a_numerical_question() {
240         question_bank::load_question_definition_classes('numerical');
241         $num = new qtype_numerical_question();
242         self::initialise_a_question($num);
243         $num->name = 'Pi to two d.p.';
244         $num->questiontext = 'What is pi to two d.p.?';
245         $num->generalfeedback = 'Generalfeedback: 3.14 is the right answer.';
246         $num->answers = array(
247             13 => new qtype_numerical_answer(13, '3.14', 1.0, 'Very good.', FORMAT_HTML, 0),
248             14 => new qtype_numerical_answer(14, '3.142', 0.0, 'Too accurate.', FORMAT_HTML, 0.005),
249             15 => new qtype_numerical_answer(15, '3.1', 0.0, 'Not accurate enough.', FORMAT_HTML, 0.05),
250             16 => new qtype_numerical_answer(16, '3', 0.0, 'Not accurate enough.', FORMAT_HTML, 0.5),
251             17 => new qtype_numerical_answer(17, '*', 0.0, 'Completely wrong.', FORMAT_HTML, 0),
252         );
253         $num->qtype = question_bank::get_qtype('numerical');
254         $num->ap = new qtype_numerical_answer_processor(array());
256         return $num;
257     }
259     /**
260      * Makes a truefalse question with correct ansewer true, defaultmark 1.
261      * @return qtype_essay_question
262      */
263     public static function make_an_essay_question() {
264         question_bank::load_question_definition_classes('essay');
265         $essay = new qtype_essay_question();
266         self::initialise_a_question($essay);
267         $essay->name = 'Essay question';
268         $essay->questiontext = 'Write an essay.';
269         $essay->generalfeedback = 'I hope you wrote an interesting essay.';
270         $essay->penalty = 0;
271         $essay->qtype = question_bank::get_qtype('essay');
273         return $essay;
274     }
276     /**
277      * Makes a truefalse question with correct ansewer true, defaultmark 1.
278      * @return question_truefalse
279      */
280     public static function make_a_description_question() {
281         question_bank::load_question_definition_classes('description');
282         $description = new qtype_description_question();
283         self::initialise_a_question($description);
284         $description->name = 'Description question';
285         $description->questiontext = 'This text tells you a bit about the next few questions in this quiz.';
286         $description->generalfeedback = 'This is what this section of the quiz should have taught you.';
287         $description->qtype = question_bank::get_qtype('description');
289         return $description;
290     }
292     /**
293      * Add some standard overall feedback to a question. You need to use these
294      * specific feedback strings for the corresponding contains_..._feedback
295      * methods in {@link qbehaviour_walkthrough_test_base} to works.
296      * @param question_definition $q the question to add the feedback to.
297      */
298     public static function set_standard_combined_feedback_fields($q) {
299         $q->correctfeedback = self::STANDARD_OVERALL_CORRECT_FEEDBACK;
300         $q->correctfeedbackformat = FORMAT_HTML;
301         $q->partiallycorrectfeedback = self::STANDARD_OVERALL_PARTIALLYCORRECT_FEEDBACK;
302         $q->partiallycorrectfeedbackformat = FORMAT_HTML;
303         $q->shownumcorrect = true;
304         $q->incorrectfeedback = self::STANDARD_OVERALL_INCORRECT_FEEDBACK;
305         $q->incorrectfeedbackformat = FORMAT_HTML;
306     }
310 /**
311  * Helper for tests that need to simulate records loaded from the database.
312  *
313  * @copyright  2009 The Open University
314  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
315  */
316 abstract class testing_db_record_builder {
317     public static function build_db_records(array $table) {
318         $columns = array_shift($table);
319         $records = array();
320         foreach ($table as $row) {
321             if (count($row) != count($columns)) {
322                 throw new coding_exception("Row contains the wrong number of fields.");
323             }
324             $rec = new stdClass();
325             foreach ($columns as $i => $name) {
326                 $rec->$name = $row[$i];
327             }
328             $records[] = $rec;
329         }
330         return $records;
331     }
335 /**
336  * Helper base class for tests that need to simulate records loaded from the
337  * database.
338  *
339  * @copyright  2009 The Open University
340  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
341  */
342 class data_loading_method_test_base extends UnitTestCase {
343     public function build_db_records(array $table) {
344         return testing_db_record_builder::build_db_records($table);
345     }
349 /**
350  * Helper base class for tests that walk a question through a sequents of
351  * interactions under the control of a particular behaviour.
352  *
353  * @copyright  2009 The Open University
354  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
355  */
356 class qbehaviour_walkthrough_test_base extends UnitTestCase {
357     /** @var question_display_options */
358     protected $displayoptions;
359     /** @var question_usage_by_activity */
360     protected $quba;
361     /** @var unknown_type integer */
362     protected $slot;
364     public function setUp() {
365         $this->displayoptions = new question_display_options();
366         $this->quba = question_engine::make_questions_usage_by_activity('unit_test',
367                 get_context_instance(CONTEXT_SYSTEM));
368     }
370     public function tearDown() {
371         $this->displayoptions = null;
372         $this->quba = null;
373     }
375     protected function start_attempt_at_question($question, $preferredbehaviour, $maxmark = null) {
376         $this->quba->set_preferred_behaviour($preferredbehaviour);
377         $this->slot = $this->quba->add_question($question, $maxmark);
378         $this->quba->start_all_questions();
379     }
380     protected function process_submission($data) {
381         $this->quba->process_action($this->slot, $data);
382     }
384     protected function manual_grade($comment, $mark) {
385         $this->quba->manual_grade($this->slot, $comment, $mark);
386     }
388     protected function check_current_state($state) {
389         $this->assertEqual($this->quba->get_question_state($this->slot), $state, 'Questions is in the wrong state: %s.');
390     }
392     protected function check_current_mark($mark) {
393         if (is_null($mark)) {
394             $this->assertNull($this->quba->get_question_mark($this->slot));
395         } else {
396             if ($mark == 0) {
397                 // PHP will think a null mark and a mark of 0 are equal,
398                 // so explicity check not null in this case.
399                 $this->assertNotNull($this->quba->get_question_mark($this->slot));
400             }
401             $this->assertWithinMargin($mark, $this->quba->get_question_mark($this->slot),
402                     0.000001, 'Expected mark and actual mark differ: %s.');
403         }
404     }
406     /**
407      * @param $condition one or more Expectations. (users varargs).
408      */
409     protected function check_current_output() {
410         $html = $this->quba->render_question($this->slot, $this->displayoptions);
411         foreach (func_get_args() as $condition) {
412             $this->assert($condition, $html);
413         }
414     }
416     protected function get_question_attempt() {
417         return $this->quba->get_question_attempt($this->slot);
418     }
420     protected function get_step_count() {
421         return $this->get_question_attempt()->get_num_steps();
422     }
424     protected function check_step_count($expectednumsteps) {
425         $this->assertEqual($expectednumsteps, $this->get_step_count());
426     }
428     protected function get_step($stepnum) {
429         return $this->get_question_attempt()->get_step($stepnum);
430     }
432     protected function get_contains_question_text_expectation($question) {
433         return new PatternExpectation('/' . preg_quote($question->questiontext) . '/');
434     }
436     protected function get_contains_general_feedback_expectation($question) {
437         return new PatternExpectation('/' . preg_quote($question->generalfeedback) . '/');
438     }
440     protected function get_does_not_contain_correctness_expectation() {
441         return new NoPatternExpectation('/class=\"correctness/');
442     }
444     protected function get_contains_correct_expectation() {
445         return new PatternExpectation('/' . preg_quote(get_string('correct', 'question')) . '/');
446     }
448     protected function get_contains_partcorrect_expectation() {
449         return new PatternExpectation('/' . preg_quote(get_string('partiallycorrect', 'question')) . '/');
450     }
452     protected function get_contains_incorrect_expectation() {
453         return new PatternExpectation('/' . preg_quote(get_string('incorrect', 'question')) . '/');
454     }
456     protected function get_contains_standard_correct_combined_feedback_expectation() {
457         return new PatternExpectation('/' . preg_quote(test_question_maker::STANDARD_OVERALL_CORRECT_FEEDBACK) . '/');
458     }
460     protected function get_contains_standard_partiallycorrect_combined_feedback_expectation() {
461         return new PatternExpectation('/' . preg_quote(test_question_maker::STANDARD_OVERALL_PARTIALLYCORRECT_FEEDBACK) . '/');
462     }
464     protected function get_contains_standard_incorrect_combined_feedback_expectation() {
465         return new PatternExpectation('/' . preg_quote(test_question_maker::STANDARD_OVERALL_INCORRECT_FEEDBACK) . '/');
466     }
468     protected function get_does_not_contain_feedback_expectation() {
469         return new NoPatternExpectation('/class="feedback"/');
470     }
472     protected function get_does_not_contain_num_parts_correct() {
473         return new NoPatternExpectation('/class="numpartscorrect"/');
474     }
476     protected function get_contains_num_parts_correct($num) {
477         $a = new stdClass();
478         $a->num = $num;
479         return new PatternExpectation('/<div class="numpartscorrect">' .
480                 preg_quote(get_string('yougotnright', 'question', $a)) . '/');
481     }
483     protected function get_does_not_contain_specific_feedback_expectation() {
484         return new NoPatternExpectation('/class="specificfeedback"/');
485     }
487     protected function get_contains_validation_error_expectation() {
488         return new ContainsTagWithAttribute('div', 'class', 'validationerror');
489     }
491     protected function get_does_not_contain_validation_error_expectation() {
492         return new NoPatternExpectation('/class="validationerror"/');
493     }
495     protected function get_contains_mark_summary($mark) {
496         $a = new stdClass();
497         $a->mark = format_float($mark, $this->displayoptions->markdp);
498         $a->max = format_float($this->quba->get_question_max_mark($this->slot),
499                 $this->displayoptions->markdp);
500         return new PatternExpectation('/' .
501                 preg_quote(get_string('markoutofmax', 'question', $a)) . '/');
502     }
504     protected function get_contains_marked_out_of_summary() {
505         $max = format_float($this->quba->get_question_max_mark($this->slot),
506                 $this->displayoptions->markdp);
507         return new PatternExpectation('/' .
508                 preg_quote(get_string('markedoutofmax', 'question', $max)) . '/');
509     }
511     protected function get_does_not_contain_mark_summary() {
512         return new NoPatternExpectation('/<div class="grade">/');
513     }
515     protected function get_contains_checkbox_expectation($baseattr, $enabled, $checked) {
516         $expectedattributes = $baseattr;
517         $forbiddenattributes = array();
518         $expectedattributes['type'] = 'checkbox';
519         if ($enabled === true) {
520             $forbiddenattributes['disabled'] = 'disabled';
521         } else if ($enabled === false) {
522             $expectedattributes['disabled'] = 'disabled';
523         }
524         if ($checked === true) {
525             $expectedattributes['checked'] = 'checked';
526         } else if ($checked === false) {
527             $forbiddenattributes['checked'] = 'checked';
528         }
529         return new ContainsTagWithAttributes('input', $expectedattributes, $forbiddenattributes);
530     }
532     protected function get_contains_mc_checkbox_expectation($index, $enabled = null, $checked = null) {
533         return $this->get_contains_checkbox_expectation(array(
534                 'name' => $this->quba->get_field_prefix($this->slot) . $index,
535                 'value' => 1,
536                 ), $enabled, $checked);
537     }
539     protected function get_contains_radio_expectation($baseattr, $enabled, $checked) {
540         $expectedattributes = $baseattr;
541         $forbiddenattributes = array();
542         $expectedattributes['type'] = 'radio';
543         if ($enabled === true) {
544             $forbiddenattributes['disabled'] = 'disabled';
545         } else if ($enabled === false) {
546             $expectedattributes['disabled'] = 'disabled';
547         }
548         if ($checked === true) {
549             $expectedattributes['checked'] = 'checked';
550         } else if ($checked === false) {
551             $forbiddenattributes['checked'] = 'checked';
552         }
553         return new ContainsTagWithAttributes('input', $expectedattributes, $forbiddenattributes);
554     }
556     protected function get_contains_mc_radio_expectation($index, $enabled = null, $checked = null) {
557         return $this->get_contains_radio_expectation(array(
558                 'name' => $this->quba->get_field_prefix($this->slot) . 'answer',
559                 'value' => $index,
560                 ), $enabled, $checked);
561     }
563     protected function get_contains_hidden_expectation($name, $value = null) {
564         $expectedattributes = array('type' => 'hidden', 'name' => s($name));
565         if (!is_null($value)) {
566             $expectedattributes['value'] = s($value);
567         }
568         return new ContainsTagWithAttributes('input', $expectedattributes);
569     }
571     protected function get_does_not_contain_hidden_expectation($name, $value = null) {
572         $expectedattributes = array('type' => 'hidden', 'name' => s($name));
573         if (!is_null($value)) {
574             $expectedattributes['value'] = s($value);
575         }
576         return new DoesNotContainTagWithAttributes('input', $expectedattributes);
577     }
579     protected function get_contains_tf_true_radio_expectation($enabled = null, $checked = null) {
580         return $this->get_contains_radio_expectation(array(
581                 'name' => $this->quba->get_field_prefix($this->slot) . 'answer',
582                 'value' => 1,
583                 ), $enabled, $checked);
584     }
586     protected function get_contains_tf_false_radio_expectation($enabled = null, $checked = null) {
587         return $this->get_contains_radio_expectation(array(
588                 'name' => $this->quba->get_field_prefix($this->slot) . 'answer',
589                 'value' => 0,
590                 ), $enabled, $checked);
591     }
593     protected function get_contains_cbm_radio_expectation($certainty, $enabled = null, $checked = null) {
594         return $this->get_contains_radio_expectation(array(
595                 'name' => $this->quba->get_field_prefix($this->slot) . '-certainty',
596                 'value' => $certainty,
597                 ), $enabled, $checked);
598     }
600     protected function get_contains_button_expectation($name, $value = null, $enabled = null) {
601         $expectedattributes = array(
602             'type' => 'submit',
603             'name' => $name,
604         );
605         $forbiddenattributes = array();
606         if (!is_null($value)) {
607             $expectedattributes['value'] = $value;
608         }
609         if ($enabled === true) {
610             $forbiddenattributes['disabled'] = 'disabled';
611         } else if ($enabled === false) {
612             $expectedattributes['disabled'] = 'disabled';
613         }
614         return new ContainsTagWithAttributes('input', $expectedattributes, $forbiddenattributes);
615     }
617     protected function get_contains_submit_button_expectation($enabled = null) {
618         return $this->get_contains_button_expectation(
619                 $this->quba->get_field_prefix($this->slot) . '-submit', null, $enabled);
620     }
622     protected function get_tries_remaining_expectation($n) {
623         return new PatternExpectation('/' . preg_quote(get_string('triesremaining', 'qbehaviour_interactive', $n)) . '/');
624     }
626     protected function get_invalid_answer_expectation() {
627         return new PatternExpectation('/' . preg_quote(get_string('invalidanswer', 'question')) . '/');
628     }
630     protected function get_contains_try_again_button_expectation($enabled = null) {
631         $expectedattributes = array(
632             'type' => 'submit',
633             'name' => $this->quba->get_field_prefix($this->slot) . '-tryagain',
634         );
635         $forbiddenattributes = array();
636         if ($enabled === true) {
637             $forbiddenattributes['disabled'] = 'disabled';
638         } else if ($enabled === false) {
639             $expectedattributes['disabled'] = 'disabled';
640         }
641         return new ContainsTagWithAttributes('input', $expectedattributes, $forbiddenattributes);
642     }
644     protected function get_does_not_contain_try_again_button_expectation() {
645         return new NoPatternExpectation('/name="' .
646                 $this->quba->get_field_prefix($this->slot) . '-tryagain"/');
647     }
649     protected function get_contains_select_expectation($name, $choices,
650             $selected = null, $enabled = null) {
651         $fullname = $this->quba->get_field_prefix($this->slot) . $name;
652         return new ContainsSelectExpectation($fullname, $choices, $selected, $enabled);
653     }
655     protected function get_mc_right_answer_index($mc) {
656         $order = $mc->get_order($this->get_question_attempt());
657         foreach ($order as $i => $ansid) {
658             if ($mc->answers[$ansid]->fraction == 1) {
659                 return $i;
660             }
661         }
662         $this->fail('This multiple choice question does not seem to have a right answer!');
663     }
665     protected function get_no_hint_visible_expectation() {
666         return new NoPatternExpectation('/class="hint"/');
667     }
669     protected function get_contains_hint_expectation($hinttext) {
670         // Does not currently verify hint text.
671         return new ContainsTagWithAttribute('div', 'class', 'hint');
672     }