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