Source: includes/class-sensei-grading-user-quiz.php

<?php
/**
 * File containing the Sensei_Grading_User_Quiz class.
 *
 * @package sensei
 */

if ( ! defined( 'ABSPATH' ) ) {
	exit; // Exit if accessed directly.
}

/**
 * Grading User Profile in Sensei.
 *
 * @package Assessment
 * @author Automattic
 *
 * @since 1.3.0
 */
class Sensei_Grading_User_Quiz {
	/**
	 * The user id which this quiz is for.
	 *
	 * @var int
	 */
	private $user_id;

	/**
	 * The lesson id which the quiz belongs to.
	 *
	 * @var int
	 */
	private $lesson_id;

	/**
	 * The quiz id.
	 *
	 * @var int
	 */
	private $quiz_id;

	/**
	 * Sensei_Grading_User_Quiz constructor.
	 *
	 * @since  1.3.0
	 *
	 * @param int $user_id  The user id.
	 * @param int $quiz_id  The quiz id.
	 */
	public function __construct( $user_id = 0, $quiz_id = 0 ) {
		$this->user_id   = intval( $user_id );
		$this->quiz_id   = intval( $quiz_id );
		$this->lesson_id = get_post_meta( $this->quiz_id, '_quiz_lesson', true );
	}

	/**
	 * Builds the data for use on the page.
	 *
	 * @since  1.3.0
	 * @return array
	 */
	public function build_data_array() {
		return Sensei_Utils::sensei_get_quiz_questions( $this->quiz_id );
	}

	/**
	 * Helper method which removes the hash from a filename if it starts with one.
	 *
	 * @param string $filename  The filename.
	 *
	 * @return string
	 */
	public static function remove_hash_prefix( $filename ) {
		$starts_with_hash = preg_match( '/^[a-f0-9]{32}_/', $filename );

		if ( $starts_with_hash ) {
			return substr( $filename, 33 );
		}

		return $filename;
	}

