123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494 |
- <?php
- /* For licensing terms, see /license.txt */
- /**
- * @author Claro Team <cvs@claroline.net>
- * @author Yannick Warnier <yannick.warnier@beeznest.com>
- * @package chamilo.exercise
- */
- require __DIR__.'/qti2_classes.php';
- /**
- * An IMS/QTI item. It corresponds to a single question.
- * This class allows export from Claroline to IMS/QTI2.0 XML format of a single question.
- * It is not usable as-is, but must be subclassed, to support different kinds of questions.
- *
- * Every start_*() and corresponding end_*(), as well as export_*() methods return a string.
- *
- * note: Attached files are NOT exported.
- * @package chamilo.exercise
- */
- class ImsAssessmentItem
- {
- public $question;
- public $question_ident;
- public $answer;
- /**
- * Constructor.
- *
- * @param Ims2Question $question Ims2Question object we want to export.
- */
- public function __construct($question)
- {
- $this->question = $question;
- $this->answer = $this->question->setAnswer();
- $this->questionIdent = "QST_".$question->id;
- }
- /**
- * Start the XML flow.
- *
- * This opens the <item> block, with correct attributes.
- *
- */
- function start_item()
- {
- $categoryTitle = '';
- if (!empty($this->question->category)) {
- $category = new TestCategory();
- $category = $category->getCategory($this->question->category);
- if ($category) {
- $categoryTitle = htmlspecialchars(formatExerciseQtiTitle($category->name));
- }
- }
- $string = '<assessmentItem xmlns="http://www.imsglobal.org/xsd/imsqti_v2p1"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://www.imsglobal.org/xsd/imsqti_v2p1 imsqti_v2p1.xsd"
- identifier="'.$this->questionIdent.'"
- title = "'.htmlspecialchars(formatExerciseQtiTitle($this->question->selectTitle())).'"
- category = "'.$categoryTitle.'"
- >'."\n";
- return $string;
- }
- /**
- * End the XML flow, closing the </item> tag.
- *
- */
- function end_item()
- {
- return "</assessmentItem>\n";
- }
- /**
- * Start the itemBody
- *
- */
- function start_item_body()
- {
- return ' <itemBody>'."\n";
- }
- /**
- * End the itemBody part.
- *
- */
- function end_item_body()
- {
- return " </itemBody>\n";
- }
- /**
- * add the response processing template used.
- *
- */
- function add_response_processing()
- {
- return ' <responseProcessing template="http://www.imsglobal.org/question/qti_v2p1/rptemplates/map_correct"/>'."\n";
- }
- /**
- * Export the question as an IMS/QTI Item.
- *
- * This is a default behaviour, some classes may want to override this.
- *
- * @param $standalone: Boolean stating if it should be exported as a stand-alone question
- * @return string string, the XML flow for an Item.
- */
- function export($standalone = false)
- {
- $head = $foot = '';
- if ($standalone) {
- $head = '<?xml version="1.0" encoding="UTF-8" standalone="no"?>'."\n";
- }
- //TODO understand why answer might be a non-object sometimes
- if (!is_object($this->answer)) {
- return $head;
- }
- $res = $head
- .$this->start_item()
- .$this->answer->imsExportResponsesDeclaration($this->questionIdent)
- .$this->start_item_body()
- .$this->answer->imsExportResponses(
- $this->questionIdent,
- $this->question->question,
- $this->question->description,
- $this->question->picture
- )
- .$this->end_item_body()
- .$this->add_response_processing()
- .$this->end_item()
- .$foot;
- return $res;
- }
- }
- /**
- * This class represents an entire exercise to be exported in IMS/QTI.
- * It will be represented by a single <section> containing several <item>.
- *
- * Some properties cannot be exported, as IMS does not support them :
- * - type (one page or multiple pages)
- * - start_date and end_date
- * - max_attempts
- * - show_answer
- * - anonymous_attempts
- *
- * @author Amand Tihon <amand@alrj.org>
- * @package chamilo.exercise
- */
- class ImsSection
- {
- public $exercise;
- /**
- * Constructor.
- * @param Exercise $exe The Exercise instance to export
- * @author Amand Tihon <amand@alrj.org>
- */
- public function __construct($exe)
- {
- $this->exercise = $exe;
- }
- function start_section()
- {
- $out = '<section
- ident = "EXO_' . $this->exercise->selectId().'"
- title = "' .cleanAttribute(formatExerciseQtiDescription($this->exercise->selectTitle())).'"
- >' . "\n";
- return $out;
- }
- function end_section()
- {
- return "</section>\n";
- }
- function export_duration()
- {
- if ($max_time = $this->exercise->selectTimeLimit()) {
- // return exercise duration in ISO8601 format.
- $minutes = floor($max_time / 60);
- $seconds = $max_time % 60;
- return '<duration>PT'.$minutes.'M'.$seconds."S</duration>\n";
- } else {
- return '';
- }
- }
- /**
- * Export the presentation (Exercise's description)
- * @author Amand Tihon <amand@alrj.org>
- */
- function export_presentation()
- {
- $out = "<presentation_material><flow_mat><material>\n"
- . " <mattext><![CDATA[".formatExerciseQtiDescription($this->exercise->selectDescription())."]]></mattext>\n"
- . "</material></flow_mat></presentation_material>\n";
- return $out;
- }
- /**
- * Export the ordering information.
- * Either sequential, through all questions, or random, with a selected number of questions.
- * @author Amand Tihon <amand@alrj.org>
- */
- function export_ordering()
- {
- $out = '';
- if ($n = $this->exercise->getShuffle()) {
- $out .= "<selection_ordering>"
- . " <selection>\n"
- . " <selection_number>".$n."</selection_number>\n"
- . " </selection>\n"
- . ' <order order_type="Random" />'
- . "\n</selection_ordering>\n";
- } else {
- $out .= '<selection_ordering sequence_type="Normal">'."\n"
- . " <selection />\n"
- . "</selection_ordering>\n";
- }
- return $out;
- }
- /**
- * Export the questions, as a succession of <items>
- * @author Amand Tihon <amand@alrj.org>
- */
- function export_questions()
- {
- $out = '';
- foreach ($this->exercise->selectQuestionList() as $q) {
- $out .= export_question_qti($q, false);
- }
- return $out;
- }
- /**
- * Export the exercise in IMS/QTI.
- *
- * @param bool $standalone Wether it should include XML tag and DTD line.
- * @return string string containing the XML flow
- * @author Amand Tihon <amand@alrj.org>
- */
- function export($standalone)
- {
- $head = $foot = '';
- if ($standalone) {
- $head = '<?xml version = "1.0" encoding = "UTF-8" standalone = "no"?>'."\n"
- . '<!DOCTYPE questestinterop SYSTEM "ims_qtiasiv2p1.dtd">'."\n"
- . "<questestinterop>\n";
- $foot = "</questestinterop>\n";
- }
- $out = $head
- . $this->start_section()
- . $this->export_duration()
- . $this->export_presentation()
- . $this->export_ordering()
- . $this->export_questions()
- . $this->end_section()
- . $foot;
- return $out;
- }
- }
- /*
- Some quick notes on identifiers generation.
- The IMS format requires some blocks, like items, responses, feedbacks, to be uniquely
- identified.
- The unicity is mandatory in a single XML, of course, but it's prefered that the identifier stays
- coherent for an entire site.
- Here's the method used to generate those identifiers.
- Question identifier :: "QST_" + <Question Id from the DB> + "_" + <Question numeric type>
- Response identifier :: <Question identifier> + "_A_" + <Response Id from the DB>
- Condition identifier :: <Question identifier> + "_C_" + <Response Id from the DB>
- Feedback identifier :: <Question identifier> + "_F_" + <Response Id from the DB>
- */
- /**
- * Class ImsItem
- *
- * An IMS/QTI item. It corresponds to a single question.
- * This class allows export from Claroline to IMS/QTI XML format.
- * It is not usable as-is, but must be subclassed, to support different kinds of questions.
- *
- * Every start_*() and corresponding end_*(), as well as export_*() methods return a string.
- *
- * warning: Attached files are NOT exported.
- * @author Amand Tihon <amand@alrj.org>
- *
- * @package chamilo.exercise
- */
- class ImsItem
- {
- public $question;
- public $question_ident;
- public $answer;
- /**
- * Constructor.
- *
- * @param Question $question The Question object we want to export.
- * @author Anamd Tihon
- */
- public function __construct($question)
- {
- $this->question = $question;
- $this->answer = $question->answer;
- $this->questionIdent = "QST_".$question->selectId();
- }
- /**
- * Start the XML flow.
- *
- * This opens the <item> block, with correct attributes.
- *
- * @author Amand Tihon <amand@alrj.org>
- */
- function start_item()
- {
- return '<item title="'.cleanAttribute(formatExerciseQtiDescription($this->question->selectTitle())).'" ident="'.$this->questionIdent.'">'."\n";
- }
- /**
- * End the XML flow, closing the </item> tag.
- *
- * @author Amand Tihon <amand@alrj.org>
- */
- function end_item()
- {
- return "</item>\n";
- }
- /**
- * Create the opening, with the question itself.
- *
- * This means it opens the <presentation> but doesn't close it, as this is the role of end_presentation().
- * In between, the export_responses from the subclass should have been called.
- *
- * @author Amand Tihon <amand@alrj.org>
- */
- function start_presentation()
- {
- return '<presentation label="'.$this->questionIdent.'"><flow>'."\n"
- . '<material><mattext>'.formatExerciseQtiDescription($this->question->selectDescription())."</mattext></material>\n";
- }
- /**
- * End the </presentation> part, opened by export_header.
- *
- * @author Amand Tihon <amand@alrj.org>
- */
- function end_presentation()
- {
- return "</flow></presentation>\n";
- }
- /**
- * Start the response processing, and declare the default variable, SCORE, at 0 in the outcomes.
- *
- * @author Amand Tihon <amand@alrj.org>
- */
- function start_processing()
- {
- return '<resprocessing><outcomes><decvar vartype="Integer" defaultval="0" /></outcomes>'."\n";
- }
- /**
- * End the response processing part.
- *
- * @author Amand Tihon <amand@alrj.org>
- */
- function end_processing()
- {
- return "</resprocessing>\n";
- }
- /**
- * Export the question as an IMS/QTI Item.
- *
- * This is a default behaviour, some classes may want to override this.
- *
- * @param $standalone: Boolean stating if it should be exported as a stand-alone question
- * @return string string, the XML flow for an Item.
- * @author Amand Tihon <amand@alrj.org>
- */
- function export($standalone = False)
- {
- global $charset;
- $head = $foot = "";
- if ($standalone) {
- $head = '<?xml version = "1.0" encoding = "'.$charset.'" standalone = "no"?>'."\n"
- . '<!DOCTYPE questestinterop SYSTEM "ims_qtiasiv2p1.dtd">'."\n"
- . "<questestinterop>\n";
- $foot = "</questestinterop>\n";
- }
- return $head
- . $this->start_item()
- . $this->start_presentation()
- . $this->answer->imsExportResponses($this->questionIdent)
- . $this->end_presentation()
- . $this->start_processing()
- . $this->answer->imsExportProcessing($this->questionIdent)
- . $this->end_processing()
- . $this->answer->imsExportFeedback($this->questionIdent)
- . $this->end_item()
- . $foot;
- }
- }
- /**
- * Send a complete exercise in IMS/QTI format, from its ID
- *
- * @param int $exerciseId The exercise to export
- * @param boolean $standalone Wether it should include XML tag and DTD line.
- * @return string XML as a string, or an empty string if there's no exercise with given ID.
- */
- function export_exercise_to_qti($exerciseId, $standalone = true)
- {
- $exercise = new Exercise();
- if (!$exercise->read($exerciseId)) {
- return '';
- }
- $ims = new ImsSection($exercise);
- $xml = $ims->export($standalone);
- return $xml;
- }
- /**
- * Returns the XML flow corresponding to one question
- *
- * @param int $questionId
- * @param bool $standalone (ie including XML tag, DTD declaration, etc)
- * @return string
- */
- function export_question_qti($questionId, $standalone = true)
- {
- $question = new Ims2Question();
- $qst = $question->read($questionId);
- if (!$qst || $qst->type == FREE_ANSWER) {
- return '';
- }
- $question->id = $qst->id;
- $question->type = $qst->type;
- $question->question = $qst->question;
- $question->description = $qst->description;
- $question->weighting = $qst->weighting;
- $question->position = $qst->position;
- $question->picture = $qst->picture;
- $question->category = $qst->category;
- $ims = new ImsAssessmentItem($question);
- return $ims->export($standalone);
- }
- /**
- * Clean text like a description
- **/
- function formatExerciseQtiDescription($text)
- {
- $entities = api_html_entity_decode($text);
- return htmlspecialchars($entities);
- }
- /**
- * Clean titles
- * @param $text
- * @return string
- */
- function formatExerciseQtiTitle($text)
- {
- return htmlspecialchars($text);
- }
- /**
- * @param string $text
- */
- function cleanAttribute($text)
- {
- return $text;
- }
|