See Release Notes
Long Term Support Release
<?php // This file is part of Moodle - http://moodle.org/ // // Moodle is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Moodle is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with Moodle. If not, see <http://www.gnu.org/licenses/>. /** * mod_feedback data generator. * * @package mod_feedback * @category test * @copyright 2013 Ankit Agarwal * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ defined('MOODLE_INTERNAL') || die(); /** * mod_feedback data generator class. * * @package mod_feedback * @category test * @copyright 2013 Ankit Agarwal * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class mod_feedback_generator extends testing_module_generator { public function create_instance($record = null, array $options = null) { global $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $record = (object)(array)$record; if (!isset($record->anonymous)) { $record->anonymous = FEEDBACK_ANONYMOUS_YES; } if (!isset($record->email_notification)) { $record->email_notification = 0; } if (!isset($record->multiple_submit)) { $record->multiple_submit = 0; } if (!isset($record->autonumbering)) { $record->autonumbering = 0; } if (!isset($record->site_after_submit)) { $record->site_after_submit = ''; } if (!isset($record->page_after_submit)) { $record->page_after_submit = 'This is page after submit'; } if (!isset($record->page_after_submitformat)) { $record->page_after_submitformat = FORMAT_MOODLE; } if (!isset($record->publish_stats)) { $record->publish_stats = 0; } if (!isset($record->timeopen)) { $record->timeopen = 0; } if (!isset($record->timeclose)) { $record->timeclose = 0; } if (!isset($record->timemodified)) { $record->timemodified = time(); } if (!isset($record->completionsubmit)) { $record->completionsubmit = 0; } // Hack to bypass draft processing of feedback_add_instance. $record->page_after_submit_editor['itemid'] = false; return parent::create_instance($record, (array)$options); } /** * Create question. * * @param array $data Question data * @return mixed Question instance */ public function create_question(array $data) { global $DB; $questiontype = $data['questiontype'] ?? 'textfield'; $cm = get_coursemodule_from_id('feedback', $data['cmid']); $feedback = $DB->get_record('feedback', ['id' => $cm->instance]); unset($data['questiontype']); unset($data['cmid']); if (isset($data['values'])) { $data['values'] = $this->format_item_values($questiontype, $data['values']); } return call_user_func([$this, "create_item_{$questiontype}"], $feedback, $data); } /** * Create response. * * @param array $data Response data. * @return stdClass feedback_completed response instance. */ public function create_response(array $data): stdClass { global $DB; $userid = $data['userid']; $responsenumber = null; $cm = get_coursemodule_from_id('feedback', $data['cmid']); $feedback = $DB->get_record('feedback', ['id' => $cm->instance]); $answers = []; if (isset($data['responsenumber']) && trim($data['responsenumber']) !== '') { $responsenumber = $data['responsenumber']; } if (isset($data['anonymous']) && trim($data['anonymous']) !== '') { $anonymous = filter_var(trim($data['anonymous']), FILTER_VALIDATE_BOOLEAN); $feedback->anonymous = $anonymous ? FEEDBACK_ANONYMOUS_YES : FEEDBACK_ANONYMOUS_NO; } unset($data['cmid']); unset($data['userid']); unset($data['anonymous']); unset($data['responsenumber']); foreach ($data as $question => $response) { $item = $DB->get_record('feedback_item', ['name' => trim($question)], '*', MUST_EXIST); $answers["{$item->typ}_{$item->id}"] = $this->get_item_response_value($item, $response); } $feedbackcompletion = new mod_feedback_completion( $feedback, $cm, $cm->course, false, null, $feedback->anonymous === FEEDBACK_ANONYMOUS_YES ? null : $userid, $userid ); if (!$feedbackcompletion->can_complete()) { throw new coding_exception("User {$userid} cannot complete this feedback activity."); } if (!$feedbackcompletion->is_open()) { throw new coding_exception("This activity is not open."); } $feedbackcompletion->set_module_viewed(); $feedbackcompletion->save_response_tmp((object) $answers); $feedbackcompletion->save_response(); $completed = $feedbackcompletion->get_completed(); if (!is_null($responsenumber)) { $DB->update_record('feedback_completed', [ 'id' => $completed->id, 'random_response' => $responsenumber, ]); } return $completed; } /** * Create info question item. * * @param object $feedback feedback record * @param array $record (optional) to override default values< * @return int> * @return stdClass*/ public function create_item_info($feedback, $record = array()) { global $DB, $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $itemobj = feedback_get_item_class('info'); $position = $DB->count_records('feedback_item', array('feedback' => $feedback->id)) + 1; $record = (array)$record + array( 'id' => 0, 'feedback' => $feedback->id, 'template' => 0, 'name' => 'Feedback question item ' . $position, 'label' => 'Feedback label ' . $position,< 'presentation' => $itemobj::MODE_COURSE,> 'presentation' => \feedback_item_info::MODE_COURSE,'typ' => 'info', 'hasvalue' => 0, 'position' => $position, 'required' => 0, 'dependitem' => 0, 'dependvalue' => '', 'options' => '', ); $itemobj->set_data((object) $record); return $itemobj->save_item(); } /** * Create label question item. * * @param object $feedback feedback record * @param array $record (optional) to override default values< * @return int> * @return stdClass*/ public function create_item_label($feedback, $record = array()) { global $DB, $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $itemobj = feedback_get_item_class('label'); $position = $DB->count_records('feedback_item', array('feedback' => $feedback->id)) + 1; $record = (array)$record + array( 'id' => 0, 'feedback' => $feedback->id, 'template' => 0, 'name' => 'label', 'label' => '', 'presentation' => '', 'typ' => 'label', 'hasvalue' => 0, 'position' => $position, 'required' => 0, 'dependitem' => 0, 'dependvalue' => '', 'options' => '', ); if (!isset($record['presentation_editor'])) { $record['presentation_editor'] = array( 'text' => "The label $position text goes here", 'format' => FORMAT_HTML, 'itemid' => 0 ); } $itemobj->set_data((object) $record); return $itemobj->save_item(); } /** * Create multichoice question item. * * @param object $feedback feedback record * @param array $record (optional) to override default values< * @return int> * @return stdClass*/ public function create_item_multichoice($feedback, $record = array()) { global $DB, $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $itemobj = feedback_get_item_class('multichoice'); $position = $DB->count_records('feedback_item', array('feedback' => $feedback->id)) + 1; $record = (array)$record + array( 'id' => 0, 'feedback' => $feedback->id, 'template' => 0, 'name' => 'Feedback question item ' . $position, 'label' => 'Feedback label ' . $position, 'presentation' => '', 'typ' => 'multichoice', 'hasvalue' => 0, 'position' => $position, 'required' => 0, 'dependitem' => 0, 'dependvalue' => '', 'options' => '', 'subtype' => 'r', 'horizontal' => 0, 'hidenoselect' => 1, 'ignoreempty' => 0, 'values' => "a\nb\nc\nd\ne" ); $presentation = str_replace("\n", FEEDBACK_MULTICHOICE_LINE_SEP, trim($record['values'])); if ($record['horizontal'] == 1 AND $record['subtype'] != 'd') { $presentation .= FEEDBACK_MULTICHOICE_ADJUST_SEP.'1'; } $record['presentation'] = $record['subtype'].FEEDBACK_MULTICHOICE_TYPE_SEP.$presentation; $itemobj->set_data((object) $record); return $itemobj->save_item(); } /** * Create multichoicerated question item. * * @param object $feedback feedback record * @param array $record (optional) to override default values< * @return int> * @return stdClass*/ public function create_item_multichoicerated($feedback, $record = array()) { global $DB, $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $itemobj = feedback_get_item_class('multichoicerated'); $position = $DB->count_records('feedback_item', array('feedback' => $feedback->id)) + 1; $record = (array)$record + array( 'id' => 0, 'feedback' => $feedback->id, 'template' => 0, 'name' => 'Feedback question item ' . $position, 'label' => 'Feedback label ' . $position, 'presentation' => '', 'typ' => 'multichoicerated', 'hasvalue' => 0, 'position' => $position, 'required' => 0, 'dependitem' => 0, 'dependvalue' => '', 'options' => '', 'subtype' => 'r', 'horizontal' => 0, 'hidenoselect' => 1, 'ignoreempty' => 0, 'values' => "0/a\n1/b\n2/c\n3/d\n4/e" ); $itemobj = new feedback_item_multichoicerated(); $presentation = $itemobj->prepare_presentation_values_save(trim($record['values']), FEEDBACK_MULTICHOICERATED_VALUE_SEP2, FEEDBACK_MULTICHOICERATED_VALUE_SEP); if ($record['horizontal'] == 1 AND $record['subtype'] != 'd') { $presentation .= FEEDBACK_MULTICHOICERATED_ADJUST_SEP.'1'; } $record['presentation'] = $record['subtype'].FEEDBACK_MULTICHOICERATED_TYPE_SEP.$presentation; $itemobj->set_data((object) $record); return $itemobj->save_item(); } /** * Create numeric question item. * * @param object $feedback feedback record * @param array $record (optional) to override default values< * @return int> * @return stdClass*/ public function create_item_numeric($feedback, $record = array()) { global $DB, $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $itemobj = feedback_get_item_class('numeric'); $position = $DB->count_records('feedback_item', array('feedback' => $feedback->id)) + 1; $record = (array)$record + array( 'id' => 0, 'feedback' => $feedback->id, 'template' => 0, 'name' => 'Feedback question item ' . $position, 'label' => 'Feedback label ' . $position, 'presentation' => '', 'typ' => 'numeric', 'hasvalue' => 0, 'position' => $position, 'required' => 0, 'dependitem' => 0, 'dependvalue' => '', 'options' => '', 'rangefrom' => '-', 'rangeto' => '-', ); if ($record['rangefrom'] === '-' OR $record['rangeto'] === '-') { $record['presentation'] = $record['rangefrom'] . '|'. $record['rangeto']; } else if ($record['rangefrom'] > $record['rangeto']) { $record['presentation'] = $record['rangeto'] . '|'. $record['rangefrom']; } else { $record['presentation'] = $record['rangefrom'] . '|'. $record['rangeto']; } $itemobj->set_data((object) $record); return $itemobj->save_item(); } /** * Create textarea question item. * * @param object $feedback feedback record * @param array $record (optional) to override default values< * @return int> * @return stdClass*/ public function create_item_textarea($feedback, $record = array()) { global $DB, $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $itemobj = feedback_get_item_class('textarea'); $position = $DB->count_records('feedback_item', array('feedback' => $feedback->id)) + 1; $record = (array)$record + array( 'id' => 0, 'feedback' => $feedback->id, 'template' => 0, 'name' => 'Feedback question item ' . $position, 'label' => 'Feedback label ' . $position, 'presentation' => '', 'typ' => 'textarea', 'hasvalue' => 0, 'position' => $position, 'required' => 0, 'dependitem' => 0, 'dependvalue' => '', 'options' => '', 'itemwidth' => '40', 'itemheight' => '20', ); $record['presentation'] = $record['itemwidth'] . '|'. $record['itemheight']; $itemobj->set_data((object) $record); return $itemobj->save_item(); } /** * Create textfield question item. * * @param object $feedback feedback record * @param array $record (optional) to override default values< * @return int> * @return stdClass*/ public function create_item_textfield($feedback, $record = array()) { global $DB, $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); $itemobj = feedback_get_item_class('textfield'); $position = $DB->count_records('feedback_item', array('feedback' => $feedback->id)) + 1; $record = (array)$record + array( 'id' => 0, 'feedback' => $feedback->id, 'template' => 0, 'name' => 'Feedback question item ' . $position, 'label' => 'Feedback label ' . $position, 'presentation' => '', 'typ' => 'textfield', 'hasvalue' => 0, 'position' => $position, 'required' => 0, 'dependitem' => 0, 'dependvalue' => '', 'options' => '', 'itemsize' => '20', 'itemmaxlength' => '30', ); $record['presentation'] = $record['itemsize'] . '|'. $record['itemmaxlength']; $itemobj->set_data((object) $record); return $itemobj->save_item(); } /** * Create pagebreak. * * @param object $feedback feedback record< * @return mixed false if there already is a pagebreak on last position or the id of the pagebreak-item> * @return int|false false if there already is a pagebreak on last position or the id of the pagebreak-item*/ public function create_item_pagebreak($feedback) { global $CFG; require_once($CFG->dirroot.'/mod/feedback/lib.php'); return feedback_create_pagebreak($feedback->id); } /** * Format feedback item values. * * This method will replace newline characters with the proper line separator for each question type. * * @param string $questiontype Question types * @param string $values Values * @return string Formatted values */ protected function format_item_values(string $questiontype, string $values): string { global $CFG; if (!file_exists($CFG->dirroot.'/mod/feedback/item/'.$questiontype.'/lib.php')) { throw new coding_exception("Question type '$questiontype' not found"); } require_once($CFG->dirroot.'/mod/feedback/item/'.$questiontype.'/lib.php'); $questiontype = strtoupper($questiontype); if (defined("FEEDBACK_{$questiontype}_LINE_SEP")) { return implode(constant("FEEDBACK_{$questiontype}_LINE_SEP"), explode('\n', $values)); } return $values; } /** * Given a response to a feedback item, return its corresponding value. * * @param mixed $record Item record * @param string $response Response name * @return int|string Response value */ protected function get_item_response_value($record, string $response) { if (strpos($record->typ, 'multichoice') === 0) { $item = feedback_get_item_class($record->typ); return $this->get_choice_item_response_value($item, $record, $response); } return $response; } /** * Given a response to a feedback choice item, return its corresponding value. * * @param feedback_item_base $item Feedback item * @param mixed $record Item record * @param string $response Response * @param int $offset Choice to start looking from * @return int Response choice index */ protected function get_choice_item_response_value(feedback_item_base $item, $record, string $response, int $offset = 1): int { $printval = $item->get_printval($record, (object) ['value' => $offset]); if (empty($printval)) { throw new coding_exception("Value '$offset' not found"); } if ($printval === $response) { return $offset; } return $this->get_choice_item_response_value($item, $record, $response, $offset + 1); } }