MDL-16559 Remove question/comment.html template, and instead just have a function in questionlib.php
MDL-16562 Regression from MDL-16263, notices when regrading.
Sorry, the three got tangled together (coupled through questionlib) while I was fixing bugs in preparation for a demo.
$string['errorduringpre'] = 'Error occurred during pre-processing!';
$string['errorduringproc'] = 'Error occurred during processing!';
$string['errorduringpost'] = 'Error occurred during post-processing!';
+$string['errorduringregrade'] = 'Could not regrade question $a->qid, going to state $a->stateid.';
$string['errorfilecannotbecopied'] = 'Error cannot copy file $a.';
$string['errorfilecannotbemoved'] = 'Error cannot move file $a.';
$string['errorfileschanged'] = 'Error files linked to from questions have changed since form was displayed.';
*/
function question_preload_states($attemptid) {
global $DB;
+ // Note, changes here probably also need to be reflected in
+ // regrade_question_in_attempt and question_load_specific_state.
// The questionid field must be listed first so that it is used as the
// array index in the array returned by $DB->get_records_sql
function question_load_specific_state($question, $cmoptions, $attempt, $stateid) {
global $DB, $QUESTION_EVENTS_GRADED;
// Load specified states for the question.
- $sql = 'SELECT st.*, sess.sumpenalty, sess.manualcomment
+ // sess.sumpenalty is probably wrong here shoul really be a sum of penalties from before the one we are asking for.
+ $sql = 'SELECT st.*, sess.sumpenalty, sess.manualcomment, sess.flagged, sess.id as questionsessionid
FROM {question_states} st, {question_sessions} sess
WHERE st.id = ?
AND st.attempt = ?
// Load the most recent graded states for the questions before the specified one.
list($eventinsql, $params) = $DB->get_in_or_equal($QUESTION_EVENTS_GRADED);
- $sql = 'SELECT st.*, sess.sumpenalty, sess.manualcomment
+ $sql = 'SELECT st.*, sess.sumpenalty, sess.manualcomment, sess.flagged, sess.id as questionsessionid
FROM {question_states} st, {question_sessions} sess
WHERE st.seq_number <= ?
AND st.attempt = ?
$attempt->sumgrades -= $states[count($states)-1]->grade;
// Initialise the replaystate
- $state = clone($states[0]);
- $state->manualcomment = $DB->get_field('question_sessions', 'manualcomment',
- array('attemptid'=> $attempt->uniqueid, 'questionid'=>$question->id));
- restore_question_state($question, $state);
- $state->sumpenalty = 0.0;
- $replaystate = clone($state);
- $replaystate->last_graded = $state;
+ $replaystate = question_load_specific_state($question, $cmoptions, $attempt, $states[0]->id);
+ $replaystate->sumpenalty = 0;
+ $replaystate->last_graded->sumpenalty = 0;
$changed = false;
for($j = 1; $j < count($states); $j++) {
$action->timestamp = $states[$j]->timestamp;
// Change event to submit so that it will be reprocessed
- if (QUESTION_EVENTCLOSE == $states[$j]->event
- or QUESTION_EVENTGRADE == $states[$j]->event
- or QUESTION_EVENTCLOSEANDGRADE == $states[$j]->event) {
+ if (in_array($states[$j]->event, array(QUESTION_EVENTCLOSE,
+ QUESTION_EVENTGRADE, QUESTION_EVENTCLOSEANDGRADE))) {
$action->event = QUESTION_EVENTSUBMIT;
// By default take the event that was saved in the database
} else {
// Reprocess (regrade) responses
if (!question_process_responses($question, $replaystate,
- $action, $cmoptions, $attempt)) {
- $verbose && notify("Couldn't regrade state #{$state->id}!");
+ $action, $cmoptions, $attempt) && $verbose) {
+ $a = new stdClass;
+ $a->qid = $question->id;
+ $a->stateid = $states[$j]->id;
+ notify(get_string('errorduringregrade', 'question', $a));
}
// We need rounding here because grades in the DB get truncated
// e.g. 0.33333 != 0.3333333, but we want them to be equal here
or (round((float)$replaystate->grade, 5) != round((float)$states[$j]->grade, 5))) {
$changed = true;
}
+ // If this was previously a closed state, and it has been knoced back to
+ // graded, then fix up the state again.
+ if ($replaystate->event == QUESTION_EVENTGRADE &&
+ ($states[$j]->event == QUESTION_EVENTCLOSE ||
+ $states[$j]->event == QUESTION_EVENTCLOSEANDGRADE)) {
+ $replaystate->event = $states[$j]->event;
+ }
}
$replaystate->id = $states[$j]->id;
return $img;
}
-function question_print_comment_box($question, $state, $attempt, $url) {
- global $CFG;
-
- $prefix = 'response';
- $usehtmleditor = can_use_html_editor();
- $grade = round($state->last_graded->grade, 3);
- echo '<form method="post" action="'.$url.'">';
- include($CFG->dirroot.'/question/comment.html');
- echo '<input type="hidden" name="attempt" value="'.$attempt->uniqueid.'" />';
- echo '<input type="hidden" name="question" value="'.$question->id.'" />';
- echo '<input type="hidden" name="sesskey" value="'.sesskey().'" />';
- echo '<input type="submit" name="submit" value="'.get_string('save', 'quiz').'" />';
- echo '</form>';
+function question_print_comment_fields($question, $state, $prefix, $cmoptions) {
+ $idprefix = preg_replace('/[^-_a-zA-Z0-9]/', '', $prefix);
+ $grade = question_format_grade($cmoptions, $state->last_graded->grade);
+ $maxgrade = question_format_grade($cmoptions, $question->maxgrade);
+ $fieldsize = strlen($maxgrade) - 1;
+ echo '<table class="que comment">' . "\n";
+ echo '<tr valign="top">' . "\n";
+ echo '<th><label for="' . $idprefix . '_comment_box">' .
+ get_string('comment', 'quiz') . "</label></th>\n";
+ echo '<td>';
+ print_textarea(can_use_html_editor(), 15, 60, 630, 300, $prefix . '[comment]',
+ $state->manualcomment, 0, false, $idprefix . '_comment_box');
+ echo "</td>\n";
+ echo "</tr>\n";
+ echo '<tr valign="top">' . "\n";
+ echo '<th><label for="' . $idprefix . '_grade_field">' . get_string('grade', 'quiz') . "</label></th>\n";
+ echo '<td><input type="text" name="' . $prefix . '[grade]" size="' . $fieldsize .
+ '" id="' . $idprefix . '_grade_field" value="' . $grade . '" />/' . $maxgrade . "</td>\n";
+ echo "</tr>\n";
+ echo "</table>\n";
}
/**
}
// Update the state if either the score has changed, or this is the first
- // manual grade event and there is actually a grade of comment to process.
+ // manual grade event and there is actually a grade or comment to process.
// We don't need to store the modified state in the database, we just need
// to set the $state->changed flag.
if (abs($state->last_graded->grade - $grade) > 0.002 ||
return $this->attempt->id;
}
+ /** @return integer the attempt unique id. */
+ public function get_uniqueid() {
+ return $this->attempt->uniqueid;
+ }
+
/** @return object the row from the quiz_attempts table. */
public function get_attempt() {
return $this->attempt;
return implode(', ', $attemptlist);
}
+ // Methods for processing manual comments ==============================================
+ // I am not sure it is a good idea to have update methods here - this class is only
+ // about getting data out of the question engine, and helping to display it, apart from
+ // this.
+ public function process_comment($questionid, $comment, $grade) {
+ $this->ensure_question_loaded($questionid);
+ $this->ensure_state_loaded($questionid);
+ $state = $this->states[$questionid];
+
+ $error = question_process_comment($this->questions[$questionid],
+ $state, $this->attempt, $comment, $grade);
+
+ // If the state was update (successfully), save the changes.
+ if (!is_string($error) && $state->changed) {
+ if (!save_question_session($this->questions[$questionid], $state)) {
+ $error = get_string('errorudpatingquestionsession', 'quiz');
+ }
+ if (!quiz_save_best_grade($this->quiz, $this->attempt->userid)) {
+ $error = get_string('errorudpatingbestgrade', 'quiz');
+ }
+ }
+ return $error;
+ }
+
+ public function question_print_comment_fields($questionid, $prefix) {
+ $this->ensure_question_loaded($questionid);
+ $this->ensure_state_loaded($questionid);
+ question_print_comment_fields($this->questions[$questionid],
+ $this->states[$questionid], $prefix, $this->quiz);
+ }
+
// Private methods =====================================================================
// Check that the state of a particular question is loaded, and if not throw an exception.
private function ensure_state_loaded($id) {
$attemptobj = new quiz_attempt($attemptid);
- // Teachers are only allowed to comment and grade on closed attempts
- if (!($attempt->timefinish > 0)) {
+/// Can only grade finished attempts.
+ if (!$attemptobj->is_finished()) {
print_error('attemptclosed', 'quiz');
}
- $cm = get_coursemodule_from_instance('quiz', $quiz->id);
- require_login($course, true, $cm);
+/// Check login and permissions.
+ require_login($attemptobj->get_courseid(), false, $attemptobj->get_cm());
+ $attemptobj->require_capability('mod/quiz:grade');
- $context = get_context_instance(CONTEXT_MODULE, $cm->id);
+/// Load the questions and states.
+ $questionids = array($questionid);
+ $attemptobj->load_questions($questionids);
+ $attemptobj->load_question_states($questionids);
- require_capability('mod/quiz:grade', $context);
-
- // Load question
- if (! $question = $DB->get_record('question', array('id' => $questionid))) {
- print_error('questionmissing', 'quiz');
- }
- $question->maxgrade = $DB->get_field('quiz_question_instances', 'grade', array('quiz' => $quiz->id, 'question' => $question->id));
- // Some of the questions code is optimised to work with several questions
- // at once so it wants the question to be in an array.
- $key = $question->id;
- $questions[$key] = &$question;
- // Add additional questiontype specific information to the question objects.
- if (!get_question_options($questions)) {
- print_error('cannotloadtypeinfo', 'quiz');
- }
-
- // Load state
- $states = get_question_states($questions, $quiz, $attempt);
- // The $states array is indexed by question id but because we are dealing
- // with only one question there is only one entry in this array
- $state = &$states[$question->id];
+/// Log this action.
+ add_to_log($attemptobj->get_courseid(), 'quiz', 'manualgrade', 'comment.php?attempt=' .
+ $attemptobj->get_attemptid() . '&question=' . $questionid,
+ $attemptobj->get_quizid(), $attemptobj->get_cmid());
+/// Print the page header
print_header();
- print_heading(format_string($question->name));
-
- //add_to_log($course->id, 'quiz', 'review', "review.php?id=$cm->id&attempt=$attempt->id", "$quiz->id", "$cm->id");
+ print_heading(format_string($attemptobj->get_question($questionid)->name));
+/// Process any data that was submitted.
if ($data = data_submitted() and confirm_sesskey()) {
- // the following will update the state and attempt
- $error = question_process_comment($question, $state, $attempt, $data->response['comment'], $data->response['grade']);
- if (is_string($error)) {
- notify($error);
- } else {
- // If the state has changed save it and update the quiz grade
- if ($state->changed) {
- save_question_session($question, $state);
- quiz_save_best_grade($quiz, $attempt->userid);
- }
-
- notify(get_string('changessaved'));
- echo '<div class="boxaligncenter"><input type="button" onclick="window.opener.location.reload(1); self.close();return false;" value="' .
- get_string('closewindow') . "\" /></div>";
+ $error = $attemptobj->process_comment($questionid,
+ $data->response['comment'], $data->response['grade']);
+ /// If success, notify and print a close button.
+ if (!is_string($error)) {
+ notify(get_string('changessaved'), 'notifysuccess');
+ close_window_button('closewindow', false, true);
print_footer();
exit;
}
+
+ /// Otherwise, display the error and fall throug to re-display the form.
+ notify($error);
}
- question_print_comment_box($question, $state, $attempt, $CFG->wwwroot.'/mod/quiz/comment.php');
+/// Print the comment form.
+ echo '<form method="post" action="' . $CFG->wwwroot . '/mod/quiz/comment.php">';
+ $attemptobj->question_print_comment_fields($questionid, 'response');
+ echo '<input type="hidden" name="attempt" value="' . $attemptobj->get_uniqueid() . '" />';
+ echo '<input type="hidden" name="question" value="' . $questionid . '" />';
+ echo '<input type="hidden" name="sesskey" value="' . sesskey() . '" />';
+ echo '<input type="submit" name="submit" value="' . get_string('save', 'quiz') . '" />';
+ echo '</form>';
+/// End of the page.
print_footer();
-
?>
$options = quiz_get_reviewoptions($quiz, $attempt, $context);
unset($options->questioncommentlink);
- $copy = $state->manualcomment;
- $state->manualcomment = '';
-
$options->readonly = 1;
- $gradedclass = question_state_is_graded($state)?' class="highlightgraded" ':'';
- $gradedstring = question_state_is_graded($state)?(' '.get_string('graded','quiz_grading')):'';
+ if (question_state_is_graded($state)) {
+ $gradedclass = ' class="highlightgraded" ';
+ $gradedstring = ' ' . get_string('graded','quiz_grading');
+ } else {
+ $gradedclass = '';
+ $gradedstring = '';
+ }
$a = new object();
$a->fullname = fullname($attempt, true);
$a->attempt = $attempt->attempt;
// print the user name, attempt count, the question, and some more hidden fields
echo '<div class="boxaligncenter" width="80%" style="clear:left;padding:15px;">';
- echo "<span$gradedclass>".get_string('gradingattempt','quiz_grading', $a);
- echo $gradedstring."</span>";
+ echo "<span$gradedclass>" . get_string('gradingattempt', 'quiz_grading', $a);
+ echo $gradedstring . '</span>';
+ // Print the question, without showing any previous comment.
+ $copy = $state->manualcomment;
+ $state->manualcomment = '';
print_question($question, $state, '', $quiz, $options);
- $prefix = "manualgrades[$attempt->uniqueid]";
- $grade = round($state->last_graded->grade, 3);
+ // The print the comment and grade fields, putting back the previous comment.
$state->manualcomment = $copy;
-
- include($CFG->dirroot . '/question/comment.html');
+ question_print_comment_fields($question, $state,
+ 'manualgrades[' . $attempt->uniqueid . ']', $quiz);
echo '</div>';
}
/// Check login.
require_login($attemptobj->get_courseid(), false, $attemptobj->get_cm());
-/// Create an object to manage all the other (non-roles) access rules.
- $accessmanager = $attemptobj->get_access_manager(time());
- $options = $attemptobj->get_review_options();
-
/// Permissions checks for normal users who do not have quiz:viewreports capability.
if (!$attemptobj->has_capability('mod/quiz:viewreports')) {
/// Can't review during the attempt - send them back to the attempt page.
}
/// Can't review unless Students may review -> Responses option is turned on.
if (!$options->responses) {
- notify($accessmanager->cannot_review_message($options));
+ $accessmanager = $attemptobj->get_access_manager(time());
+ notify($accessmanager->cannot_review_message($attemptobj->get_review_options()));
close_window_button();
}
}
+++ /dev/null
-<?php
-// This template determines the layout of the manual grading form. It is
-// included by both by question_print_comment_box method from questionlib.php,
-// and the print_questions_and_form method from the manual grading report.
-?>
-<table class="que comment">
- <tr valign="top">
- <td>
- <b><?php print_string('comment', 'quiz'); ?>: </b>
- </td>
- <td>
- <?php
- print_textarea($usehtmleditor, 15, 60, 630, 300, $prefix.'[comment]', $state->manualcomment);
- ?>
- </td>
- </tr>
- <tr valign="top">
- <td>
- <b><?php print_string('grade', 'quiz'); ?>: </b>
- </td>
- <td>
- <input type="text" name="<?php echo $prefix; ?>[grade]" size="2" value="<?php echo $grade; ?>" />/<?php echo $question->maxgrade; ?>
- </td>
- </tr>
-</table>
\ No newline at end of file