3 // This file is part of Moodle - http://moodle.org/
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.
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.
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/>.
19 * Defines backup_qtype_plugin class
21 * @package core_backup
24 * @copyright 2010 onwards Eloy Lafuente (stronk7) {@link http://stronk7.com}
25 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
28 defined('MOODLE_INTERNAL') ||
die();
31 * Class extending standard backup_plugin in order to implement some
32 * helper methods related with the questions (qtype plugin)
34 * TODO: Finish phpdocs
36 abstract class backup_qtype_plugin
extends backup_plugin
{
39 * Attach to $element (usually questions) the needed backup structures
40 * for question_answers for a given question
41 * Used by various qtypes (calculated, essay, multianswer,
42 * multichoice, numerical, shortanswer, truefalse)
44 protected function add_question_question_answers($element) {
45 // Check $element is one nested_backup_element
46 if (! $element instanceof backup_nested_element
) {
47 throw new backup_step_exception('question_answers_bad_parent_element', $element);
50 // Define the elements
51 $answers = new backup_nested_element('answers');
52 $answer = new backup_nested_element('answer', array('id'), array(
53 'answertext', 'answerformat', 'fraction', 'feedback',
57 $element->add_child($answers);
58 $answers->add_child($answer);
61 $answer->set_source_table('question_answers', array('question' => backup
::VAR_PARENTID
), 'id ASC');
64 $answer->set_source_alias('answer', 'answertext');
66 // don't need to annotate ids nor files
70 * Attach to $element (usually questions) the needed backup structures
71 * for question_numerical_units for a given question
72 * Used both by calculated and numerical qtypes
74 protected function add_question_numerical_units($element) {
75 // Check $element is one nested_backup_element
76 if (! $element instanceof backup_nested_element
) {
77 throw new backup_step_exception('question_numerical_units_bad_parent_element', $element);
80 // Define the elements
81 $units = new backup_nested_element('numerical_units');
82 $unit = new backup_nested_element('numerical_unit', array('id'), array(
83 'multiplier', 'unit'));
86 $element->add_child($units);
87 $units->add_child($unit);
90 $unit->set_source_table('question_numerical_units', array('question' => backup
::VAR_PARENTID
), 'id ASC');
92 // don't need to annotate ids nor files
96 * Attach to $element (usually questions) the needed backup structures
97 * for question_numerical_options for a given question
98 * Used both by calculated and numerical qtypes
100 protected function add_question_numerical_options($element) {
101 // Check $element is one nested_backup_element
102 if (! $element instanceof backup_nested_element
) {
103 throw new backup_step_exception('question_numerical_options_bad_parent_element', $element);
106 // Define the elements
107 $options = new backup_nested_element('numerical_options');
108 $option = new backup_nested_element('numerical_option', array('id'), array(
109 'showunits', 'unitsleft', 'unitgradingtype', 'unitpenalty'));
112 $element->add_child($options);
113 $options->add_child($option);
116 $option->set_source_table('question_numerical_options', array('question' => backup
::VAR_PARENTID
));
118 // don't need to annotate ids nor files
122 * Attach to $element (usually questions) the needed backup structures
123 * for question_datasets for a given question
124 * Used by calculated qtypes
126 protected function add_question_datasets($element) {
127 // Check $element is one nested_backup_element
128 if (! $element instanceof backup_nested_element
) {
129 throw new backup_step_exception('question_datasets_bad_parent_element', $element);
132 // Define the elements
133 $definitions = new backup_nested_element('dataset_definitions');
134 $definition = new backup_nested_element('dataset_definition', array('id'), array(
135 'category', 'name', 'type', 'options',
138 $items = new backup_nested_element('dataset_items');
139 $item = new backup_nested_element('dataset_item', array('id'), array(
143 $element->add_child($definitions);
144 $definitions->add_child($definition);
146 $definition->add_child($items);
147 $items->add_child($item);
150 $definition->set_source_sql('SELECT qdd.*
151 FROM {question_dataset_definitions} qdd
152 JOIN {question_datasets} qd ON qd.datasetdefinition = qdd.id
153 WHERE qd.question = ?', array(backup
::VAR_PARENTID
));
155 $item->set_source_table('question_dataset_items', array('definition' => backup
::VAR_PARENTID
));
158 $item->set_source_alias('itemnumber', 'number');
160 // don't need to annotate ids nor files
164 * Returns all the components and fileareas used by all the installed qtypes
166 * The method introspects each qtype, asking it about fileareas used. Then,
167 * one 2-level array is returned. 1st level is the component name (qtype_xxxx)
168 * and 2nd level is one array of filearea => mappings to look
170 * Note that this function is used both in backup and restore, so it is important
171 * to use the same mapping names (usually, name of the table in singular) always
173 * TODO: Surely this can be promoted to backup_plugin easily and make it to
174 * work for ANY plugin, not only qtypes (but we don't need it for now)
176 public static function get_components_and_fileareas($filter = null) {
177 $components = array();
178 // Get all the plugins of this type
179 $qtypes = core_component
::get_plugin_list('qtype');
180 foreach ($qtypes as $name => $path) {
181 // Apply filter if specified
182 if (!is_null($filter) && $filter != $name) {
185 // Calculate the componentname
186 $componentname = 'qtype_' . $name;
187 // Get the plugin fileareas (all them MUST belong to the same component)
188 $classname = 'backup_qtype_' . $name . '_plugin';
189 if (class_exists($classname)) {
190 $elements = call_user_func(array($classname, 'get_qtype_fileareas'));
192 // If there are elements, add them to $components
193 $components[$componentname] = $elements;
201 * Returns one array with filearea => mappingname elements for the qtype
203 * Used by {@link get_components_and_fileareas} to know about all the qtype
204 * files to be processed both in backup and restore.
206 public static function get_qtype_fileareas() {
207 // By default, return empty array, only qtypes having own fileareas will override this