mirror of
https://github.com/alexmickelson/canvasManagement.git
synced 2026-03-26 07:38:33 -06:00
starting to handle feedback parsing bug
This commit is contained in:
@@ -1,188 +0,0 @@
|
|||||||
import { describe, it, expect, vi, beforeEach } from "vitest";
|
|
||||||
import { LocalQuiz } from "@/features/local/quizzes/models/localQuiz";
|
|
||||||
import { QuestionType } from "@/features/local/quizzes/models/localQuizQuestion";
|
|
||||||
import { DayOfWeek } from "@/features/local/course/localCourseSettings";
|
|
||||||
import { AssignmentSubmissionType } from "@/features/local/assignments/models/assignmentSubmissionType";
|
|
||||||
|
|
||||||
// Mock the dependencies
|
|
||||||
vi.mock("@/services/axiosUtils", () => ({
|
|
||||||
axiosClient: {
|
|
||||||
get: vi.fn(),
|
|
||||||
post: vi.fn(),
|
|
||||||
delete: vi.fn(),
|
|
||||||
},
|
|
||||||
}));
|
|
||||||
|
|
||||||
vi.mock("./canvasServiceUtils", () => ({
|
|
||||||
canvasApi: "https://test.instructure.com/api/v1",
|
|
||||||
paginatedRequest: vi.fn(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
vi.mock("./canvasAssignmentService", () => ({
|
|
||||||
canvasAssignmentService: {
|
|
||||||
getAll: vi.fn(() => Promise.resolve([])),
|
|
||||||
delete: vi.fn(() => Promise.resolve()),
|
|
||||||
},
|
|
||||||
}));
|
|
||||||
|
|
||||||
vi.mock("@/services/htmlMarkdownUtils", () => ({
|
|
||||||
markdownToHTMLSafe: vi.fn(({ markdownString }) => `<p>${markdownString}</p>`),
|
|
||||||
}));
|
|
||||||
|
|
||||||
vi.mock("@/features/local/utils/timeUtils", () => ({
|
|
||||||
getDateFromStringOrThrow: vi.fn((dateString) => new Date(dateString)),
|
|
||||||
}));
|
|
||||||
|
|
||||||
vi.mock("@/services/utils/questionHtmlUtils", () => ({
|
|
||||||
escapeMatchingText: vi.fn((text) => text),
|
|
||||||
}));
|
|
||||||
|
|
||||||
describe("Quiz Order Verification Integration", () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
vi.clearAllMocks();
|
|
||||||
});
|
|
||||||
|
|
||||||
it("demonstrates the question order verification workflow", async () => {
|
|
||||||
// This test demonstrates that the verification step is properly integrated
|
|
||||||
// into the quiz creation workflow
|
|
||||||
|
|
||||||
const testQuiz: LocalQuiz = {
|
|
||||||
name: "Test Quiz - Order Verification",
|
|
||||||
description: "Testing question order verification",
|
|
||||||
dueAt: "2023-12-01T23:59:00Z",
|
|
||||||
shuffleAnswers: false,
|
|
||||||
showCorrectAnswers: true,
|
|
||||||
oneQuestionAtATime: false,
|
|
||||||
allowedAttempts: 1,
|
|
||||||
questions: [
|
|
||||||
{
|
|
||||||
text: "First Question",
|
|
||||||
questionType: QuestionType.SHORT_ANSWER,
|
|
||||||
points: 5,
|
|
||||||
answers: [],
|
|
||||||
matchDistractors: [],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: "Second Question",
|
|
||||||
questionType: QuestionType.ESSAY,
|
|
||||||
points: 10,
|
|
||||||
answers: [],
|
|
||||||
matchDistractors: [],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
|
||||||
|
|
||||||
// Import the service after mocks are set up
|
|
||||||
const { canvasQuizService } = await import("./canvasQuizService");
|
|
||||||
const { axiosClient } = await import("@/services/axiosUtils");
|
|
||||||
const { paginatedRequest } = await import("./canvasServiceUtils");
|
|
||||||
|
|
||||||
// Mock successful quiz creation
|
|
||||||
vi.mocked(axiosClient.post).mockResolvedValueOnce({
|
|
||||||
data: { id: 123, title: "Test Quiz - Order Verification" },
|
|
||||||
});
|
|
||||||
|
|
||||||
// Mock question creation responses
|
|
||||||
vi.mocked(axiosClient.post)
|
|
||||||
.mockResolvedValueOnce({ data: { id: 1, position: 1 } })
|
|
||||||
.mockResolvedValueOnce({ data: { id: 2, position: 2 } });
|
|
||||||
|
|
||||||
// Mock reordering call
|
|
||||||
vi.mocked(axiosClient.post).mockResolvedValueOnce({ data: {} });
|
|
||||||
|
|
||||||
// Mock assignment cleanup (empty assignments)
|
|
||||||
vi.mocked(paginatedRequest).mockResolvedValueOnce([]);
|
|
||||||
|
|
||||||
// Mock the verification call - questions in correct order
|
|
||||||
vi.mocked(paginatedRequest).mockResolvedValueOnce([
|
|
||||||
{
|
|
||||||
id: 1,
|
|
||||||
quiz_id: 123,
|
|
||||||
position: 1,
|
|
||||||
question_name: "Question 1",
|
|
||||||
question_type: "short_answer_question",
|
|
||||||
question_text: "<p>First Question</p>",
|
|
||||||
correct_comments: "",
|
|
||||||
incorrect_comments: "",
|
|
||||||
neutral_comments: "",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 2,
|
|
||||||
quiz_id: 123,
|
|
||||||
position: 2,
|
|
||||||
question_name: "Question 2",
|
|
||||||
question_type: "essay_question",
|
|
||||||
question_text: "<p>Second Question</p>",
|
|
||||||
correct_comments: "",
|
|
||||||
incorrect_comments: "",
|
|
||||||
neutral_comments: "",
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Create the quiz and trigger verification
|
|
||||||
const result = await canvasQuizService.create(12345, testQuiz, {
|
|
||||||
name: "Test Course",
|
|
||||||
canvasId: 12345,
|
|
||||||
assignmentGroups: [],
|
|
||||||
daysOfWeek: [DayOfWeek.Monday, DayOfWeek.Wednesday, DayOfWeek.Friday],
|
|
||||||
startDate: "2023-08-15",
|
|
||||||
endDate: "2023-12-15",
|
|
||||||
defaultDueTime: { hour: 23, minute: 59 },
|
|
||||||
defaultAssignmentSubmissionTypes: [AssignmentSubmissionType.ONLINE_TEXT_ENTRY],
|
|
||||||
defaultFileUploadTypes: [],
|
|
||||||
holidays: [],
|
|
||||||
assets: []
|
|
||||||
});
|
|
||||||
|
|
||||||
// Verify the quiz was created
|
|
||||||
expect(result).toBe(123);
|
|
||||||
|
|
||||||
// Verify that the question verification API call was made
|
|
||||||
expect(vi.mocked(paginatedRequest)).toHaveBeenCalledWith({
|
|
||||||
url: "https://test.instructure.com/api/v1/courses/12345/quizzes/123/questions",
|
|
||||||
});
|
|
||||||
|
|
||||||
// The verification would have run and logged success/failure
|
|
||||||
// In a real scenario, this would catch order mismatches
|
|
||||||
});
|
|
||||||
|
|
||||||
it("demonstrates successful verification workflow", async () => {
|
|
||||||
const { canvasQuizService } = await import("./canvasQuizService");
|
|
||||||
const { paginatedRequest } = await import("./canvasServiceUtils");
|
|
||||||
|
|
||||||
// Mock questions returned from Canvas in correct order
|
|
||||||
vi.mocked(paginatedRequest).mockResolvedValueOnce([
|
|
||||||
{
|
|
||||||
id: 1,
|
|
||||||
quiz_id: 1,
|
|
||||||
position: 1,
|
|
||||||
question_name: "Question 1",
|
|
||||||
question_type: "short_answer_question",
|
|
||||||
question_text: "First question",
|
|
||||||
correct_comments: "",
|
|
||||||
incorrect_comments: "",
|
|
||||||
neutral_comments: "",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 2,
|
|
||||||
quiz_id: 1,
|
|
||||||
position: 2,
|
|
||||||
question_name: "Question 2",
|
|
||||||
question_type: "essay_question",
|
|
||||||
question_text: "Second question",
|
|
||||||
correct_comments: "",
|
|
||||||
incorrect_comments: "",
|
|
||||||
neutral_comments: "",
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
|
|
||||||
const result = await canvasQuizService.getQuizQuestions(1, 1);
|
|
||||||
|
|
||||||
// Verify questions are returned in correct order
|
|
||||||
expect(result).toHaveLength(2);
|
|
||||||
expect(result[0].position).toBe(1);
|
|
||||||
expect(result[1].position).toBe(2);
|
|
||||||
expect(result[0].question_text).toBe("First question");
|
|
||||||
expect(result[1].question_text).toBe("Second question");
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
import { describe, it, expect } from "vitest";
|
||||||
|
import { QuestionType } from "@/features/local/quizzes/models/localQuizQuestion";
|
||||||
|
import { quizMarkdownUtils } from "@/features/local/quizzes/models/utils/quizMarkdownUtils";
|
||||||
|
|
||||||
|
describe("Question Feedback options", () => {
|
||||||
|
it("essay questions can have feedback", () => {
|
||||||
|
const name = "Test Quiz";
|
||||||
|
const rawMarkdownQuiz = `
|
||||||
|
ShuffleAnswers: true
|
||||||
|
OneQuestionAtATime: false
|
||||||
|
DueAt: 08/21/2023 23:59:00
|
||||||
|
LockAt: 08/21/2023 23:59:00
|
||||||
|
AssignmentGroup: Assignments
|
||||||
|
AllowedAttempts: -1
|
||||||
|
Description:
|
||||||
|
---
|
||||||
|
this is the description
|
||||||
|
... this is general feedback
|
||||||
|
essay
|
||||||
|
`;
|
||||||
|
|
||||||
|
const quiz = quizMarkdownUtils.parseMarkdown(rawMarkdownQuiz, name);
|
||||||
|
const firstQuestion = quiz.questions[0];
|
||||||
|
|
||||||
|
expect(firstQuestion.questionType).toBe(QuestionType.ESSAY);
|
||||||
|
expect(firstQuestion.text).not.toContain("this is general feedback");
|
||||||
|
expect(firstQuestion.neutralComments).toBe("this is general feedback");
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -5,7 +5,6 @@ import { QuestionType } from "@/features/local/quizzes/models/localQuizQuestion"
|
|||||||
import { quizMarkdownUtils } from "@/features/local/quizzes/models/utils/quizMarkdownUtils";
|
import { quizMarkdownUtils } from "@/features/local/quizzes/models/utils/quizMarkdownUtils";
|
||||||
import { quizQuestionMarkdownUtils } from "@/features/local/quizzes/models/utils/quizQuestionMarkdownUtils";
|
import { quizQuestionMarkdownUtils } from "@/features/local/quizzes/models/utils/quizQuestionMarkdownUtils";
|
||||||
|
|
||||||
// Test suite for QuizMarkdown
|
|
||||||
describe("QuizMarkdownTests", () => {
|
describe("QuizMarkdownTests", () => {
|
||||||
it("can serialize quiz to markdown", () => {
|
it("can serialize quiz to markdown", () => {
|
||||||
const quiz: LocalQuiz = {
|
const quiz: LocalQuiz = {
|
||||||
|
|||||||
@@ -0,0 +1,125 @@
|
|||||||
|
type FeedbackType = "+" | "-" | "...";
|
||||||
|
|
||||||
|
const isFeedbackStart = (
|
||||||
|
trimmedLine: string,
|
||||||
|
feedbackType: FeedbackType
|
||||||
|
): boolean => {
|
||||||
|
const prefix = feedbackType === "..." ? "... " : `${feedbackType} `;
|
||||||
|
return trimmedLine.startsWith(prefix) || trimmedLine === feedbackType;
|
||||||
|
};
|
||||||
|
|
||||||
|
const extractFeedbackContent = (
|
||||||
|
trimmedLine: string,
|
||||||
|
feedbackType: FeedbackType
|
||||||
|
): string => {
|
||||||
|
if (trimmedLine === feedbackType) return "";
|
||||||
|
|
||||||
|
const prefixLength = feedbackType === "..." ? 4 : 2; // "... " is 4 chars, "+ " and "- " are 2
|
||||||
|
return trimmedLine.substring(prefixLength);
|
||||||
|
};
|
||||||
|
|
||||||
|
const saveFeedback = (
|
||||||
|
feedbackType: FeedbackType | null,
|
||||||
|
feedbackLines: string[],
|
||||||
|
comments: {
|
||||||
|
correct?: string;
|
||||||
|
incorrect?: string;
|
||||||
|
neutral?: string;
|
||||||
|
}
|
||||||
|
): void => {
|
||||||
|
if (!feedbackType || feedbackLines.length === 0) return;
|
||||||
|
|
||||||
|
const feedbackText = feedbackLines.join("\n");
|
||||||
|
if (feedbackType === "+") {
|
||||||
|
comments.correct = feedbackText;
|
||||||
|
} else if (feedbackType === "-") {
|
||||||
|
comments.incorrect = feedbackText;
|
||||||
|
} else if (feedbackType === "...") {
|
||||||
|
comments.neutral = feedbackText;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const quizFeedbackMarkdownUtils = {
|
||||||
|
extractFeedback(
|
||||||
|
linesWithoutPoints: string[],
|
||||||
|
isAnswerLine: (trimmedLine: string) => boolean
|
||||||
|
): {
|
||||||
|
correctComments?: string;
|
||||||
|
incorrectComments?: string;
|
||||||
|
neutralComments?: string;
|
||||||
|
linesWithoutFeedback: string[];
|
||||||
|
} {
|
||||||
|
const comments: {
|
||||||
|
correct?: string;
|
||||||
|
incorrect?: string;
|
||||||
|
neutral?: string;
|
||||||
|
} = {};
|
||||||
|
const linesWithoutFeedback: string[] = [];
|
||||||
|
|
||||||
|
let currentFeedbackType: FeedbackType | null = null;
|
||||||
|
let currentFeedbackLines: string[] = [];
|
||||||
|
|
||||||
|
for (const line of linesWithoutPoints) {
|
||||||
|
const trimmed = line.trim();
|
||||||
|
|
||||||
|
// Check if this is a new feedback line
|
||||||
|
let newFeedbackType: FeedbackType | null = null;
|
||||||
|
if (isFeedbackStart(trimmed, "+")) {
|
||||||
|
newFeedbackType = "+";
|
||||||
|
} else if (isFeedbackStart(trimmed, "-")) {
|
||||||
|
newFeedbackType = "-";
|
||||||
|
} else if (isFeedbackStart(trimmed, "...")) {
|
||||||
|
newFeedbackType = "...";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newFeedbackType) {
|
||||||
|
// Save previous feedback if any
|
||||||
|
saveFeedback(currentFeedbackType, currentFeedbackLines, comments);
|
||||||
|
|
||||||
|
// Start new feedback
|
||||||
|
currentFeedbackType = newFeedbackType;
|
||||||
|
const content = extractFeedbackContent(trimmed, newFeedbackType);
|
||||||
|
currentFeedbackLines = content ? [content] : [];
|
||||||
|
} else if (currentFeedbackType && !isAnswerLine(trimmed)) {
|
||||||
|
// This is a continuation of the current feedback
|
||||||
|
currentFeedbackLines.push(line);
|
||||||
|
} else {
|
||||||
|
// Save any pending feedback
|
||||||
|
saveFeedback(currentFeedbackType, currentFeedbackLines, comments);
|
||||||
|
currentFeedbackType = null;
|
||||||
|
currentFeedbackLines = [];
|
||||||
|
|
||||||
|
// This is a regular line
|
||||||
|
linesWithoutFeedback.push(line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save any remaining feedback
|
||||||
|
saveFeedback(currentFeedbackType, currentFeedbackLines, comments);
|
||||||
|
|
||||||
|
return {
|
||||||
|
correctComments: comments.correct,
|
||||||
|
incorrectComments: comments.incorrect,
|
||||||
|
neutralComments: comments.neutral,
|
||||||
|
linesWithoutFeedback,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
formatFeedback(
|
||||||
|
correctComments?: string,
|
||||||
|
incorrectComments?: string,
|
||||||
|
neutralComments?: string
|
||||||
|
): string {
|
||||||
|
let feedbackText = "";
|
||||||
|
if (correctComments) {
|
||||||
|
feedbackText += `+ ${correctComments}\n`;
|
||||||
|
}
|
||||||
|
if (incorrectComments) {
|
||||||
|
feedbackText += `- ${incorrectComments}\n`;
|
||||||
|
}
|
||||||
|
if (neutralComments) {
|
||||||
|
feedbackText += `... ${neutralComments}\n`;
|
||||||
|
}
|
||||||
|
return feedbackText;
|
||||||
|
},
|
||||||
|
};
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import { LocalQuizQuestion, QuestionType } from "../localQuizQuestion";
|
import { LocalQuizQuestion, QuestionType } from "../localQuizQuestion";
|
||||||
import { LocalQuizQuestionAnswer } from "../localQuizQuestionAnswer";
|
import { LocalQuizQuestionAnswer } from "../localQuizQuestionAnswer";
|
||||||
import { quizQuestionAnswerMarkdownUtils } from "./quizQuestionAnswerMarkdownUtils";
|
import { quizQuestionAnswerMarkdownUtils } from "./quizQuestionAnswerMarkdownUtils";
|
||||||
|
import { quizFeedbackMarkdownUtils } from "./quizFeedbackMarkdownUtils";
|
||||||
|
|
||||||
const _validFirstAnswerDelimiters = [
|
const _validFirstAnswerDelimiters = [
|
||||||
"*a)",
|
"*a)",
|
||||||
@@ -14,97 +15,11 @@ const _validFirstAnswerDelimiters = [
|
|||||||
];
|
];
|
||||||
const _multipleChoicePrefix = ["a)", "*a)", "*)", ")"];
|
const _multipleChoicePrefix = ["a)", "*a)", "*)", ")"];
|
||||||
const _multipleAnswerPrefix = ["[ ]", "[*]", "[]"];
|
const _multipleAnswerPrefix = ["[ ]", "[*]", "[]"];
|
||||||
const _feedbackPrefixes = ["+", "-", "..."];
|
|
||||||
|
|
||||||
const extractFeedback = (
|
const isAnswerLine = (trimmedLine: string): boolean => {
|
||||||
linesWithoutPoints: string[]
|
return _validFirstAnswerDelimiters.some((prefix) =>
|
||||||
): {
|
trimmedLine.startsWith(prefix)
|
||||||
correctComments?: string;
|
);
|
||||||
incorrectComments?: string;
|
|
||||||
neutralComments?: string;
|
|
||||||
linesWithoutFeedback: string[];
|
|
||||||
} => {
|
|
||||||
let correctComments: string | undefined;
|
|
||||||
let incorrectComments: string | undefined;
|
|
||||||
let neutralComments: string | undefined;
|
|
||||||
const linesWithoutFeedback: string[] = [];
|
|
||||||
|
|
||||||
let currentFeedbackType: "+" | "-" | "..." | null = null;
|
|
||||||
let currentFeedbackLines: string[] = [];
|
|
||||||
|
|
||||||
for (const line of linesWithoutPoints) {
|
|
||||||
const trimmed = line.trim();
|
|
||||||
|
|
||||||
// Check if this is a new feedback line
|
|
||||||
if (trimmed.startsWith("+ ") || trimmed === "+") {
|
|
||||||
// Save previous feedback if any
|
|
||||||
if (currentFeedbackType && currentFeedbackLines.length > 0) {
|
|
||||||
const feedbackText = currentFeedbackLines.join("\n");
|
|
||||||
if (currentFeedbackType === "+") correctComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "-") incorrectComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "...") neutralComments = feedbackText;
|
|
||||||
}
|
|
||||||
|
|
||||||
currentFeedbackType = "+";
|
|
||||||
currentFeedbackLines = trimmed === "+" ? [] : [trimmed.substring(2)]; // Remove "+ " or handle standalone "+"
|
|
||||||
} else if (trimmed.startsWith("- ") || trimmed === "-") {
|
|
||||||
// Save previous feedback if any
|
|
||||||
if (currentFeedbackType && currentFeedbackLines.length > 0) {
|
|
||||||
const feedbackText = currentFeedbackLines.join("\n");
|
|
||||||
if (currentFeedbackType === "+") correctComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "-") incorrectComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "...") neutralComments = feedbackText;
|
|
||||||
}
|
|
||||||
|
|
||||||
currentFeedbackType = "-";
|
|
||||||
currentFeedbackLines = trimmed === "-" ? [] : [trimmed.substring(2)]; // Remove "- " or handle standalone "-"
|
|
||||||
} else if (trimmed.startsWith("... ") || trimmed === "...") {
|
|
||||||
// Save previous feedback if any
|
|
||||||
if (currentFeedbackType && currentFeedbackLines.length > 0) {
|
|
||||||
const feedbackText = currentFeedbackLines.join("\n");
|
|
||||||
if (currentFeedbackType === "+") correctComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "-") incorrectComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "...") neutralComments = feedbackText;
|
|
||||||
}
|
|
||||||
|
|
||||||
currentFeedbackType = "...";
|
|
||||||
currentFeedbackLines = trimmed === "..." ? [] : [trimmed.substring(4)]; // Remove "... " or handle standalone "..."
|
|
||||||
} else if (
|
|
||||||
currentFeedbackType &&
|
|
||||||
!_validFirstAnswerDelimiters.some((prefix) => trimmed.startsWith(prefix))
|
|
||||||
) {
|
|
||||||
// This is a continuation of the current feedback
|
|
||||||
currentFeedbackLines.push(line);
|
|
||||||
} else {
|
|
||||||
// Save any pending feedback
|
|
||||||
if (currentFeedbackType && currentFeedbackLines.length > 0) {
|
|
||||||
const feedbackText = currentFeedbackLines.join("\n");
|
|
||||||
if (currentFeedbackType === "+") correctComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "-") incorrectComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "...") neutralComments = feedbackText;
|
|
||||||
currentFeedbackType = null;
|
|
||||||
currentFeedbackLines = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
// This is a regular line
|
|
||||||
linesWithoutFeedback.push(line);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save any remaining feedback
|
|
||||||
if (currentFeedbackType && currentFeedbackLines.length > 0) {
|
|
||||||
const feedbackText = currentFeedbackLines.join("\n");
|
|
||||||
if (currentFeedbackType === "+") correctComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "-") incorrectComments = feedbackText;
|
|
||||||
else if (currentFeedbackType === "...") neutralComments = feedbackText;
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
correctComments,
|
|
||||||
incorrectComments,
|
|
||||||
neutralComments,
|
|
||||||
linesWithoutFeedback,
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const getAnswerStringsWithMultilineSupport = (
|
const getAnswerStringsWithMultilineSupport = (
|
||||||
@@ -246,16 +161,11 @@ export const quizQuestionMarkdownUtils = {
|
|||||||
: "";
|
: "";
|
||||||
|
|
||||||
// Build feedback lines
|
// Build feedback lines
|
||||||
let feedbackText = "";
|
const feedbackText = quizFeedbackMarkdownUtils.formatFeedback(
|
||||||
if (question.correctComments) {
|
question.correctComments,
|
||||||
feedbackText += `+ ${question.correctComments}\n`;
|
question.incorrectComments,
|
||||||
}
|
question.neutralComments
|
||||||
if (question.incorrectComments) {
|
);
|
||||||
feedbackText += `- ${question.incorrectComments}\n`;
|
|
||||||
}
|
|
||||||
if (question.neutralComments) {
|
|
||||||
feedbackText += `... ${question.neutralComments}\n`;
|
|
||||||
}
|
|
||||||
|
|
||||||
const answersText = answerArray.join("\n");
|
const answersText = answerArray.join("\n");
|
||||||
const questionTypeIndicator =
|
const questionTypeIndicator =
|
||||||
@@ -292,7 +202,10 @@ export const quizQuestionMarkdownUtils = {
|
|||||||
incorrectComments,
|
incorrectComments,
|
||||||
neutralComments,
|
neutralComments,
|
||||||
linesWithoutFeedback,
|
linesWithoutFeedback,
|
||||||
} = extractFeedback(linesWithoutPoints);
|
} = quizFeedbackMarkdownUtils.extractFeedback(
|
||||||
|
linesWithoutPoints,
|
||||||
|
isAnswerLine
|
||||||
|
);
|
||||||
|
|
||||||
const { linesWithoutAnswers } = linesWithoutFeedback.reduce(
|
const { linesWithoutAnswers } = linesWithoutFeedback.reduce(
|
||||||
({ linesWithoutAnswers, taking }, currentLine) => {
|
({ linesWithoutAnswers, taking }, currentLine) => {
|
||||||
|
|||||||
Reference in New Issue
Block a user