class MultichoiceResponse in Quiz 7
Same name and namespace in other branches
- 6.4 question_types/multichoice/multichoice.classes.inc \MultichoiceResponse
- 7.6 question_types/multichoice/multichoice.classes.inc \MultichoiceResponse
- 7.4 question_types/multichoice/multichoice.classes.inc \MultichoiceResponse
- 7.5 question_types/multichoice/multichoice.classes.inc \MultichoiceResponse
Extension of QuizQuestionResponse
Hierarchy
- class \QuizQuestionResponse
- class \MultichoiceResponse
Expanded class hierarchy of MultichoiceResponse
1 string reference to 'MultichoiceResponse'
- multichoice_quiz_question_info in question_types/
multichoice/ multichoice.module - Implements hook_quiz_question_info().
File
- question_types/
multichoice/ multichoice.classes.inc, line 758 - The main classes for the multichoice question type.
View source
class MultichoiceResponse extends QuizQuestionResponse {
/**
* ID of the answers.
*/
protected $user_answer_ids;
protected $choice_order;
/**
* Constructor
*/
public function __construct($result_id, stdClass $question_node, $tries = NULL) {
parent::__construct($result_id, $question_node, $tries);
$this->user_answer_ids = array();
// tries is the tries part of the post data
if (is_array($tries)) {
if (isset($tries['choice_order'])) {
$this->choice_order = $tries['choice_order'];
}
unset($tries['choice_order']);
if (is_array($tries['answer'])) {
foreach ($tries['answer'] as $answer_id) {
$this->user_answer_ids[] = $answer_id;
$this->answer = $this->user_answer_ids;
//@todo: Stop using user_answer_ids and only use answer instead...
}
}
elseif (isset($tries['answer'])) {
$this->user_answer_ids[] = $tries['answer'];
}
}
else {
$res = db_query('SELECT answer_id FROM {quiz_multichoice_user_answers} ua
LEFT OUTER JOIN {quiz_multichoice_user_answer_multi} uam ON(uam.user_answer_id = ua.id)
WHERE ua.result_id = :result_id AND ua.question_nid = :question_nid AND ua.question_vid = :question_vid', array(
':result_id' => $result_id,
':question_nid' => $this->question->nid,
':question_vid' => $this->question->vid,
));
while ($res_o = $res
->fetch()) {
$this->user_answer_ids[] = $res_o->answer_id;
}
}
}
/**
* Implementation of isValid
*
* @see QuizQuestionResponse#isValid()
*/
public function isValid() {
if ($this->question->choice_multi) {
return TRUE;
}
if (empty($this->user_answer_ids)) {
return t('You must provide an answer');
}
// Perform extra check since FAPI isn't beeing used:
if (count($this->user_answer_ids) > 1) {
return t('You are only allowed to select one answer');
}
return TRUE;
}
/**
* Implementation of save
*
* @see QuizQuestionResponse#save()
*/
public function save() {
$user_answer_id = db_insert('quiz_multichoice_user_answers')
->fields(array(
'result_id' => $this->rid,
'question_vid' => $this->question->vid,
'question_nid' => $this->question->nid,
'choice_order' => $this->choice_order,
))
->execute();
$query = db_insert('quiz_multichoice_user_answer_multi')
->fields(array(
'user_answer_id',
'answer_id',
));
for ($i = 0; $i < count($this->user_answer_ids); $i++) {
$query
->values(array(
$user_answer_id,
$this->user_answer_ids[$i],
));
}
$query
->execute();
}
/**
* Implementation of delete
*
* @see QuizQuestionResponse#delete()
*/
public function delete() {
$user_answer_id = array();
$query = db_query('SELECT id FROM {quiz_multichoice_user_answers} WHERE question_nid = :nid AND question_vid = :vid AND result_id = :result_id', array(
':nid' => $this->question->nid,
':vid' => $this->question->vid,
':result_id' => $this->rid,
));
while ($user_answer = $query
->fetch()) {
$user_answer_id[] = $user_answer->id;
}
if (!empty($user_answer_id)) {
db_delete('quiz_multichoice_user_answer_multi')
->condition('user_answer_id', $user_answer_id, 'IN')
->execute();
}
db_delete('quiz_multichoice_user_answers')
->condition('result_id', $this->rid)
->condition('question_nid', $this->question->nid)
->condition('question_vid', $this->question->vid)
->execute();
}
/**
* Implementation of score
*
* @return uint
*
* @see QuizQuestionResponse#score()
*/
public function score() {
if ($this->question->choice_boolean || $this
->isAllWrong()) {
$score = 1;
foreach ($this->question->alternatives as $key => $alt) {
if (in_array($alt['id'], $this->user_answer_ids)) {
if ($alt['score_if_chosen'] <= $alt['score_if_not_chosen']) {
$score = 0;
}
}
else {
if ($alt['score_if_chosen'] > $alt['score_if_not_chosen']) {
$score = 0;
}
}
}
}
else {
$score = 0;
foreach ($this->question->alternatives as $key => $alt) {
if (in_array($alt['id'], $this->user_answer_ids)) {
$score += $alt['score_if_chosen'];
}
else {
$score += $alt['score_if_not_chosen'];
}
}
}
return $score;
}
/**
* If all answers in a question is wrong
*
* @return boolean
* TRUE if all answers are wrong. False otherwise.
*/
public function isAllWrong() {
foreach ($this->question->alternatives as $key => $alt) {
if ($alt['score_if_chosen'] > 0 || $alt['score_if_not_chosen'] > 0) {
return FALSE;
}
}
return TRUE;
}
/**
* Implementation of getResponse
*
* @return answer
*
* @see QuizQuestionResponse#getResponse()
*/
public function getResponse() {
return $this->user_answer_ids;
}
/**
* Implementation of getReportFormResponse
*
* @see getReportFormResponse($showpoints, $showfeedback, $allow_scoring)
*/
public function getReportFormResponse($showpoints = TRUE, $showfeedback = TRUE, $allow_scoring = FALSE) {
$i = 0;
$this
->orderAlternatives($this->question->alternatives);
// Find the alternative with the highest score
if ($this->question->choice_multi == 0) {
$max_score_if_chosen = -999;
while (isset($this->question->alternatives[$i]) && is_array($this->question->alternatives[$i])) {
$short = $this->question->alternatives[$i];
if ($short['score_if_chosen'] > $max_score_if_chosen) {
$max_score_if_chosen = $short['score_if_chosen'];
}
$i++;
}
$i = 0;
}
// Fetch all data for the report
$data = array();
while (isset($this->question->alternatives[$i])) {
$short = $this->question->alternatives[$i];
if (drupal_strlen($this
->checkMarkup($short['answer'], $short['answer_format'])) > 0) {
$alternative = array();
// Did the user choose the alternative?
$alternative['is_chosen'] = in_array($short['id'], $this->user_answer_ids);
// Questions where multiple answers isn't allowed are scored differently...
if ($this->question->choice_multi == 0) {
if ($this->question->choice_boolean == 0) {
if ($short['score_if_chosen'] > $short['score_if_not_chosen']) {
$alternative['is_correct'] = $short['score_if_chosen'] < $max_score_if_chosen ? 1 : 2;
}
else {
$alternative['is_correct'] = 0;
}
}
else {
$alternative['is_correct'] = $short['score_if_chosen'] > $short['score_if_not_chosen'] ? 2 : 0;
}
}
else {
$alternative['is_correct'] = $short['score_if_chosen'] > $short['score_if_not_chosen'] ? 2 : 0;
}
$alternative['answer'] = $this
->checkMarkup($short['answer'], $short['answer_format'], FALSE);
$not = $alternative['is_chosen'] ? '' : '_not';
$alternative['feedback'] = $this
->checkMarkup($short['feedback_if' . $not . '_chosen'], $short['feedback_if' . $not . '_chosen_format'], FALSE);
$data[] = $alternative;
}
$i++;
}
// Return themed report
return array(
'#markup' => theme('multichoice_response', array(
'data' => $data,
)),
);
}
/**
* Order the alternatives according to the choice order stored in the database
*
* @param array $alternatives
* The alternatives to be ordered
*/
private function orderAlternatives(array &$alternatives) {
if (!$this->question->choice_random) {
return;
}
$result = db_query('SELECT choice_order FROM {quiz_multichoice_user_answers}
WHERE result_id = :result_id AND question_nid = :question_nid AND question_vid = :question_vid', array(
':result_id' => $this->rid,
':question_nid' => $this->question->nid,
':question_vid' => $this->question->vid,
))
->fetchField();
if (!$result) {
return;
}
$order = explode(',', $result);
$newAlternatives = array();
foreach ($order as $value) {
foreach ($alternatives as $alternative) {
if ($alternative['id'] == $value) {
$newAlternatives[] = $alternative;
break;
}
}
}
$alternatives = $newAlternatives;
}
/**
* Run check_markup() on the field of the specified choice alternative
*
* @param $alternative
* String to be checked
* @param $format
* The input format to be used
* @param $check_user_access
* Whether or not we are to check the users access to the chosen format
* @return HTML markup
*/
private function checkMarkup($alternative, $format, $check_user_access = FALSE) {
// If the string is empty we don't run it through input filters(They might add empty tags).
if (drupal_strlen($alternative) == 0) {
return '';
}
return check_markup($alternative, $format, $langcode = '', $check_user_access);
}
}
Members
Name | Modifiers | Type | Description | Overrides |
---|---|---|---|---|
MultichoiceResponse:: |
protected | property | ||
MultichoiceResponse:: |
protected | property | ID of the answers. | |
MultichoiceResponse:: |
private | function | Run check_markup() on the field of the specified choice alternative | |
MultichoiceResponse:: |
public | function |
Implementation of delete Overrides QuizQuestionResponse:: |
|
MultichoiceResponse:: |
public | function |
Implementation of getReportFormResponse Overrides QuizQuestionResponse:: |
|
MultichoiceResponse:: |
public | function |
Implementation of getResponse Overrides QuizQuestionResponse:: |
|
MultichoiceResponse:: |
public | function | If all answers in a question is wrong | |
MultichoiceResponse:: |
public | function |
Implementation of isValid Overrides QuizQuestionResponse:: |
|
MultichoiceResponse:: |
private | function | Order the alternatives according to the choice order stored in the database | |
MultichoiceResponse:: |
public | function |
Implementation of save Overrides QuizQuestionResponse:: |
|
MultichoiceResponse:: |
public | function |
Implementation of score Overrides QuizQuestionResponse:: |
|
MultichoiceResponse:: |
public | function |
Constructor Overrides QuizQuestionResponse:: |
|
QuizQuestionResponse:: |
protected | property | ||
QuizQuestionResponse:: |
protected | property | ||
QuizQuestionResponse:: |
protected | property | ||
QuizQuestionResponse:: |
public | property | ||
QuizQuestionResponse:: |
public | property | ||
QuizQuestionResponse:: |
protected | property | ||
QuizQuestionResponse:: |
protected | property | 7 | |
QuizQuestionResponse:: |
public | function | Returns stored max score if it exists, if not the max score is calculated and returned. | |
QuizQuestionResponse:: |
public | function | Get data suitable for reporting a user's score on the question. This expects an object with the following attributes: | |
QuizQuestionResponse:: |
public | function | Creates the report form for the admin pages, and for when a user gets feedback after answering questions. | 1 |
QuizQuestionResponse:: |
public | function | get the question part of the reportForm | |
QuizQuestionResponse:: |
public | function | Get the score part of the report form | 2 |
QuizQuestionResponse:: |
public | function | Get the submit function for the reportForm | 2 |
QuizQuestionResponse:: |
public | function | Get the theme key for the reportForm | |
QuizQuestionResponse:: |
public | function | Get the validate function for the reportForm | 2 |
QuizQuestionResponse:: |
function | Returns stored score if it exists, if not the score is calculated and returned. | ||
QuizQuestionResponse:: |
function | Check to see if the answer is marked as correct. | 1 | |
QuizQuestionResponse:: |
public | function | Indicate whether the response has been evaluated (scored) yet. Questions that require human scoring (e.g. essays) may need to manually toggle this. | |
QuizQuestionResponse:: |
public | function | Used to refresh this instances question node in case drupal has changed it. | |
QuizQuestionResponse:: |
public | function | Saves the quiz result. This is not used when a question is skipped! | |
QuizQuestionResponse:: |
function | Represent the response as a stdClass object. |