fb5f6fc569850d0ead58728896cca448daf880ab
[moodle.git] / mod / feedback / item / numeric / lib.php
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/>.
17 defined('MOODLE_INTERNAL') OR die('not allowed');
18 require_once($CFG->dirroot.'/mod/feedback/item/feedback_item_class.php');
20 class feedback_item_numeric extends feedback_item_base {
21     protected $type = "numeric";
23     public function build_editform($item, $feedback, $cm) {
24         global $DB, $CFG;
25         require_once('numeric_form.php');
27         //get the lastposition number of the feedback_items
28         $position = $item->position;
29         $lastposition = $DB->count_records('feedback_item', array('feedback'=>$feedback->id));
30         if ($position == -1) {
31             $i_formselect_last = $lastposition + 1;
32             $i_formselect_value = $lastposition + 1;
33             $item->position = $lastposition + 1;
34         } else {
35             $i_formselect_last = $lastposition;
36             $i_formselect_value = $item->position;
37         }
38         //the elements for position dropdownlist
39         $positionlist = array_slice(range(0, $i_formselect_last), 1, $i_formselect_last, true);
41         $item->presentation = empty($item->presentation) ? '' : $item->presentation;
43         $range_from_to = explode('|', $item->presentation);
44         if (isset($range_from_to[0]) AND is_numeric($range_from_to[0])) {
45             $range_from = $this->format_float($range_from_to[0]);
46         } else {
47             $range_from = '-';
48         }
50         if (isset($range_from_to[1]) AND is_numeric($range_from_to[1])) {
51             $range_to = $this->format_float($range_from_to[1]);
52         } else {
53             $range_to = '-';
54         }
56         $item->rangefrom = $range_from;
57         $item->rangeto = $range_to;
59         //all items for dependitem
60         $feedbackitems = feedback_get_depend_candidates_for_item($feedback, $item);
61         $commonparams = array('cmid'=>$cm->id,
62                              'id'=>isset($item->id) ? $item->id : null,
63                              'typ'=>$item->typ,
64                              'items'=>$feedbackitems,
65                              'feedback'=>$feedback->id);
67         //build the form
68         $customdata = array('item' => $item,
69                             'common' => $commonparams,
70                             'positionlist' => $positionlist,
71                             'position' => $position);
73         $this->item_form = new feedback_numeric_form('edit_item.php', $customdata);
74     }
76     public function save_item() {
77         global $DB;
79         if (!$item = $this->item_form->get_data()) {
80             return false;
81         }
83         if (isset($item->clone_item) AND $item->clone_item) {
84             $item->id = ''; //to clone this item
85             $item->position++;
86         }
88         $item->hasvalue = $this->get_hasvalue();
89         if (!$item->id) {
90             $item->id = $DB->insert_record('feedback_item', $item);
91         } else {
92             $DB->update_record('feedback_item', $item);
93         }
95         return $DB->get_record('feedback_item', array('id'=>$item->id));
96     }
98     /**
99      * Helper function for collected data, both for analysis page and export to excel
100      *
101      * @param stdClass $item the db-object from feedback_item
102      * @param int $groupid
103      * @param int $courseid
104      * @return stdClass
105      */
106     protected function get_analysed($item, $groupid = false, $courseid = false) {
107         global $DB;
109         $analysed = new stdClass();
110         $analysed->data = array();
111         $analysed->name = $item->name;
112         $values = feedback_get_group_values($item, $groupid, $courseid);
114         $avg = 0.0;
115         $counter = 0;
116         if ($values) {
117             $data = array();
118             foreach ($values as $value) {
119                 if (is_numeric($value->value)) {
120                     $data[] = $value->value;
121                     $avg += $value->value;
122                     $counter++;
123                 }
124             }
125             $avg = $counter > 0 ? $avg / $counter : null;
126             $analysed->data = $data;
127             $analysed->avg = $avg;
128         }
129         return $analysed;
130     }
132     public function get_printval($item, $value) {
133         if (!isset($value->value)) {
134             return '';
135         }
137         return $value->value;
138     }
140     public function print_analysed($item, $itemnr = '', $groupid = false, $courseid = false) {
142         $values = $this->get_analysed($item, $groupid, $courseid);
144         if (isset($values->data) AND is_array($values->data)) {
145             echo '<tr><th colspan="2" align="left">';
146             echo $itemnr . ' ';
147             if (strval($item->label) !== '') {
148                 echo '('. format_string($item->label).') ';
149             }
150             echo format_text($item->name, FORMAT_HTML, array('noclean' => true, 'para' => false));
151             echo '</th></tr>';
153             foreach ($values->data as $value) {
154                 echo '<tr><td colspan="2" class="singlevalue">';
155                 echo $this->format_float($value);
156                 echo '</td></tr>';
157             }
159             if (isset($values->avg)) {
160                 $avg = format_float($values->avg, 2);
161             } else {
162                 $avg = '-';
163             }
164             echo '<tr><td colspan="2"><b>';
165             echo get_string('average', 'feedback').': '.$avg;
166             echo '</b></td></tr>';
167         }
168     }
170     public function excelprint_item(&$worksheet, $row_offset,
171                              $xls_formats, $item,
172                              $groupid, $courseid = false) {
174         $analysed_item = $this->get_analysed($item, $groupid, $courseid);
176         $worksheet->write_string($row_offset, 0, $item->label, $xls_formats->head2);
177         $worksheet->write_string($row_offset, 1, $item->name, $xls_formats->head2);
178         $data = $analysed_item->data;
179         if (is_array($data)) {
181             // Export average.
182             $worksheet->write_string($row_offset,
183                                      2,
184                                      get_string('average', 'feedback'),
185                                      $xls_formats->value_bold);
187             if (isset($analysed_item->avg)) {
188                 $worksheet->write_number($row_offset + 1,
189                                          2,
190                                          $analysed_item->avg,
191                                          $xls_formats->value_bold);
192             } else {
193                 $worksheet->write_string($row_offset + 1,
194                                          2,
195                                          '',
196                                          $xls_formats->value_bold);
197             }
198             $row_offset++;
199         }
200         $row_offset++;
201         return $row_offset;
202     }
204     /**
205      * Prints the float nicely in the localized format
206      *
207      * Similar to format_float() but automatically calculates the number of decimal places
208      *
209      * @param float $value The float to print
210      * @return string
211      */
212     protected function format_float($value) {
213         if (!is_numeric($value)) {
214             return null;
215         }
216         $decimal = is_int($value) ? 0 : strlen(substr(strrchr($value, '.'), 1));
217         return format_float($value, $decimal);
218     }
220     /**
221      * Returns human-readable boundaries (min - max)
222      * @param stdClass $item
223      * @return string
224      */
225     protected function get_boundaries_for_display($item) {
226         list($rangefrom, $rangeto) = explode('|', $item->presentation);
227         if (!isset($rangefrom) || !is_numeric($rangefrom)) {
228             $rangefrom = null;
229         }
230         if (!isset($rangeto) || !is_numeric($rangeto)) {
231             $rangeto = null;
232         }
234         if (is_null($rangefrom) && is_numeric($rangeto)) {
235             return ' (' . get_string('maximal', 'feedback') .
236                         ': ' . $this->format_float($rangeto) . ')';
237         }
238         if (is_numeric($rangefrom) && is_null($rangeto)) {
239             return ' (' . get_string('minimal', 'feedback') .
240                         ': ' . $this->format_float($rangefrom) . ')';
241         }
242         if (is_null($rangefrom) && is_null($rangeto)) {
243             return '';
244         }
245         return ' (' . $this->format_float($rangefrom) .
246                 ' - ' . $this->format_float($rangeto) . ')';
247     }
249     /**
250      * Returns the postfix to be appended to the display name that is based on other settings
251      *
252      * @param stdClass $item
253      * @return string
254      */
255     public function get_display_name_postfix($item) {
256         return html_writer::span($this->get_boundaries_for_display($item), 'boundaries');
257     }
259     /**
260      * Adds an input element to the complete form
261      *
262      * @param stdClass $item
263      * @param mod_feedback_complete_form $form
264      */
265     public function complete_form_element($item, $form) {
266         $name = $this->get_display_name($item);
267         $inputname = $item->typ . '_' . $item->id;
268         $form->add_form_element($item,
269                 ['text', $inputname, $name],
270                 true,
271                 false
272                 );
273         $form->set_element_type($inputname, PARAM_NOTAGS);
274         $tmpvalue = $this->format_float($form->get_item_value($item));
275         $form->set_element_default($inputname, $tmpvalue);
277         // Add form validation rule to check for boundaries.
278         $form->add_validation_rule(function($values, $files) use ($item) {
279             $inputname = $item->typ . '_' . $item->id;
280             list($rangefrom, $rangeto) = explode('|', $item->presentation);
281             if (!isset($values[$inputname]) || trim($values[$inputname]) === '') {
282                 return $item->required ? array($inputname => get_string('required')) : true;
283             }
284             $value = unformat_float($values[$inputname], true);
285             if ($value === false) {
286                 return array($inputname => get_string('invalidnum', 'error'));
287             }
288             if ((is_numeric($rangefrom) && $value < floatval($rangefrom)) ||
289                     (is_numeric($rangeto) && $value > floatval($rangeto))) {
290                 return array($inputname => get_string('numberoutofrange', 'feedback'));
291             }
292             return true;
293         });
294     }
296     public function create_value($data) {
297         $data = unformat_float($data, true);
299         if (is_numeric($data)) {
300             $data = floatval($data);
301         } else {
302             $data = '';
303         }
304         return $data;
305     }