	/**
	 * Display output to the admin view
	 *
	 * This view is shown when grading a quiz for a single user in admin under grading
	 *
	 * @since  1.3.0
	 */
	public function display() {
		// Get data for the user.
		$questions = $this->build_data_array();

		$count                 = 0;
		$graded_count          = 0;
		$user_quiz_grade_total = 0;
		$quiz_grade_total      = 0;
		$quiz_grade            = 0;
		$lesson_id             = $this->lesson_id;
		$user_id               = $this->user_id;

		?><form name="<?php echo esc_attr( 'quiz_' . $this->quiz_id ); ?>" action="" method="post">
			<?php wp_nonce_field( 'sensei_manual_grading', '_wp_sensei_manual_grading_nonce' ); ?>
			<input type="hidden" name="sensei_manual_grade" value="<?php echo esc_attr( $this->quiz_id ); ?>" />
			<input type="hidden" name="sensei_grade_next_learner" value="<?php echo esc_attr( $this->user_id ); ?>" />
			<div class="total_grade_display">
				<span><?php esc_attr_e( 'Grade:', 'sensei-lms' ); ?></span>
				<span class="total_grade_total"><?php echo esc_html( $user_quiz_grade_total ); ?></span> / <span class="quiz_grade_total"><?php echo esc_html( $quiz_grade_total ); ?></span> (<span class="total_grade_percent"><?php echo esc_html( $quiz_grade ); ?></span>%)
			</div>
			<div class="buttons">
				<input type="submit" value="<?php esc_attr_e( 'Save', 'sensei-lms' ); ?>" class="grade-button button-primary" title="<?php esc_attr_e( 'Saves grades as currently marked on this page', 'sensei-lms' ); ?>" />
				<input type="button" value="<?php esc_attr_e( 'Auto grade', 'sensei-lms' ); ?>" class="autograde-button button-secondary" title="<?php esc_attr_e( 'Where possible, automatically grades questions that have not yet been graded', 'sensei-lms' ); ?>" />
				<input type="button" value="<?php esc_attr_e( 'Reset', 'sensei-lms' ); ?>" class="reset-button button-link button-link-delete" title="<?php esc_attr_e( 'Resets all questions to ungraded and total grade to 0', 'sensei-lms' ); ?>" />
			</div>
			<div class="clear"></div><br/>
			<?php

			$lesson_status_id = Sensei_Utils::sensei_get_activity_value(
				array(
					'post_id' => $this->lesson_id,
					'user_id' => $this->user_id,
					'type'    => 'sensei_lesson_status',
					'field'   => 'comment_ID',
				)
			);
			$user_quiz_grade  = get_comment_meta( $lesson_status_id, 'grade', true );

			foreach ( $questions as $question ) {
				$question_id = $question->ID;
				++$count;

				$type      = false;
				$type_name = '';

				$type = Sensei()->question->get_question_type( $question_id );

				$custom_feedback       = Sensei()->quiz->get_user_answers_feedback( $lesson_id, $user_id );
				$custom_feedback       = $custom_feedback[ $question_id ] ?? '';
				$correct_feedback      = Sensei_Quiz::get_correct_answer_feedback( $question_id );
				$incorrect_feedback    = Sensei_Quiz::get_incorrect_answer_feedback( $question_id );
				$question_answer_notes = Sensei()->quiz->get_user_question_feedback( $lesson_id, $question_id, $user_id );

				if ( ! $correct_feedback && ! $incorrect_feedback ) {
					$custom_feedback = $question_answer_notes;
				}

				$question_grade_total = Sensei()->question->get_question_grade( $question_id );
				$quiz_grade_total    += $question_grade_total;

				$right_answer        = get_post_meta( $question_id, '_question_right_answer', true );
				$user_answer_content = Sensei()->quiz->get_user_question_answer( $lesson_id, $question_id, $user_id );
				$type_name           = __( 'Multiple Choice', 'sensei-lms' );
				$grade_type          = 'manual-grade';

				switch ( $type ) {
					case 'boolean':
						$type_name           = __( 'True/False', 'sensei-lms' );
						$right_answer        = ucfirst( $right_answer );
						$user_answer_content = ucfirst( $user_answer_content );
						$grade_type          = 'auto-grade';
						break;
					case 'multiple-choice':
						$type_name  = __( 'Multiple Choice', 'sensei-lms' );
						$grade_type = 'auto-grade';
						break;
					case 'gap-fill':
						$type_name = __( 'Gap Fill', 'sensei-lms' );

						$right_answer_array = explode( '||', $right_answer );
						if ( isset( $right_answer_array[0] ) ) {
							$gapfill_pre = $right_answer_array[0];
						} else {
							$gapfill_pre = ''; }
						if ( isset( $right_answer_array[1] ) ) {
							$gapfill_gap = $right_answer_array[1];
						} else {
							$gapfill_gap = ''; }
						if ( isset( $right_answer_array[2] ) ) {
							$gapfill_post = $right_answer_array[2];
						} else {
							$gapfill_post = ''; }

						if ( ! $user_answer_content ) {
							$user_answer_content = '______';
						}

						$right_answer        = $gapfill_pre . ' <span class="highlight">' . $gapfill_gap . '</span> ' . $gapfill_post;
						$user_answer_content = $gapfill_pre . ' <span class="highlight">' . $user_answer_content . '</span> ' . $gapfill_post;
						$grade_type          = 'auto-grade';

						break;
					case 'multi-line':
						$type_name  = __( 'Multi Line', 'sensei-lms' );
						$grade_type = 'manual-grade';
						break;
					case 'single-line':
						$type_name  = __( 'Single Line', 'sensei-lms' );
						$grade_type = 'manual-grade';
						break;
					case 'file-upload':
						$type_name  = __( 'File Upload', 'sensei-lms' );
						$grade_type = 'manual-grade';

						// Get uploaded file.
						if ( $user_answer_content ) {
							$attachment_id = $user_answer_content;
							if ( 0 < intval( $attachment_id ) ) {
								$answer_media_url      = wp_get_attachment_url( $attachment_id );
								$filename_raw          = basename( $answer_media_url );
								$answer_media_filename = self::remove_hash_prefix( $filename_raw );

								if ( $answer_media_url && $answer_media_filename ) {
									// translators: Placeholder %1$s is a link to the submitted file.
									$user_answer_content = sprintf( __( 'Submitted file: %1$s', 'sensei-lms' ), '<a href="' . esc_url( $answer_media_url ) . '" target="_blank">' . esc_html( $answer_media_filename ) . '</a>' );
								}
							}
						} else {
							$user_answer_content = '';
						}
						break;
					default:
						// Nothing.
						break;
				}

				/**
				 * Filters the various values which are displayed in the grading admin page for each quiz question.
				 * The expected values are type_name, right_answer, user_answer_content and grade_type
				 *
				 * @since 3.15.0
				 *
				 * @hook sensei_grading_display_quiz_question
				 *
				 * @param {array|null}   $display_values {
				 *     Optional. An array of arguments or null.
				 *
				 *     @key {string}       $type_name           The question type.
				 *     @key {string|array} $right_answer        The right answer to the quiz.
				 *     @key {string|array} $user_answer_content The user supplied answer to the quiz.
				 *     @key {string}       $grade_type          Auto or manual grading.
				 * }
				 * @param {string} $type
				 * @param {int}    $question_id
				 * @return {array|null}
				 */
				$possibly_new_args = apply_filters( 'sensei_grading_display_quiz_question', null, $type, $question_id, $right_answer, $user_answer_content );

				if ( null !== $possibly_new_args && $possibly_new_args ) {
					$type_name           = $possibly_new_args['type_name'] ?? $type_name;
					$right_answer        = $possibly_new_args['right_answer'] ?? $right_answer;
					$user_answer_content = $possibly_new_args['user_answer_content'] ?? $user_answer_content;
					$grade_type          = $possibly_new_args['grade_type'] ?? $grade_type;
				}

				$quiz_grade_type = get_post_meta( $this->quiz_id, '_quiz_grade_type', true );

				// Don't auto-grade if "Grade quiz automatically" isn't selected in Quiz Settings,
				// regardless of question type.
				if ( 'manual' === $quiz_grade_type ) {
					$grade_type = 'manual-grade';
				}

				$user_answer_content = (array) $user_answer_content;
				$right_answer        = (array) $right_answer;
				// translators: Placeholder is the question number.
				$question_title = sprintf( __( 'Question %d: ', 'sensei-lms' ), $count ) . $type_name;

				$graded_class        = '';
				$user_question_grade = Sensei()->quiz->get_user_question_grade( $lesson_id, $question_id, $user_id );
				$graded_class        = 'ungraded';

				// Question with no grade value associated with it.
				if ( 0 === $question_grade_total ) {
					$grade_type          = 'zero-graded';
					$graded_class        = '';
					$user_question_grade = 0;
					$graded_count++;
				} else {
					$user_right = intval( $user_question_grade ) > 0;
					// The user's grade will be 0 if they answered incorrectly.
					// Don't set a grade for questions that are part of an auto-graded quiz, but that must be manually graded.
					$user_wrong =
						( 'manual' === $quiz_grade_type && 0 === $user_question_grade )
						|| ( 'auto' === $quiz_grade_type && 'manual-grade' === $grade_type && 0 === $user_question_grade );

					if ( $user_right ) {
						$graded_class           = 'user_right';
						$user_quiz_grade_total += $user_question_grade;
						$graded_count++;
					} elseif ( $user_wrong ) {
						$graded_class        = 'user_wrong';
						$user_question_grade = 0;
						$graded_count++;
					}
				}
				?>

			<div class="postbox question_box <?php echo esc_attr( $type ); ?> <?php echo esc_attr( $grade_type ); ?> <?php echo esc_attr( $graded_class ); ?>" id="<?php echo esc_attr( 'question_' . $question_id . '_box' ); ?>">
				<div class="handlediv" title="Click to toggle"><br></div>
				<h3 class="hndle"><span><?php echo esc_html( $question_title ); ?></span></h3>
				<div class="inside">
					<div class="sensei-grading-actions">
						<div class="actions">
							<input type="hidden" class="question_id" value="<?php echo esc_attr( $question_id ); ?>" />
							<input type="hidden" class="question_total_grade" name="question_total_grade" value="<?php echo esc_attr( $question_grade_total ); ?>" />
							<span class="grading-mark icon_right"><input type="radio" class="<?php echo esc_attr( 'question_' . $question_id . '_right_option' ); ?>" name="<?php echo esc_attr( 'question_' . $question_id ); ?>" value="right" <?php checked( $graded_class, 'user_right', true ); ?> /></span>
							<span class="grading-mark icon_wrong"><input type="radio" class="<?php echo esc_attr( 'question_' . $question_id . '_wrong_option' ); ?>" name="<?php echo esc_attr( 'question_' . $question_id ); ?>" value="wrong" <?php checked( $graded_class, 'user_wrong', true ); ?> /></span>
							<input type="number" class="question-grade" name="<?php echo esc_attr( 'question_' . $question_id . '_grade' ); ?>" id="<?php echo esc_attr( 'question_' . $question_id . '_grade' ); ?>" value="<?php echo esc_attr( $user_question_grade ); ?>" min="0" max="<?php echo esc_attr( $question_grade_total ); ?>" />
							<span class="question-grade-total"><?php echo esc_html( $question_grade_total ); ?></span>
						</div>
					</div>
					<div class="sensei-grading-answer">
						<h4>
						<?php
							/**
							 * Filters the question title.
							 *
							 * @hook sensei_question_title
							 *
							 * @param {string} $question_title The question title.
							 * @return {string} Filtered question title.
							 */
							echo wp_kses_post( apply_filters( 'sensei_question_title', $question->post_title ) );
						?>
						</h4>
						<?php
						// phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- Output escaped before core filter applied.
						echo Sensei_Question::get_the_question_description( $question_id );
						?>
						<?php
						foreach ( $user_answer_content as $_user_answer ) {

							if ( 'multi-line' === Sensei()->question->get_question_type( $question->ID ) ) {
								$is_plaintext = sanitize_textarea_field( $_user_answer ) === $_user_answer;
								if ( $is_plaintext ) {
									$_user_answer = nl2br( $_user_answer );
								}

								$_user_answer = htmlspecialchars_decode( $_user_answer );
							}

							/**
							 * Filter user answer text.
							 *
							 * @hook sensei_answer_text
							 *
							 * @param {string} Answer text.
							 * @return {string} Filtered answer text.
							 */
							$_user_answer = apply_filters( 'sensei_answer_text', $_user_answer );

							$html = wp_kses_post( $_user_answer );
							$html = '<html><head><title></title></head><body>' . $html . '</body></html>';
							?>
							<iframe class="user-answer" srcdoc="<?php echo esc_attr( $html ); ?>" sandbox="allow-same-origin" height="auto"></iframe>
							<?php
						}
						?>

						<div class="right-answer">
							<h5><?php esc_html_e( 'Correct answer', 'sensei-lms' ); ?></h5>
							<span class="correct-answer">
							<?php
							foreach ( $right_answer as $_right_answer ) {

								if ( 'multi-line' === Sensei()->question->get_question_type( $question->ID ) ) {
									$_right_answer = htmlspecialchars_decode( nl2br( $_right_answer ) );
								}

								/**
								 * Filter user answer text.
								 *
								 * @hook sensei_answer_text
								 *
								 * @param {string} Answer text.
								 * @return {string} Filtered answer text.
								 */
								echo wp_kses_post( apply_filters( 'sensei_answer_text', $_right_answer ) ) . '<br>';

							}
							?>
							</span>
						</div>
						<div class="answer-notes">
							<h5><?php esc_html_e( 'Answer Feedback', 'sensei-lms' ); ?></h5>
							<div class="answer-feedback-correct"><?php echo wp_kses_post( $correct_feedback ); ?></div>
							<div class="answer-feedback-incorrect"><?php echo wp_kses_post( $incorrect_feedback ); ?></div>
							<textarea class="correct-answer" name="questions_feedback[<?php echo esc_attr( $question_id ); ?>]" placeholder="<?php esc_attr_e( 'Add custom feedback here...', 'sensei-lms' ); ?>"><?php echo esc_html( $custom_feedback ); ?></textarea>
						</div>
					</div>
				</div>
			</div>
				<?php
			}

			$quiz_grade = intval( $user_quiz_grade );
			$all_graded = 'no';
			if ( intval( $count ) === intval( $graded_count ) ) {
				$all_graded = 'yes';
			}

			?>
			<input type="hidden" name="total_grade" id="total_grade" value="<?php echo esc_attr( $user_quiz_grade_total ); ?>" />
			<input type="hidden" name="total_questions" id="total_questions" value="<?php echo esc_attr( $count ); ?>" />
			<input type="hidden" name="quiz_grade_total" id="quiz_grade_total" value="<?php echo esc_attr( $quiz_grade_total ); ?>" />
			<input type="hidden" name="total_graded_questions" id="total_graded_questions" value="<?php echo esc_attr( $graded_count ); ?>" />
			<input type="hidden" name="all_questions_graded" id="all_questions_graded" value="<?php echo esc_attr( $all_graded ); ?>" />
			<div class="total_grade_display">
				<span><?php esc_attr_e( 'Grade:', 'sensei-lms' ); ?></span>
				<span class="total_grade_total"><?php echo esc_html( $user_quiz_grade_total ); ?></span> / <span class="quiz_grade_total"><?php echo esc_html( $quiz_grade_total ); ?></span> (<span class="total_grade_percent"><?php echo esc_html( $quiz_grade ); ?></span>%)
			</div>
			<div class="buttons">
				<input type="submit" value="<?php esc_attr_e( 'Save', 'sensei-lms' ); ?>" class="grade-button button-primary" title="Saves grades as currently marked on this page" />
				<input type="button" value="<?php esc_attr_e( 'Auto grade', 'sensei-lms' ); ?>" class="autograde-button button-secondary" title="Where possible, automatically grades questions that have not yet been graded" />
				<input type="button" value="<?php esc_attr_e( 'Reset', 'sensei-lms' ); ?>" class="reset-button button-link button-link-delete" title="Resets all questions to ungraded and total grade to 0" />
			</div>
			<div class="clear"></div>
		</form>
		<?php
	}
}

/**
 * Class WooThemes_Sensei_Grading_User_Quiz
 *
 * @ignore only for backward compatibility
 * @since 1.9.0
 */
class WooThemes_Sensei_Grading_User_Quiz extends Sensei_Grading_User_Quiz{}