From 4a26a1aa152934818044d26bd74438262001234e Mon Sep 17 00:00:00 2001 From: Stephan Robotta Date: Wed, 28 Aug 2024 13:01:46 +0200 Subject: [PATCH] Add new question type file --- classes/question/file.php | 237 ++++++++++++ classes/question/question.php | 2 + classes/responsetype/file.php | 421 +++++++++++++++++++++ classes/responsetype/response/response.php | 3 +- db/install.php | 6 + db/install.xml | 15 + db/upgrade.php | 31 ++ lang/en/questionnaire.php | 2 + lib.php | 13 +- locallib.php | 4 + tests/behat/add_questions.feature | 4 + tests/behat/behat_mod_questionnaire.php | 189 ++++++++- tests/behat/file_question.feature | 80 ++++ tests/fixtures/testfilequestion.pdf | Bin 0 -> 16338 bytes tests/fixtures/testfilequestion2.pdf | Bin 0 -> 16338 bytes version.php | 2 +- 16 files changed, 1002 insertions(+), 7 deletions(-) create mode 100644 classes/question/file.php create mode 100644 classes/responsetype/file.php create mode 100644 tests/behat/file_question.feature create mode 100644 tests/fixtures/testfilequestion.pdf create mode 100644 tests/fixtures/testfilequestion2.pdf diff --git a/classes/question/file.php b/classes/question/file.php new file mode 100644 index 00000000..a77ddda8 --- /dev/null +++ b/classes/question/file.php @@ -0,0 +1,237 @@ +. +namespace mod_questionnaire\question; +use core_media_manager; +use form_filemanager; +use mod_questionnaire\responsetype\response\response; +use moodle_url; +use MoodleQuickForm; + +/** + * This file contains the parent class for text question types. + * + * @author Laurent David + * @author Martin Cornu-Mansuy + * @copyright 2023 onward CALL Learning + * @license http://www.gnu.org/copyleft/gpl.html GNU Public License + * @package mod_questionnaire + */ +class file extends question { + + /** + * Get name. + * + * @return string + */ + public function helpname() { + return 'file'; + } + + /** + * Override and return a form template if provided. Output of question_survey_display is iterpreted based on this. + * + * @return boolean | string + */ + public function question_template() { + return false; + } + + /** + * Override and return a response template if provided. Output of response_survey_display is iterpreted based on this. + * + * @return boolean | string + */ + public function response_template() { + return false; + } + + /** + * Get response class. + * + * @return object|string + */ + protected function responseclass() { + return '\\mod_questionnaire\\responsetype\\file'; + } + + /** + * Survey display output. + * + * @param response $response + * @param object $descendantsdata + * @param bool $blankquestionnaire + * @return string + */ + protected function question_survey_display($response, $descendantsdata, $blankquestionnaire = false) { + global $CFG, $PAGE; + require_once($CFG->libdir . '/filelib.php'); + + $elname = 'q' . $this->id; + // Make sure there is a response, fetch the draft id from the original request. + if (isset($response->answers[$this->id]) && !empty($response->answers[$this->id]) && isset($_REQUEST[$elname . 'draft'])) { + $draftitemid = (int)$_REQUEST[$elname . 'draft']; + } else { + $draftitemid = file_get_submitted_draft_itemid($elname); + } + if ($draftitemid > 0) { + file_prepare_draft_area($draftitemid, $this->context->id, + 'mod_questionnaire', 'file', $this->id, self::get_file_manager_option()); + } else { + $draftitemid = file_get_unused_draft_itemid(); + } + // Filemanager form element implementation is far from optimal, we need to rework this if we ever fix it... + require_once("$CFG->dirroot/lib/form/filemanager.php"); + + $options = array_merge(self::get_file_manager_option(), [ + 'client_id' => uniqid(), + 'itemid' => $draftitemid, + 'target' => $this->id, + 'name' => $elname, + ]); + $fm = new form_filemanager((object)$options); + $output = $PAGE->get_renderer('core', 'files'); + + $html = '
' . + $output->render($fm) . + '' . + '' . + '
'; + + return $html; + } + + /** + * Check question's form data for complete response. + * @param \stdClass $responsedata The data entered into the response. + * @return bool + */ + public function response_complete($responsedata) { + $answered = false; + // If $responsedata is a response object, look through the answers. + if (is_a($responsedata, 'mod_questionnaire\responsetype\response\response') && + isset($responsedata->answers[$this->id]) && !empty($responsedata->answers[$this->id]) + ) { + $answer = reset($responsedata->answers[$this->id]); + $answered = ((int)$answer->value > 0); + } else if (isset($responsedata->{'q'.$this->id})) { + // If $responsedata is webform data, check that it is not empty. + $draftitemid = (int)$responsedata->{'q' . $this->id}; + if ($draftitemid > 0) { + $info = file_get_draft_area_info($draftitemid); + $answered = $info['filecount'] > 0; + } + } + return !($this->required() && ($this->deleted == 'n') && !$answered); + } + + /** + * Get file manager options + * + * @return array + */ + public static function get_file_manager_option() { + return [ + 'mainfile' => '', + 'subdirs' => false, + 'accepted_types' => ['image', '.pdf'], + 'maxfiles' => 1, + ]; + } + + /** + * Response display output. + * + * @param \stdClass $data + * @return string + */ + protected function response_survey_display($data) { + global $PAGE, $CFG; + require_once($CFG->libdir . '/filelib.php'); + require_once($CFG->libdir . '/resourcelib.php'); + if (isset($data->answers[$this->id])) { + $answer = reset($data->answers[$this->id]); + } else { + return ''; + } + $fs = get_file_storage(); + $file = $fs->get_file_by_id($answer->value); + $code = ''; + + if ($file) { + // There is a file. + $moodleurl = moodle_url::make_pluginfile_url( + $file->get_contextid(), + $file->get_component(), + $file->get_filearea(), + $file->get_itemid(), + $file->get_filepath(), + $file->get_filename() + ); + + $mimetype = $file->get_mimetype(); + $title = ''; + + $mediamanager = core_media_manager::instance($PAGE); + $embedoptions = array( + core_media_manager::OPTION_TRUSTED => true, + core_media_manager::OPTION_BLOCK => true, + ); + + if (file_mimetype_in_typegroup($mimetype, 'web_image')) { // It's an image. + $code = resourcelib_embed_image($moodleurl->out(), $title); + + } else if ($mimetype === 'application/pdf') { + // PDF document. + $code = resourcelib_embed_pdf($moodleurl->out(), $title, get_string('view')); + + } else if ($mediamanager->can_embed_url($moodleurl, $embedoptions)) { + // Media (audio/video) file. + $code = $mediamanager->embed_url($moodleurl, $title, 0, 0, $embedoptions); + + } else { + // We need a way to discover if we are loading remote docs inside an iframe. + $moodleurl->param('embed', 1); + + // Anything else - just try object tag enlarged as much as possible. + $code = resourcelib_embed_general($moodleurl, $title, get_string('view'), $mimetype); + } + } + return '
' . $code . '
'; + } + + /** + * Add the length element as hidden. + * + * @param \MoodleQuickForm $mform + * @param string $helpname + * @return \MoodleQuickForm + */ + protected function form_length(MoodleQuickForm $mform, $helpname = '') { + return question::form_length_hidden($mform); + } + + /** + * Add the precise element as hidden. + * + * @param \MoodleQuickForm $mform + * @param string $helpname + * @return \MoodleQuickForm + */ + protected function form_precise(MoodleQuickForm $mform, $helpname = '') { + return question::form_precise_hidden($mform); + } + +} diff --git a/classes/question/question.php b/classes/question/question.php index 7b4594b5..05a1379d 100644 --- a/classes/question/question.php +++ b/classes/question/question.php @@ -42,6 +42,7 @@ define('QUESDATE', 9); define('QUESNUMERIC', 10); define('QUESSLIDER', 11); +define('QUESFILE', 12); define('QUESPAGEBREAK', 99); define('QUESSECTIONTEXT', 100); @@ -123,6 +124,7 @@ abstract class question { QUESPAGEBREAK => 'pagebreak', QUESSECTIONTEXT => 'sectiontext', QUESSLIDER => 'slider', + QUESFILE => 'file', ]; /** @var array $notifications Array of extra messages for display purposes. */ diff --git a/classes/responsetype/file.php b/classes/responsetype/file.php new file mode 100644 index 00000000..0af324c7 --- /dev/null +++ b/classes/responsetype/file.php @@ -0,0 +1,421 @@ +. +namespace mod_questionnaire\responsetype; + +use mod_questionnaire\db\bulk_sql_config; +use moodle_url; + +/** + * Class for text response types. + * + * @author Laurent David + * @author Martin Cornu-Mansuy + * @copyright 2023 onward CALL Learning + * @license http://www.gnu.org/copyleft/gpl.html GNU Public License + * @package mod_questionnaire + */ +class file extends responsetype { + /** + * Provide an array of answer objects from web form data for the question. + * + * @param \stdClass $responsedata All of the responsedata as an object. + * @param \mod_questionnaire\question\question $question + * @return array \mod_questionnaire\responsetype\answer\answer An array of answer objects. + */ + public static function answers_from_webform($responsedata, $question) { + $answers = []; + if (isset($responsedata->{'q' . $question->id}) && (strlen($responsedata->{'q' . $question->id}) > 0)) { + $val = $responsedata->{'q' . $question->id}; + $record = new \stdClass(); + $record->responseid = $responsedata->rid; + $record->questionid = $question->id; + + file_save_draft_area_files($val, $question->context->id, + 'mod_questionnaire', 'file', $val, + \mod_questionnaire\question\file::get_file_manager_option()); + $fs = get_file_storage(); + $files = $fs->get_area_files($question->context->id, 'mod_questionnaire', + 'file', $val, + "itemid, filepath, filename", + false); + if (!empty($files)) { + $file = reset($files); + $record->value = $file->get_id(); + $answers[] = answer\answer::create_from_data($record); + } else { + self::delete_old_response((int)$question->id, (int)$record->responseid); + } + } + return $answers; + } + + /** + * Return an array of answers by question/choice for the given response. Must be implemented by the subclass. + * + * @param int $rid The response id. + * @return array + */ + public static function response_select($rid) { + global $DB; + + $values = []; + $sql = 'SELECT q.id, q.content, a.fileid as aresponse ' . + 'FROM {' . static::response_table() . '} a, {questionnaire_question} q ' . + 'WHERE a.response_id=? AND a.question_id=q.id '; + $records = $DB->get_records_sql($sql, [$rid]); + foreach ($records as $qid => $row) { + unset($row->id); + $row = (array) $row; + $newrow = []; + foreach ($row as $key => $val) { + if (!is_numeric($key)) { + $newrow[] = $val; + } + } + $values[$qid] = $newrow; + $val = array_pop($values[$qid]); + array_push($values[$qid], $val, $val); + } + + return $values; + } + + /** + * Return an array of answer objects by question for the given response id. + * THIS SHOULD REPLACE response_select. + * + * @param int $rid The response id. + * @return array array answer + * @throws \dml_exception + */ + public static function response_answers_by_question($rid) { + global $DB; + + $answers = []; + $sql = 'SELECT id, response_id as responseid, question_id as questionid, 0 as choiceid, fileid as value ' . + 'FROM {' . static::response_table() . '} ' . + 'WHERE response_id = ? '; + $records = $DB->get_records_sql($sql, [$rid]); + foreach ($records as $record) { + $answers[$record->questionid][] = answer\answer::create_from_data($record); + } + + return $answers; + } + + /** + * Delete old entries from the questionnaire_response_file table and also the corresponding entries + * in the files table. + * @param int $questionid + * @param int $responseid + * @return void + * @throws \dml_exception + */ + public static function delete_old_response(int $questionid, int $responseid) { + global $DB; + // Check, if we have an old response file from a former attempt. + $record = $DB->get_record(static::response_table(), [ + 'response_id' => $responseid, + 'question_id' => $questionid, + ]); + if ($record) { + // Old record found, then delete all referenced entries in the files table and then delete this entry. + $DB->delete_records('files', ['component' => 'mod_questionnaire', 'itemid' => $record->itemid]); + $DB->delete_records(self::response_table(), ['id' => $record->id]); + } + } + + /** + * Insert a provided response to the question. + * + * @param \mod_questionnaire\responsetype\response\response|\stdClass $responsedata + * @return bool|int + * @throws \dml_exception + */ + public function insert_response($responsedata) { + global $DB; + + if (!$responsedata instanceof \mod_questionnaire\responsetype\response\response) { + $response = \mod_questionnaire\responsetype\response\response::response_from_webform($responsedata, [$this->question]); + } else { + $response = $responsedata; + } + + if (!empty($response) && isset($response->answers[$this->question->id][0])) { + $record = new \stdClass(); + $record->response_id = $response->id; + $record->question_id = $this->question->id; + $record->fileid = intval(clean_text($response->answers[$this->question->id][0]->value)); + + // Delete any previous attempts. + self::delete_old_response((int)$this->question->id, (int)$response->id); + + // When saving the draft file, the itemid was the same as the draftitemid. This must now be + // corrected to the primary key that is questionaire_response_file.id to have a correct reference. + $recordid = $DB->insert_record(static::response_table(), $record); + if ($recordid) { + $olditem = $DB->get_record('files', ['id' => $record->fileid], 'itemid'); + if (!$olditem) { + return false; + } + $siblings = $DB->get_records('files', + ['component' => 'mod_questionnaire', 'itemid' => $olditem->itemid]); + foreach ($siblings as $sibling) { + if (!self::fix_file_itemid($recordid, $sibling)) { + return false; + } + } + return $recordid; + } + } + return false; + } + + /** + * Update records in the table file with the new given itemid. To do this, the pathnamehash + * needs to be recalculated as well. + * @param int $recordid + * @param \stdClass $filerecord + * @return bool + * @throws \dml_exception + */ + public static function fix_file_itemid(int $recordid, \stdClass $filerecord): bool { + global $DB; + if ((int)$filerecord->itemid === $recordid) { + return true; // Reference is already good, nothing to do. + } + $fs = get_file_storage(); + $file = $fs->get_file_instance($filerecord); + $newhash = $fs->get_pathname_hash($filerecord->contextid, $filerecord->component, + $filerecord->filearea, $recordid, $file->get_filepath(), $file->get_filename()); + $filerecord->itemid = $recordid; + $filerecord->pathnamehash = $newhash; + return $DB->update_record('files', $filerecord); + } + + /** + * Provide the necessary response data table name. Should probably always be used with late static binding 'static::' form + * rather than 'self::' form to allow for class extending. + * + * @return string response table name. + */ + public static function response_table() { + return 'questionnaire_response_file'; + } + + /** + * Provide a template for results screen if defined. + * + * @param bool $pdf + * @return mixed The template string or false/ + */ + public function results_template($pdf = false) { + if ($pdf) { + return 'mod_questionnaire/resultspdf_text'; + } else { + return 'mod_questionnaire/results_text'; + } + } + + /** + * Provide the result information for the specified result records. + * + * @param int|array $rids - A single response id, or array. + * @param string $sort - Optional display sort. + * @param boolean $anonymous - Whether or not responses are anonymous. + * @return string - Display output. + */ + public function display_results($rids = false, $sort = '', $anonymous = false) { + if (is_array($rids)) { + $prtotal = 1; + } else if (is_int($rids)) { + $prtotal = 0; + } + if ($rows = $this->get_results($rids, $anonymous)) { + $numrespondents = count($rids); + $numresponses = count($rows); + $pagetags = $this->get_results_tags($rows, $numrespondents, $numresponses, $prtotal); + } else { + $pagetags = ""; + } + return $pagetags; + } + + /** + * Provide the result information for the specified result records. + * + * @param int|array $rids - A single response id, or array. + * @param boolean $anonymous - Whether or not responses are anonymous. + * @return array - Array of data records. + */ + public function get_results($rids = false, $anonymous = false) { + global $DB; + + $rsql = ''; + if (!empty($rids)) { + list($rsql, $params) = $DB->get_in_or_equal($rids); + $rsql = ' AND response_id ' . $rsql; + } + + if ($anonymous) { + $sql = 'SELECT t.id, t.fileid, r.submitted AS submitted, ' . + 'r.questionnaireid, r.id AS rid ' . + 'FROM {' . static::response_table() . '} t, ' . + '{questionnaire_response} r ' . + 'WHERE question_id=' . $this->question->id . $rsql . + ' AND t.response_id = r.id ' . + 'ORDER BY r.submitted DESC'; + } else { + $sql = 'SELECT t.id, t.fileid, r.submitted AS submitted, r.userid, u.username AS username, ' . + 'u.id as usrid, ' . + 'r.questionnaireid, r.id AS rid ' . + 'FROM {' . static::response_table() . '} t, ' . + '{questionnaire_response} r, ' . + '{user} u ' . + 'WHERE question_id=' . $this->question->id . $rsql . + ' AND t.response_id = r.id' . + ' AND u.id = r.userid ' . + 'ORDER BY u.lastname, u.firstname, r.submitted'; + } + return $DB->get_records_sql($sql, $params); + } + + /** + * Override the results tags function for templates for questions with dates. + * + * @param array $weights + * @param int $participants Number of questionnaire participants. + * @param int $respondents Number of question respondents. + * @param bool $showtotals + * @param string $sort + * @return \stdClass + */ + public function get_results_tags($weights, $participants, $respondents, $showtotals = 1, $sort = '') { + $pagetags = new \stdClass(); + if ($respondents == 0) { + return $pagetags; + } + + // If array element is an object, outputting non-numeric responses. + if (is_object(reset($weights))) { + global $CFG, $SESSION, $questionnaire, $DB; + $viewsingleresponse = $questionnaire->capabilities->viewsingleresponse; + $nonanonymous = $questionnaire->respondenttype != 'anonymous'; + if ($viewsingleresponse && $nonanonymous) { + $currentgroupid = ''; + if (isset($SESSION->questionnaire->currentgroupid)) { + $currentgroupid = $SESSION->questionnaire->currentgroupid; + } + $url = $CFG->wwwroot . '/mod/questionnaire/report.php?action=vresp&sid=' . $questionnaire->survey->id . + '¤tgroupid=' . $currentgroupid; + } + $users = []; + $evencolor = false; + foreach ($weights as $row) { + $response = new \stdClass(); + $fs = get_file_storage(); + $file = $fs->get_file_by_id($row->fileid); + + if ($file) { + // There is a file. + $imageurl = moodle_url::make_pluginfile_url( + $file->get_contextid(), + $file->get_component(), + $file->get_filearea(), + $file->get_itemid(), + $file->get_filepath(), + $file->get_filename()); + + $response->text = \html_writer::link($imageurl, $file->get_filename()); + if ($viewsingleresponse && $nonanonymous) { + $rurl = $url . '&rid=' . $row->rid . '&individualresponse=1'; + $title = userdate($row->submitted); + if (!isset($users[$row->userid])) { + $users[$row->userid] = $DB->get_record('user', ['id' => $row->userid]); + } + $response->respondent = + '' . fullname($users[$row->userid]) . ''; + } + } else { + $response->respondent = ''; + } + // The 'evencolor' attribute is used by the PDF template. + $response->evencolor = $evencolor; + $pagetags->responses[] = (object) ['response' => $response]; + $evencolor = !$evencolor; + } + + if ($showtotals == 1) { + $pagetags->total = new \stdClass(); + $pagetags->total->total = "$respondents/$participants"; + } + } else { + $nbresponses = 0; + $sum = 0; + $strtotal = get_string('totalofnumbers', 'questionnaire'); + $straverage = get_string('average', 'questionnaire'); + + if (!empty($weights) && is_array($weights)) { + ksort($weights); + $evencolor = false; + foreach ($weights as $text => $num) { + $response = new \stdClass(); + $response->text = $text; + $response->respondent = $num; + // The 'evencolor' attribute is used by the PDF template. + $response->evencolor = $evencolor; + $nbresponses += $num; + $sum += $text * $num; + $evencolor = !$evencolor; + $pagetags->responses[] = (object) ['response' => $response]; + } + + $response = new \stdClass(); + $response->text = $sum; + $response->respondent = $strtotal; + $response->evencolor = $evencolor; + $pagetags->responses[] = (object) ['response' => $response]; + $evencolor = !$evencolor; + + $response = new \stdClass(); + $response->respondent = $straverage; + $avg = $sum / $nbresponses; + $response->text = sprintf('%.' . $this->question->precise . 'f', $avg); + $response->evencolor = $evencolor; + $pagetags->responses[] = (object) ['response' => $response]; + $evencolor = !$evencolor; + + if ($showtotals == 1) { + $pagetags->total = new \stdClass(); + $pagetags->total->total = "$respondents/$participants"; + $pagetags->total->evencolor = $evencolor; + } + } + } + + return $pagetags; + } + + /** + * Configure bulk sql + * + * @return bulk_sql_config + */ + protected function bulk_sql_config() { + return new bulk_sql_config(static::response_table(), 'qrt', false, false, false); + } +} + diff --git a/classes/responsetype/response/response.php b/classes/responsetype/response/response.php index 070160d2..0f214d0f 100644 --- a/classes/responsetype/response/response.php +++ b/classes/responsetype/response/response.php @@ -124,7 +124,7 @@ public static function response_from_webform($responsedata, $questions) { * @param id $responseid * @param \stdClass $responsedata All of the responsedata as an object. * @param array $questions Array of question objects. - * @return bool|response A response object. + * @return response A response object. */ public static function response_from_appdata($questionnaireid, $responseid, $responsedata, $questions) { global $USER; @@ -169,5 +169,6 @@ public function add_questions_answers() { $this->answers += \mod_questionnaire\responsetype\boolean::response_answers_by_question($this->id); $this->answers += \mod_questionnaire\responsetype\date::response_answers_by_question($this->id); $this->answers += \mod_questionnaire\responsetype\text::response_answers_by_question($this->id); + $this->answers += \mod_questionnaire\responsetype\file::response_answers_by_question($this->id); } } diff --git a/db/install.php b/db/install.php index 55b0eda4..33153962 100644 --- a/db/install.php +++ b/db/install.php @@ -114,4 +114,10 @@ function xmldb_questionnaire_install() { $questiontype->response_table = ''; $id = $DB->insert_record('questionnaire_question_type', $questiontype); + $questiontype = new stdClass(); + $questiontype->typeid = 12; + $questiontype->type = 'File'; + $questiontype->has_choices = 'n'; + $questiontype->response_table = 'response_file'; + $id = $DB->insert_record('questionnaire_question_type', $questiontype); } diff --git a/db/install.xml b/db/install.xml index a1b7af0b..4fdc7ff7 100644 --- a/db/install.xml +++ b/db/install.xml @@ -225,6 +225,21 @@ + + + + + + + + + + + + + + +
diff --git a/db/upgrade.php b/db/upgrade.php index dd5fd989..74cd7cf7 100644 --- a/db/upgrade.php +++ b/db/upgrade.php @@ -1002,6 +1002,37 @@ function xmldb_questionnaire_upgrade($oldversion=0) { upgrade_mod_savepoint(true, 2022121600.02, 'questionnaire'); } + if ($oldversion < 2022121600.03) { + $questiontype = new stdClass(); + $questiontype->typeid = 12; + $questiontype->type = 'File'; + $questiontype->has_choices = 'n'; + $questiontype->response_table = 'response_file'; + $id = $DB->insert_record('questionnaire_question_type', $questiontype); + + // Define table questionnaire_response_file to be created. + $table = new xmldb_table('questionnaire_response_file'); + + // Adding fields to table questionnaire_response_file. + $table->add_field('id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, XMLDB_SEQUENCE, null); + $table->add_field('response_id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0'); + $table->add_field('question_id', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, '0'); + $table->add_field('fileid', XMLDB_TYPE_INTEGER, '10', null, XMLDB_NOTNULL, null, null); + + // Adding keys to table questionnaire_response_file. + $table->add_key('primary', XMLDB_KEY_PRIMARY, ['id']); + $table->add_key('file_fk', XMLDB_KEY_FOREIGN, ['fileid'], 'files', ['id']); + + // Adding indexes to table questionnaire_response_file. + $table->add_index('response_question', XMLDB_INDEX_NOTUNIQUE, ['response_id', 'question_id']); + + // Conditionally launch create table for questionnaire_response_file. + if (!$dbman->table_exists($table)) { + $dbman->create_table($table); + } + upgrade_mod_savepoint(true, 2022121600.03, 'questionnaire'); + } + return true; } diff --git a/lang/en/questionnaire.php b/lang/en/questionnaire.php index 7de8d9f7..d62f3630 100644 --- a/lang/en/questionnaire.php +++ b/lang/en/questionnaire.php @@ -246,6 +246,8 @@ respondent. Default values are 20 characters for the Input Box width and 25 characters for the maximum length of text entered.'; +$string['file'] = 'File'; +$string['file_help'] = 'Allow user to submit a file (only simple, virus safe formats are allowed)'; $string['finished'] = 'You have answered all the questions in this questionnaire!'; $string['firstrespondent'] = 'First Respondent'; $string['formateditor'] = 'HTML editor'; diff --git a/lib.php b/lib.php index dd7f2ea0..2452997c 100644 --- a/lib.php +++ b/lib.php @@ -521,11 +521,11 @@ function questionnaire_scale_used_anywhere($scaleid) { * @param string $filearea * @param array $args * @param bool $forcedownload + * @param mixed $options * @return bool false if file not found, does not return if found - justsend the file * - * $forcedownload is unused, but API requires it. Suppress PHPMD warning. */ -function questionnaire_pluginfile($course, $cm, $context, $filearea, $args, $forcedownload) { +function questionnaire_pluginfile($course, $cm, $context, $filearea, $args, $forcedownload, $options) { global $DB; if ($context->contextlevel != CONTEXT_MODULE) { @@ -534,7 +534,7 @@ function questionnaire_pluginfile($course, $cm, $context, $filearea, $args, $for require_course_login($course, true, $cm); - $fileareas = ['intro', 'info', 'thankbody', 'question', 'feedbacknotes', 'sectionheading', 'feedback']; + $fileareas = ['intro', 'info', 'thankbody', 'question', 'feedbacknotes', 'sectionheading', 'feedback', 'file']; if (!in_array($filearea, $fileareas)) { return false; } @@ -553,6 +553,10 @@ function questionnaire_pluginfile($course, $cm, $context, $filearea, $args, $for if (!$DB->record_exists('questionnaire_feedback', ['id' => $componentid])) { return false; } + } else if ($filearea == 'file') { + if (!$DB->record_exists('questionnaire_response_file', ['id' => $componentid])) { + return false; + } } else { if (!$DB->record_exists('questionnaire_survey', ['id' => $componentid])) { return false; @@ -571,8 +575,9 @@ function questionnaire_pluginfile($course, $cm, $context, $filearea, $args, $for } // Finally send the file. - send_stored_file($file, 0, 0, true); // Download MUST be forced - security! + send_stored_file($file, null, 0, $forcedownload, $options); // Download MUST be forced - security! } + /** * Adds module specific settings to the settings block * diff --git a/locallib.php b/locallib.php index 4fce8590..df208c47 100644 --- a/locallib.php +++ b/locallib.php @@ -324,6 +324,7 @@ function questionnaire_delete_response($response, $questionnaire='') { $DB->delete_records('questionnaire_response_rank', array('response_id' => $rid)); $DB->delete_records('questionnaire_resp_single', array('response_id' => $rid)); $DB->delete_records('questionnaire_response_text', array('response_id' => $rid)); + $DB->delete_records('questionnaire_response_file', ['response_id' => $rid]); $status = $status && $DB->delete_records('questionnaire_response', array('id' => $rid)); @@ -354,6 +355,7 @@ function questionnaire_delete_responses($qid) { $DB->delete_records('questionnaire_response_rank', ['question_id' => $qid]); $DB->delete_records('questionnaire_resp_single', ['question_id' => $qid]); $DB->delete_records('questionnaire_response_text', ['question_id' => $qid]); + $DB->delete_records('questionnaire_response_file', ['question_id' => $qid]); return true; } @@ -498,6 +500,8 @@ function questionnaire_get_type ($id) { return get_string('numeric', 'questionnaire'); case 11: return get_string('slider', 'questionnaire'); + case 12: + return get_string('file', 'questionnaire'); case 100: return get_string('sectiontext', 'questionnaire'); case 99: diff --git a/tests/behat/add_questions.feature b/tests/behat/add_questions.feature index 54b4fdf0..4f020693 100644 --- a/tests/behat/add_questions.feature +++ b/tests/behat/add_questions.feature @@ -99,4 +99,8 @@ Feature: Add questions to a questionnaire activity And I should see "Choose yes or no" And I set the field "id_type_id" to "----- Page Break -----" And I press "Add selected question type" + And I add a "File" question and I fill the form with: + | Question Name | Q10 | + | Yes | Yes | + | Question Text | Add a file as an answer | Then I should see "[----- Page Break -----]" diff --git a/tests/behat/behat_mod_questionnaire.php b/tests/behat/behat_mod_questionnaire.php index 57c97c58..8b25cfd9 100644 --- a/tests/behat/behat_mod_questionnaire.php +++ b/tests/behat/behat_mod_questionnaire.php @@ -120,7 +120,9 @@ public function i_add_a_question_and_i_fill_the_form_with($questiontype, TableNo 'Rate (scale 1..5)', 'Text Box', 'Yes/No', - 'Slider'); + 'Slider', + 'File', + ); if (!in_array($questiontype, $validtypes)) { throw new ExpectationException('Invalid question type specified.', $this->getSession()); @@ -465,4 +467,189 @@ protected function get_cm_by_questionnaire_name(string $name): stdClass { $questionnaire = $this->get_questionnaire_by_name($name); return get_coursemodule_from_instance('questionnaire', $questionnaire->id, $questionnaire->course); } + + /** + * Uploads a file to the specified filemanager leaving other fields in upload form default. + * + * The paths should be relative to moodle codebase. + * + * @When /^I upload "(?P(?:[^"]|\\")*)" to questionnaire "(?P(?:[^"]|\\")*)" filemanager$/ + * @param string $filepath + * @param string $question + */ + public function i_upload_file_to_questionnaire_question_filemanager($filepath, $question) { + $this->upload_file_to_question_filemanager_questionnaire($filepath, $question, new TableNode([]), false); + } + + /** + * Try to get the filemanager node of a given question. + * + * @param $question + * @return \Behat\Mink\Element\NodeElement|null + */ + protected function get_filepicker_node($question) { + // More info about the problem (in case there is a problem). + $exception = new ExpectationException('The filepicker for the question with text "' . $question . + '" can not be found', $this->getSession()); + + $filepickercontainer = $this->find( + 'xpath', + "//p[contains(.,'" . $question . "')]" . + "//parent::div[contains(concat(' ', normalize-space(@class), ' '), ' no-overflow ')]" . + "//parent::div[contains(concat(' ', normalize-space(@class), ' '), ' qn-question ')]" . + "//following::div[contains(concat(' ', normalize-space(@class), ' '), ' qn-answer ')]" . + "//descendant::*[@data-fieldtype = 'filemanager' or @data-fieldtype = 'filepicker']", + $exception + ); + + return $filepickercontainer; + } + + /** + * Uploads a file to filemanager + * + * @param string $filepath Normally a path relative to $CFG->dirroot, but can be an absolute path too. + * @param string $question A question text. + * @param TableNode $data Data to fill in upload form + * @param false|string $overwriteaction false if we don't expect that file with the same name already exists, + * or button text in overwrite dialogue ("Overwrite", "Rename to ...", "Cancel") + * @throws DriverException + * @throws ExpectationException Thrown by behat_base::find + */ + protected function upload_file_to_question_filemanager_questionnaire($filepath, $question, TableNode $data, + $overwriteaction = false) { + global $CFG; + + if (!$this->has_tag('_file_upload')) { + throw new DriverException('File upload tests must have the @_file_upload tag on either the scenario or feature.'); + } + + $filemanagernode = $this->get_filepicker_node($question); + + // Opening the select repository window and selecting the upload repository. + $this->open_add_file_window($filemanagernode, get_string('pluginname', 'repository_upload')); + + // Ensure all the form is ready. + $noformexception = new ExpectationException('The upload file form is not ready', $this->getSession()); + $this->find( + 'xpath', + "//div[contains(concat(' ', normalize-space(@class), ' '), ' container ')]" . + "[contains(concat(' ', normalize-space(@class), ' '), ' repository_upload ')]" . + "/descendant::div[contains(concat(' ', normalize-space(@class), ' '), ' file-picker ')]" . + "/descendant::div[contains(concat(' ', normalize-space(@class), ' '), ' fp-content ')]" . + "/descendant::div[contains(concat(' ', normalize-space(@class), ' '), ' fp-upload-form ')]" . + "/descendant::form", + $noformexception + ); + // After this we have the elements we want to interact with. + + // Form elements to interact with. + $file = $this->find_file('repo_upload_file'); + + // Attaching specified file to the node. + // Replace 'admin/' if it is in start of path with $CFG->admin . + if (substr($filepath, 0, 6) === 'admin/') { + $filepath = $CFG->dirroot . DIRECTORY_SEPARATOR . $CFG->admin . + DIRECTORY_SEPARATOR . substr($filepath, 6); + } + $filepath = str_replace('/', DIRECTORY_SEPARATOR, $filepath); + if (!is_readable($filepath)) { + $filepath = $CFG->dirroot . DIRECTORY_SEPARATOR . $filepath; + if (!is_readable($filepath)) { + throw new ExpectationException('The file to be uploaded does not exist.', $this->getSession()); + } + } + $file->attachFile($filepath); + + // Fill the form in Upload window. + $datahash = $data->getRowsHash(); + + // The action depends on the field type. + foreach ($datahash as $locator => $value) { + + $field = behat_field_manager::get_form_field_from_label($locator, $this); + + // Delegates to the field class. + $field->set_value($value); + } + + // Submit the file. + $submit = $this->find_button(get_string('upload', 'repository')); + $submit->press(); + + // We wait for all the JS to finish as it is performing an action. + $this->getSession()->wait(self::get_timeout(), self::PAGE_READY_JS); + + if ($overwriteaction !== false) { + $overwritebutton = $this->find_button($overwriteaction); + $this->ensure_node_is_visible($overwritebutton); + $overwritebutton->click(); + + // We wait for all the JS to finish. + $this->getSession()->wait(self::get_timeout(), self::PAGE_READY_JS); + } + + } + + /** + * Opens the filepicker modal window and selects the repository. + * + * @param NodeElement $filemanagernode The filemanager or filepicker form element DOM node. + * @param mixed $repositoryname The repo name. + * @return void + * @throws ExpectationException Thrown by behat_base::find + */ + protected function open_add_file_window($filemanagernode, $repositoryname) { + $exception = new ExpectationException('No files can be added to the specified filemanager', $this->getSession()); + + // We should deal with single-file and multiple-file filemanagers, + // catching the exception thrown by behat_base::find() in case is not multiple. + $this->execute('behat_general::i_click_on_in_the', [ + 'div.fp-btn-add a, input.fp-btn-choose', 'css_element', + $filemanagernode, 'NodeElement' + ]); + + // Wait for the default repository (if any) to load. This checks that + // the relevant div exists and that it does not include the loading image. + $this->ensure_element_exists( + "//div[contains(concat(' ', normalize-space(@class), ' '), ' file-picker ')]" . + "//div[contains(concat(' ', normalize-space(@class), ' '), ' fp-content ')]" . + "[not(descendant::div[contains(concat(' ', normalize-space(@class), ' '), ' fp-content-loading ')])]", + 'xpath_element'); + + // Getting the repository link and opening it. + $repoexception = + new ExpectationException('The "' . $repositoryname . '" repository has not been found', $this->getSession()); + + // Avoid problems with both double and single quotes in the same string. + $repositoryname = behat_context_helper::escape($repositoryname); + + // Here we don't need to look inside the selected element because there can only be one modal window. + // Apparently there are some of these repo elements. So if the first one is not visible, check out + // the next one. + $repositorylinks = $this->find_all( + 'xpath', + "//div[contains(concat(' ', normalize-space(@class), ' '), ' fp-repo-area ')]" . + "//descendant::span[contains(concat(' ', normalize-space(@class), ' '), ' fp-repo-name ')]" . + "[normalize-space(.)=$repositoryname]", + $repoexception + ); + + foreach ($repositorylinks as $repositorylink) { + try { + $this->ensure_node_is_visible($repositorylink); + } catch (Exception $exception) { + $repositorylink = $exception; + } + } + if ($repositorylink instanceof \Exception) { + throw new $repositorylink; + } + // Selecting the repo. + if (!$repositorylink->getParent()->getParent()->hasClass('active')) { + // If the repository link is active, then the repository is already loaded. + // Clicking it while it's active causes issues, so only click it when it isn't (see MDL-51014). + $this->execute('behat_general::i_click_on', [$repositorylink, 'NodeElement']); + } + } } diff --git a/tests/behat/file_question.feature b/tests/behat/file_question.feature new file mode 100644 index 00000000..855d8350 --- /dev/null +++ b/tests/behat/file_question.feature @@ -0,0 +1,80 @@ +@mod @mod_questionnaire +Feature: Add a question requiring a file upload in questionnaire. + In order to use this plugin + As a teacher + I need to add a a file question to a questionnaire created in my course + and a student answers to it. Then the file has to be accessible. + + Background: + Given the following "users" exist: + | username | firstname | lastname | email | + | teacher1 | Teacher | 1 | teacher1@example.com | + | student1 | Student | 1 | student1@example.com | + And the following "courses" exist: + | fullname | shortname | category | + | Course 1 | C1 | 0 | + And the following "course enrolments" exist: + | user | course | role | + | teacher1 | C1 | editingteacher | + | student1 | C1 | student | + And the following "activities" exist: + | activity | name | description | course | idnumber | resume | navigate | + | questionnaire | Test questionnaire | Test questionnaire description | C1 | questionnaire0 | 1 | 1 | + + @javascript @_file_upload + Scenario: Add a single file question to a questionnaire and view an answer with an uploaded file. + Given I log in as "teacher1" + When I am on the "Test questionnaire" "questionnaire activity" page + And I navigate to "Questions" in current page administration + And I should see "Add questions" + And I add a "File" question and I fill the form with: + | Question Name | File question | + | Yes | Yes | + | Question Text | Add a file as an answer | + And I log out + And I log in as "student1" + And I am on the "Test questionnaire" "questionnaire activity" page + And I navigate to "Answer the questions..." in current page administration + And I upload "mod/questionnaire/tests/fixtures/testfilequestion.pdf" to questionnaire "Add a file as an answer" filemanager + And I press "Submit questionnaire" + And I should see "Thank you for completing this Questionnaire" + And I press "Continue" + And I should see "View your response(s)" + And ".resourcecontent.resourcepdf" "css_element" should exist + And I log out + And I log in as "teacher1" + And I am on the "Test questionnaire" "questionnaire activity" page + And I navigate to "View all responses" in current page administration + Then I should see "testfilequestion.pdf" + + @javascript @_file_upload + Scenario: Add two file questions to a questionnaire and view an answer with two uploaded file. + Given I log in as "teacher1" + When I am on the "Test questionnaire" "questionnaire activity" page + And I navigate to "Questions" in current page administration + And I should see "Add questions" + And I add a "File" question and I fill the form with: + | Question Name | File question one | + | Yes | Yes | + | Question Text | Add a first file as an answer | + And I add a "File" question and I fill the form with: + | Question Name | File question two | + | Yes | Yes | + | Question Text | Add a second file as an answer | + And I log out + And I log in as "student1" + And I am on the "Test questionnaire" "questionnaire activity" page + And I navigate to "Answer the questions..." in current page administration + And I upload "mod/questionnaire/tests/fixtures/testfilequestion.pdf" to questionnaire "Add a first file as an answer" filemanager + And I upload "mod/questionnaire/tests/fixtures/testfilequestion2.pdf" to questionnaire "Add a second file as an answer" filemanager + And I press "Submit questionnaire" + And I should see "Thank you for completing this Questionnaire" + And I press "Continue" + And I should see "View your response(s)" + And ".resourcecontent.resourcepdf" "css_element" should exist + And I log out + And I log in as "teacher1" + And I am on the "Test questionnaire" "questionnaire activity" page + And I navigate to "View all responses" in current page administration + Then I should see "testfilequestion.pdf" + And I should see "testfilequestion2.pdf" diff --git a/tests/fixtures/testfilequestion.pdf b/tests/fixtures/testfilequestion.pdf new file mode 100644 index 0000000000000000000000000000000000000000..90588c3015a7864cb0ee78639de3b564b569a1a6 GIT binary patch literal 16338 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 = 2022121600.02; // The current module version (Date: YYYYMMDDXX). +$plugin->version = 2022121600.03; // The current module version (Date: YYYYMMDDXX). $plugin->requires = 2022112800.00; // Moodle version (4.1.0). $plugin->component = 'mod_questionnaire';