Proyectos de Subversion Moodle

Rev

Autoría | Ultima modificación | Ver Log |

<?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/>.

/**
 * Upgrade library code for the randomsamatch question type.
 *
 * @package   qtype_randomsamatch
 * @copyright 2013 Jean-Michel Vedrine
 * @license   http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */


defined('MOODLE_INTERNAL') || die();


/**
 * Class for converting attempt data for randomsamatch questions when upgrading
 * attempts to the new question engine.
 *
 * This class is used by the code in question/engine/upgrade/upgradelib.php.
 *
 * @copyright 2013 Jean-Michel Vedrine
 * @license   http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */
class qtype_randomsamatch_qe2_attempt_updater extends question_qtype_attempt_updater {
    /** @var array of question stems. */
    protected $stems;
    /** @var array of question stems format. */
    protected $stemformat;
    /** @var array of choices that can be matched to each stem. */
    protected $choices;
    /** @var array index of the right choice for each stem. */
    protected $right;
    /** @var array id of the right answer for each stem (used by {@link lookup_choice}). */
    protected $rightanswerid;
    /** @var array shuffled stem indexes. */
    protected $stemorder;
    /** @var array shuffled choice indexes. */
    protected $choiceorder;
    /** @var array flipped version of the choiceorder array. */
    protected $flippedchoiceorder;
    /** @var array of right answer for each stem. */
    protected $rightanswer;

    public function question_summary() {
        return ''; // Done later, after we know which shortanswer questions are used.
    }

    public function right_answer() {
        return ''; // Done later, after we know which shortanswer questions are used.
    }

    /**
     * Explode the answer saved as a string in state
     *
     * @param string $answer comma separated list of dash separated pairs
     * @return array
     */
    protected function explode_answer($answer) {
        if (!$answer) {
            return array();
        }
        $bits = explode(',', $answer);
        $selections = array();
        foreach ($bits as $bit) {
            list($stem, $choice) = explode('-', $bit);
            $selections[$stem] = $choice;
        }
        return $selections;
    }

    protected function make_summary($pairs) {
        $bits = array();
        foreach ($pairs as $stem => $answer) {
            $bits[] = $stem . ' -> ' . $answer;
        }
        return implode('; ', $bits);
    }

    /**
     * Find the index corresponding to a choice
     *
     * @param integer $choice
     * @return integer
     */
    protected function lookup_choice($choice) {
        if (array_key_exists($choice, $this->choices)) {
            // Easy case: choice is a key in the choices array.
            return $choice;
        } else {
            // But choice can also be the id of a shortanser correct answer
            // without been a key of the choices array, in that case we need
            // to first find the shortanswer id, then find the choices index
            // associated to it.
            $questionid = array_search($choice, $this->rightanswerid);
            if ($questionid) {
                return $this->right[$questionid];
            }
        }
        return null;
    }

    public function response_summary($state) {
        $choices = $this->explode_answer($state->answer);
        if (empty($choices)) {
            return null;
        }

        $pairs = array();
        foreach ($choices as $stemid => $choicekey) {
            if (array_key_exists($stemid, $this->stems) && $choices[$stemid]) {
                $choiceid = $this->lookup_choice($choicekey);
                if ($choiceid) {
                    $pairs[$this->stems[$stemid]] = $this->choices[$choiceid];
                } else {
                    $this->logger->log_assumption("Dealing with a place where the
                            student selected a choice that was later deleted for
                            randomsamatch question {$this->question->id}");
                    $pairs[$this->stems[$stemid]] = '[CHOICE THAT WAS LATER DELETED]';
                }
            }
        }

        if ($pairs) {
            return $this->make_summary($pairs);
        } else {
            return '';
        }
    }

    public function was_answered($state) {
        $choices = $this->explode_answer($state->answer);
        foreach ($choices as $choice) {
            if ($choice) {
                return true;
            }
        }
        return false;
    }

    public function set_first_step_data_elements($state, &$data) {
        $this->stems = array();
        $this->stemformat = array();
        $this->choices = array();
        $this->right = array();
        $this->rightanswer = array();
        $choices = $this->explode_answer($state->answer);
        $this->stemorder = array();
        foreach ($choices as $key => $notused) {
            $this->stemorder[] = $key;
        }
        $wrappedquestions = array();
        // TODO test what happen when some questions are missing.
        foreach ($this->stemorder as $questionid) {
            $wrappedquestions[] = $this->load_question($questionid);
        }
        foreach ($wrappedquestions as $wrappedquestion) {

            // We only take into account the first correct answer.
            $foundcorrect = false;
            foreach ($wrappedquestion->options->answers as $answer) {
                if ($foundcorrect || $answer->fraction != 1.0) {
                    unset($wrappedquestion->options->answers[$answer->id]);
                } else if (!$foundcorrect) {
                    $foundcorrect = true;
                    // Store right answer id, so we can use it later in lookup_choice.
                    $this->rightanswerid[$wrappedquestion->id] = $answer->id;
                    $key = array_search($answer->answer, $this->choices);
                    if ($key === false) {
                        $key = $answer->id;
                        $this->choices[$key] = $answer->answer;
                        $data['_choice_' . $key] = $answer->answer;
                    }
                    $this->stems[$wrappedquestion->id] = $wrappedquestion->questiontext;
                    $this->stemformat[$wrappedquestion->id] = $wrappedquestion->questiontextformat;
                    $this->right[$wrappedquestion->id] = $key;
                    $this->rightanswer[$wrappedquestion->id] = $answer->answer;

                    $data['_stem_' . $wrappedquestion->id] = $wrappedquestion->questiontext;
                    $data['_stemformat_' . $wrappedquestion->id] = $wrappedquestion->questiontextformat;
                    $data['_right_' . $wrappedquestion->id] = $key;

                }
            }
        }
        $this->choiceorder = array_keys($this->choices);
        // We don't shuffle the choices as that seems unnecessary for old upgraded attempts.
        $this->flippedchoiceorder = array_combine(
                array_values($this->choiceorder), array_keys($this->choiceorder));

        $data['_stemorder'] = implode(',', $this->stemorder);
        $data['_choiceorder'] = implode(',', $this->choiceorder);

        $this->updater->qa->questionsummary = $this->to_text($this->question->questiontext) . ' {' .
                implode('; ', $this->stems) . '} -> {' . implode('; ', $this->choices) . '}';

        $answer = array();
        foreach ($this->stems as $key => $stem) {
            $answer[$stem] = $this->choices[$this->right[$key]];
        }
        $this->updater->qa->rightanswer = $this->make_summary($answer);
    }

    public function supply_missing_first_step_data(&$data) {
        throw new coding_exception('qtype_randomsamatch_updater::supply_missing_first_step_data ' .
                'not tested');
        $data['_stemorder'] = array();
        $data['_choiceorder'] = array();
    }

    public function set_data_elements_for_step($state, &$data) {
        $choices = $this->explode_answer($state->answer);

        foreach ($this->stemorder as $i => $key) {
            if (empty($choices[$key])) {
                $data['sub' . $i] = 0;
                continue;
            }
            $choice = $this->lookup_choice($choices[$key]);

            if (array_key_exists($choice, $this->flippedchoiceorder)) {
                $data['sub' . $i] = $this->flippedchoiceorder[$choice] + 1;
            } else {
                $data['sub' . $i] = 0;
            }
        }
    }

    public function load_question($questionid) {
        return $this->qeupdater->load_question($questionid);
    }
}