2 // This file is part of Moodle - http://moodle.org/
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.
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.
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/>.
19 * Numerical question renderer class.
21 * @package qtype_numerical
22 * @copyright 2009 The Open University
23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
28 * Generates the output for short answer questions.
30 * @copyright 2009 The Open University
31 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
33 class qtype_numerical_renderer
extends qtype_renderer
{
34 public function formulation_and_controls(question_attempt
$qa,
35 question_display_options
$options) {
37 $question = $qa->get_question();
38 $currentanswer = $qa->get_last_qt_var('answer');
39 if ($question->has_separate_unit_field()) {
40 $selectedunit = $qa->get_last_qt_var('unit');
45 $inputname = $qa->get_qt_field_name('answer');
46 $inputattributes = array(
49 'value' => $currentanswer,
54 if ($options->readonly
) {
55 $inputattributes['readonly'] = 'readonly';
59 if ($options->correctness
) {
60 list($value, $unit, $multiplier) = $question->ap
->apply_units(
61 $currentanswer, $selectedunit);
62 $answer = $question->get_matching_answer($value, $multiplier);
64 $fraction = $question->apply_unit_penalty($answer->fraction
, $answer->unitisright
);
68 $inputattributes['class'] = $this->feedback_class($fraction);
69 $feedbackimg = $this->feedback_image($fraction);
72 $questiontext = $question->format_questiontext($qa);
74 if (preg_match('/_____+/', $questiontext, $matches)) {
75 $placeholder = $matches[0];
76 $inputattributes['size'] = round(strlen($placeholder) * 1.1);
79 $input = html_writer
::empty_tag('input', $inputattributes) . $feedbackimg;
81 if ($question->has_separate_unit_field()) {
82 if ($question->unitdisplay
== qtype_numerical
::UNITRADIO
) {
85 foreach ($question->ap
->get_unit_options() as $unit) {
86 $id = $qa->get_qt_field_name('unit') . '_' . $i++
;
87 $radioattrs = array('type' => 'radio', 'id' => $id, 'value' => $unit,
88 'name' => $qa->get_qt_field_name('unit'));
89 if ($unit == $selectedunit) {
90 $radioattrs['checked'] = 'checked';
92 $choices[] = html_writer
::tag('label',
93 html_writer
::empty_tag('input', $radioattrs) . $unit,
94 array('for' => $id, 'class' => 'unitchoice'));
97 $unitchoice = html_writer
::tag('span', implode(' ', $choices),
98 array('class' => 'unitchoices'));
100 } else if ($question->unitdisplay
== qtype_numerical
::UNITSELECT
) {
101 $unitchoice = html_writer
::select($question->ap
->get_unit_options(),
102 $qa->get_qt_field_name('unit'), $selectedunit, array(''=>'choosedots'),
103 array('disabled' => $options->readonly
));
106 if ($question->ap
->are_units_before()) {
107 $input = $unitchoice . ' ' . $input;
109 $input = $input . ' ' . $unitchoice;
114 $questiontext = substr_replace($questiontext, $input,
115 strpos($questiontext, $placeholder), strlen($placeholder));
118 $result = html_writer
::tag('div', $questiontext, array('class' => 'qtext'));
121 $result .= html_writer
::start_tag('div', array('class' => 'ablock'));
122 $result .= get_string('answer', 'qtype_shortanswer',
123 html_writer
::tag('div', $input, array('class' => 'answer')));
124 $result .= html_writer
::end_tag('div');
127 if ($qa->get_state() == question_state
::$invalid) {
128 $result .= html_writer
::nonempty_tag('div',
129 $question->get_validation_error(array('answer' => $currentanswer)),
130 array('class' => 'validationerror'));
136 public function specific_feedback(question_attempt
$qa) {
137 $question = $qa->get_question();
139 if ($question->has_separate_unit_field()) {
140 $selectedunit = $qa->get_last_qt_var('unit');
142 $selectedunit = null;
144 list($value, $unit, $multiplier) = $question->ap
->apply_units(
145 $qa->get_last_qt_var('answer'), $selectedunit);
146 $answer = $question->get_matching_answer($value, $multiplier);
148 if ($answer && $answer->feedback
) {
149 $feedback = $question->format_text($answer->feedback
, $answer->feedbackformat
,
150 $qa, 'question', 'answerfeedback', $answer->id
);
155 if ($question->unitgradingtype
&& !$question->ap
->is_known_unit($unit)) {
156 $feedback .= html_writer
::tag('p', get_string('unitincorrect', 'qtype_numerical'));
162 public function correct_response(question_attempt
$qa) {
163 $question = $qa->get_question();
164 $answer = $question->get_correct_answer();
169 $response = str_replace('.', $question->ap
->get_point(), $answer->answer
);
170 if ($question->unitdisplay
!= qtype_numerical
::UNITNONE
) {
171 $response = $question->ap
->add_unit($response);
174 return get_string('correctansweris', 'qtype_shortanswer', $response);