@schoolio/player 1.4.3 → 1.4.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.mjs CHANGED
@@ -103,6 +103,13 @@ var QuizApiClient = class {
103
103
  }
104
104
  return response.blob();
105
105
  }
106
+ async logError(params) {
107
+ try {
108
+ await this.request("POST", "/api/external/log-error", params);
109
+ } catch (e) {
110
+ console.warn("[QuizEngine] Failed to log error:", e);
111
+ }
112
+ }
106
113
  };
107
114
 
108
115
  // src/utils.ts
@@ -158,15 +165,16 @@ function checkAnswer(question, selectedAnswer) {
158
165
  return { isCorrect, pointsEarned: isCorrect ? points : 0 };
159
166
  }
160
167
  case "essay":
161
- case "assessment":
162
168
  return { isCorrect: false, pointsEarned: 0 };
169
+ case "assessment":
170
+ return { isCorrect: true, pointsEarned: points };
163
171
  default:
164
172
  return { isCorrect: false, pointsEarned: 0 };
165
173
  }
166
174
  }
167
175
  function createAnswerDetail(question, selectedAnswer) {
168
176
  const { isCorrect, pointsEarned } = checkAnswer(question, selectedAnswer);
169
- return {
177
+ const detail = {
170
178
  questionId: question.id,
171
179
  questionText: question.question,
172
180
  questionType: question.type,
@@ -178,6 +186,14 @@ function createAnswerDetail(question, selectedAnswer) {
178
186
  explanation: question.explanation,
179
187
  hint: question.hint
180
188
  };
189
+ if (question.type === "sorting") {
190
+ detail.items = question.items;
191
+ detail.correctOrder = question.correctOrder;
192
+ } else if (question.type === "matrix") {
193
+ detail.leftItems = question.leftItems;
194
+ detail.rightItems = question.rightItems;
195
+ }
196
+ return detail;
181
197
  }
182
198
  function calculateScore(answers) {
183
199
  const totalPoints = answers.reduce((sum, a) => sum + a.points, 0);
@@ -1014,8 +1030,191 @@ function QuestionChatPanel({
1014
1030
  ] });
1015
1031
  }
1016
1032
 
1033
+ // src/errors.ts
1034
+ var ERROR_DEFINITIONS = {
1035
+ QUIZ_NOT_FOUND: {
1036
+ code: "QUIZ_NOT_FOUND",
1037
+ userMessage: "We couldn't find this quiz",
1038
+ subMessage: "The quiz may have been removed or the link is incorrect.",
1039
+ cause: "The quiz ID does not exist in the database, or the quiz has been deleted.",
1040
+ isBlocking: true
1041
+ },
1042
+ ATTEMPT_NOT_FOUND: {
1043
+ code: "ATTEMPT_NOT_FOUND",
1044
+ userMessage: "We couldn't find this quiz attempt",
1045
+ subMessage: "The attempt may have expired or the link is incorrect.",
1046
+ cause: "The attempt ID does not exist, or the attempt has been deleted/archived.",
1047
+ isBlocking: true
1048
+ },
1049
+ QUIZ_EXPIRED: {
1050
+ code: "QUIZ_EXPIRED",
1051
+ userMessage: "This quiz has expired",
1052
+ subMessage: "The deadline for this quiz has passed.",
1053
+ cause: "The quiz end date/time has passed and submissions are no longer accepted.",
1054
+ isBlocking: true
1055
+ },
1056
+ QUIZ_NOT_STARTED: {
1057
+ code: "QUIZ_NOT_STARTED",
1058
+ userMessage: "This quiz is not available yet",
1059
+ subMessage: "Please check back when the quiz opens.",
1060
+ cause: "The quiz start date/time has not yet been reached.",
1061
+ isBlocking: true
1062
+ },
1063
+ NETWORK_ERROR: {
1064
+ code: "NETWORK_ERROR",
1065
+ userMessage: "Connection problem",
1066
+ subMessage: "Please check your internet connection and try again.",
1067
+ cause: "Unable to reach the server due to network connectivity issues.",
1068
+ isBlocking: true
1069
+ },
1070
+ SERVER_ERROR: {
1071
+ code: "SERVER_ERROR",
1072
+ userMessage: "Something went wrong on our end",
1073
+ subMessage: "Our team has been notified. Please try again later.",
1074
+ cause: "The server encountered an internal error (HTTP 500+).",
1075
+ isBlocking: true
1076
+ },
1077
+ UNAUTHORIZED: {
1078
+ code: "UNAUTHORIZED",
1079
+ userMessage: "Please sign in to continue",
1080
+ subMessage: "You need to be logged in to access this quiz.",
1081
+ cause: "The user is not authenticated (HTTP 401).",
1082
+ isBlocking: true
1083
+ },
1084
+ FORBIDDEN: {
1085
+ code: "FORBIDDEN",
1086
+ userMessage: "You don't have access to this quiz",
1087
+ subMessage: "Contact your instructor if you believe this is a mistake.",
1088
+ cause: "The user does not have permission to access this resource (HTTP 403).",
1089
+ isBlocking: true
1090
+ },
1091
+ TTS_FAILED: {
1092
+ code: "TTS_FAILED",
1093
+ userMessage: "Text-to-speech unavailable",
1094
+ subMessage: "Audio features are temporarily unavailable.",
1095
+ cause: "The text-to-speech service failed or is unreachable.",
1096
+ isBlocking: false
1097
+ },
1098
+ CHAT_FAILED: {
1099
+ code: "CHAT_FAILED",
1100
+ userMessage: "Chat assistance unavailable",
1101
+ subMessage: "The AI helper is temporarily unavailable.",
1102
+ cause: "The chat/AI service failed to initialize or respond.",
1103
+ isBlocking: false
1104
+ },
1105
+ SUBMISSION_FAILED: {
1106
+ code: "SUBMISSION_FAILED",
1107
+ userMessage: "Failed to submit your answer",
1108
+ subMessage: "Please try again. Your progress has been saved.",
1109
+ cause: "The answer submission request failed due to network or server issues.",
1110
+ isBlocking: false
1111
+ },
1112
+ UNKNOWN_ERROR: {
1113
+ code: "UNKNOWN_ERROR",
1114
+ userMessage: "Something unexpected happened",
1115
+ subMessage: "Please try refreshing the page.",
1116
+ cause: "An unclassified error occurred.",
1117
+ isBlocking: true
1118
+ }
1119
+ };
1120
+ function getErrorFromHttpStatus(status, context) {
1121
+ switch (status) {
1122
+ case 401:
1123
+ return "UNAUTHORIZED";
1124
+ case 403:
1125
+ return "FORBIDDEN";
1126
+ case 404:
1127
+ return context === "attempt" ? "ATTEMPT_NOT_FOUND" : "QUIZ_NOT_FOUND";
1128
+ case 410:
1129
+ return "QUIZ_EXPIRED";
1130
+ case 500:
1131
+ case 502:
1132
+ case 503:
1133
+ case 504:
1134
+ return "SERVER_ERROR";
1135
+ default:
1136
+ if (status >= 400 && status < 500) {
1137
+ return context === "attempt" ? "ATTEMPT_NOT_FOUND" : "QUIZ_NOT_FOUND";
1138
+ }
1139
+ return "UNKNOWN_ERROR";
1140
+ }
1141
+ }
1142
+ function getErrorFromMessage(message, context) {
1143
+ const lowerMessage = message.toLowerCase();
1144
+ if (lowerMessage.includes("network") || lowerMessage.includes("fetch") || lowerMessage.includes("connection")) {
1145
+ return "NETWORK_ERROR";
1146
+ }
1147
+ if (lowerMessage.includes("not found") || lowerMessage.includes("404")) {
1148
+ return context === "attempt" ? "ATTEMPT_NOT_FOUND" : "QUIZ_NOT_FOUND";
1149
+ }
1150
+ if (lowerMessage.includes("unauthorized") || lowerMessage.includes("401")) {
1151
+ return "UNAUTHORIZED";
1152
+ }
1153
+ if (lowerMessage.includes("forbidden") || lowerMessage.includes("403")) {
1154
+ return "FORBIDDEN";
1155
+ }
1156
+ if (lowerMessage.includes("expired")) {
1157
+ return "QUIZ_EXPIRED";
1158
+ }
1159
+ if (lowerMessage.includes("tts") || lowerMessage.includes("speech")) {
1160
+ return "TTS_FAILED";
1161
+ }
1162
+ if (lowerMessage.includes("chat")) {
1163
+ return "CHAT_FAILED";
1164
+ }
1165
+ return "UNKNOWN_ERROR";
1166
+ }
1167
+
1168
+ // src/MaintenanceScreen.tsx
1169
+ import { jsx as jsx3, jsxs as jsxs3 } from "react/jsx-runtime";
1170
+ function MaintenanceScreen({ errorCode }) {
1171
+ const errorDef = errorCode ? ERROR_DEFINITIONS[errorCode] : null;
1172
+ const message = errorDef?.userMessage || "Your quiz is on its way...";
1173
+ const subMessage = errorDef?.subMessage || "Please check back soon!";
1174
+ const containerStyle = {
1175
+ display: "flex",
1176
+ flexDirection: "column",
1177
+ alignItems: "center",
1178
+ justifyContent: "center",
1179
+ minHeight: "300px",
1180
+ padding: "40px",
1181
+ background: "linear-gradient(135deg, #f8f7ff 0%, #e8e4f8 50%, #f0ebff 100%)",
1182
+ borderRadius: "16px"
1183
+ };
1184
+ const iconStyle = {
1185
+ fontSize: "48px",
1186
+ marginBottom: "24px",
1187
+ color: "#8b5cf6"
1188
+ };
1189
+ const messageStyle = {
1190
+ fontSize: "20px",
1191
+ fontWeight: "600",
1192
+ color: "#4c1d95",
1193
+ textAlign: "center",
1194
+ marginBottom: "12px"
1195
+ };
1196
+ const submessageStyle = {
1197
+ fontSize: "14px",
1198
+ color: "#7c3aed",
1199
+ textAlign: "center",
1200
+ opacity: 0.8
1201
+ };
1202
+ return /* @__PURE__ */ jsxs3("div", { style: containerStyle, "data-testid": "maintenance-screen", "data-error-code": errorCode || "none", children: [
1203
+ /* @__PURE__ */ jsx3("div", { style: iconStyle, children: /* @__PURE__ */ jsxs3("svg", { width: "64", height: "64", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1204
+ /* @__PURE__ */ jsx3("circle", { cx: "12", cy: "12", r: "10" }),
1205
+ /* @__PURE__ */ jsx3("path", { d: "M12 8v4" }),
1206
+ /* @__PURE__ */ jsx3("path", { d: "M12 16h.01" })
1207
+ ] }) }),
1208
+ /* @__PURE__ */ jsx3("div", { style: messageStyle, "data-testid": "text-error-message", children: message }),
1209
+ /* @__PURE__ */ jsx3("div", { style: submessageStyle, "data-testid": "text-error-submessage", children: subMessage })
1210
+ ] });
1211
+ }
1212
+
1213
+ // src/assets/astronautData.ts
1214
+ var astronautImage = "";
1215
+
1017
1216
  // src/QuizPlayer.tsx
1018
- import { Fragment as Fragment2, jsx as jsx3, jsxs as jsxs3 } from "react/jsx-runtime";
1217
+ import { Fragment as Fragment2, jsx as jsx4, jsxs as jsxs4 } from "react/jsx-runtime";
1019
1218
  var defaultStyles = {
1020
1219
  container: {
1021
1220
  fontFamily: "system-ui, -apple-system, sans-serif",
@@ -1316,6 +1515,13 @@ var defaultStyles = {
1316
1515
  backgroundColor: "#fef2f2",
1317
1516
  borderColor: "#ef4444"
1318
1517
  },
1518
+ feedbackNeutral: {
1519
+ backgroundColor: "#f0f9ff",
1520
+ borderColor: "#0ea5e9"
1521
+ },
1522
+ feedbackTitleNeutral: {
1523
+ color: "#0369a1"
1524
+ },
1319
1525
  feedbackTitle: {
1320
1526
  fontSize: "16px",
1321
1527
  fontWeight: "600",
@@ -1336,8 +1542,292 @@ var defaultStyles = {
1336
1542
  lineHeight: "1.5"
1337
1543
  }
1338
1544
  };
1545
+ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOrderChange }) {
1546
+ const [draggedIndex, setDraggedIndex] = useState3(null);
1547
+ const [dragOverIndex, setDragOverIndex] = useState3(null);
1548
+ const handleDragStart = (e, position) => {
1549
+ if (showFeedback) return;
1550
+ setDraggedIndex(position);
1551
+ e.dataTransfer.effectAllowed = "move";
1552
+ e.dataTransfer.setData("text/plain", position.toString());
1553
+ };
1554
+ const handleDragOver = (e, position) => {
1555
+ e.preventDefault();
1556
+ if (showFeedback) return;
1557
+ e.dataTransfer.dropEffect = "move";
1558
+ setDragOverIndex(position);
1559
+ };
1560
+ const handleDragLeave = () => {
1561
+ setDragOverIndex(null);
1562
+ };
1563
+ const handleDrop = (e, toPosition) => {
1564
+ e.preventDefault();
1565
+ if (showFeedback) return;
1566
+ const fromPosition = parseInt(e.dataTransfer.getData("text/plain"), 10);
1567
+ if (fromPosition !== toPosition) {
1568
+ const newOrder = [...currentOrder];
1569
+ const [movedItem] = newOrder.splice(fromPosition, 1);
1570
+ newOrder.splice(toPosition, 0, movedItem);
1571
+ onOrderChange(newOrder);
1572
+ }
1573
+ setDraggedIndex(null);
1574
+ setDragOverIndex(null);
1575
+ };
1576
+ const handleDragEnd = () => {
1577
+ setDraggedIndex(null);
1578
+ setDragOverIndex(null);
1579
+ };
1580
+ return /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentOrder.map((itemIndex, position) => {
1581
+ const isCorrectPosition = correctOrder?.[position] === itemIndex;
1582
+ const isDragging = draggedIndex === position;
1583
+ const isDragOver = dragOverIndex === position;
1584
+ let itemStyle = {
1585
+ ...defaultStyles.option,
1586
+ display: "flex",
1587
+ alignItems: "center",
1588
+ gap: "12px",
1589
+ cursor: showFeedback ? "default" : "grab",
1590
+ opacity: isDragging ? 0.5 : 1,
1591
+ transition: "all 0.2s ease",
1592
+ transform: isDragOver && !showFeedback ? "scale(1.02)" : "scale(1)"
1593
+ };
1594
+ if (showFeedback) {
1595
+ if (isCorrectPosition) {
1596
+ itemStyle = { ...itemStyle, ...defaultStyles.optionCorrect };
1597
+ } else {
1598
+ itemStyle = { ...itemStyle, ...defaultStyles.optionIncorrect };
1599
+ }
1600
+ } else if (isDragOver) {
1601
+ itemStyle = { ...itemStyle, borderColor: "#6366f1", backgroundColor: "#eef2ff" };
1602
+ }
1603
+ return /* @__PURE__ */ jsxs4(
1604
+ "div",
1605
+ {
1606
+ style: itemStyle,
1607
+ "data-testid": `sorting-item-${position}`,
1608
+ draggable: !showFeedback,
1609
+ onDragStart: (e) => handleDragStart(e, position),
1610
+ onDragOver: (e) => handleDragOver(e, position),
1611
+ onDragLeave: handleDragLeave,
1612
+ onDrop: (e) => handleDrop(e, position),
1613
+ onDragEnd: handleDragEnd,
1614
+ children: [
1615
+ !showFeedback && /* @__PURE__ */ jsx4("div", { style: {
1616
+ cursor: "grab",
1617
+ padding: "4px",
1618
+ display: "flex",
1619
+ alignItems: "center",
1620
+ color: "#9ca3af"
1621
+ }, children: /* @__PURE__ */ jsxs4("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1622
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "5", r: "1" }),
1623
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "12", r: "1" }),
1624
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "19", r: "1" }),
1625
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "5", r: "1" }),
1626
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "12", r: "1" }),
1627
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "19", r: "1" })
1628
+ ] }) }),
1629
+ showFeedback && /* @__PURE__ */ jsx4("div", { style: {
1630
+ display: "flex",
1631
+ alignItems: "center",
1632
+ color: isCorrectPosition ? "#22c55e" : "#ef4444"
1633
+ }, children: isCorrectPosition ? /* @__PURE__ */ jsxs4("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1634
+ /* @__PURE__ */ jsx4("path", { d: "M22 11.08V12a10 10 0 1 1-5.93-9.14" }),
1635
+ /* @__PURE__ */ jsx4("polyline", { points: "22 4 12 14.01 9 11.01" })
1636
+ ] }) : /* @__PURE__ */ jsxs4("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1637
+ /* @__PURE__ */ jsx4("circle", { cx: "12", cy: "12", r: "10" }),
1638
+ /* @__PURE__ */ jsx4("line", { x1: "15", y1: "9", x2: "9", y2: "15" }),
1639
+ /* @__PURE__ */ jsx4("line", { x1: "9", y1: "9", x2: "15", y2: "15" })
1640
+ ] }) }),
1641
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: items[itemIndex] }),
1642
+ /* @__PURE__ */ jsx4("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx4(TextToSpeech, { text: items[itemIndex], size: "sm" }) })
1643
+ ]
1644
+ },
1645
+ itemIndex
1646
+ );
1647
+ }) });
1648
+ }
1649
+ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatches, showFeedback, onMatchChange }) {
1650
+ const [draggedItem, setDraggedItem] = useState3(null);
1651
+ const [dragOverLeft, setDragOverLeft] = useState3(null);
1652
+ const matchedRightItems = Object.values(currentMatches);
1653
+ const unmatchedRightItems = rightItems.filter((item) => !matchedRightItems.includes(item));
1654
+ const handleDragStart = (e, rightItem) => {
1655
+ if (showFeedback) return;
1656
+ setDraggedItem(rightItem);
1657
+ e.dataTransfer.effectAllowed = "move";
1658
+ e.dataTransfer.setData("text/plain", rightItem);
1659
+ };
1660
+ const handleDragOver = (e, leftItem) => {
1661
+ e.preventDefault();
1662
+ if (showFeedback) return;
1663
+ e.dataTransfer.dropEffect = "move";
1664
+ setDragOverLeft(leftItem);
1665
+ };
1666
+ const handleDragLeave = () => {
1667
+ setDragOverLeft(null);
1668
+ };
1669
+ const handleDrop = (e, leftItem) => {
1670
+ e.preventDefault();
1671
+ if (showFeedback) return;
1672
+ const rightItem = e.dataTransfer.getData("text/plain");
1673
+ if (rightItem) {
1674
+ const newMatches = { ...currentMatches };
1675
+ Object.keys(newMatches).forEach((key) => {
1676
+ if (newMatches[key] === rightItem) {
1677
+ delete newMatches[key];
1678
+ }
1679
+ });
1680
+ newMatches[leftItem] = rightItem;
1681
+ onMatchChange(newMatches);
1682
+ }
1683
+ setDraggedItem(null);
1684
+ setDragOverLeft(null);
1685
+ };
1686
+ const handleDragEnd = () => {
1687
+ setDraggedItem(null);
1688
+ setDragOverLeft(null);
1689
+ };
1690
+ const handleClearMatch = (leftItem) => {
1691
+ if (showFeedback) return;
1692
+ const newMatches = { ...currentMatches };
1693
+ delete newMatches[leftItem];
1694
+ onMatchChange(newMatches);
1695
+ };
1696
+ return /* @__PURE__ */ jsxs4("div", { style: { display: "flex", gap: "24px", flexWrap: "wrap" }, children: [
1697
+ /* @__PURE__ */ jsxs4("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1698
+ /* @__PURE__ */ jsx4("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Match these items:" }),
1699
+ leftItems.map((leftItem, idx) => {
1700
+ const matchedRight = currentMatches[leftItem];
1701
+ const correctMatch = correctMatches?.[leftItem];
1702
+ const isCorrect = matchedRight === correctMatch;
1703
+ const isDragOver = dragOverLeft === leftItem;
1704
+ let rowStyle = {
1705
+ display: "flex",
1706
+ alignItems: "center",
1707
+ gap: "12px",
1708
+ padding: "12px 16px",
1709
+ border: "2px dashed #e5e7eb",
1710
+ borderRadius: "8px",
1711
+ backgroundColor: "#ffffff",
1712
+ minHeight: "56px",
1713
+ transition: "all 0.2s ease"
1714
+ };
1715
+ if (showFeedback) {
1716
+ rowStyle.borderStyle = "solid";
1717
+ if (isCorrect) {
1718
+ rowStyle = { ...rowStyle, borderColor: "#22c55e", backgroundColor: "#f0fdf4" };
1719
+ } else {
1720
+ rowStyle = { ...rowStyle, borderColor: "#ef4444", backgroundColor: "#fef2f2" };
1721
+ }
1722
+ } else if (isDragOver) {
1723
+ rowStyle = { ...rowStyle, borderColor: "#6366f1", backgroundColor: "#eef2ff", borderStyle: "solid" };
1724
+ } else if (matchedRight) {
1725
+ rowStyle = { ...rowStyle, borderStyle: "solid", borderColor: "#22c55e" };
1726
+ }
1727
+ return /* @__PURE__ */ jsxs4(
1728
+ "div",
1729
+ {
1730
+ style: rowStyle,
1731
+ "data-testid": `matrix-row-${idx}`,
1732
+ onDragOver: (e) => handleDragOver(e, leftItem),
1733
+ onDragLeave: handleDragLeave,
1734
+ onDrop: (e) => handleDrop(e, leftItem),
1735
+ children: [
1736
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1, fontWeight: "500" }, children: leftItem }),
1737
+ /* @__PURE__ */ jsx4("span", { style: { color: "#6b7280" }, children: "\u2192" }),
1738
+ matchedRight ? /* @__PURE__ */ jsxs4("div", { style: {
1739
+ display: "flex",
1740
+ alignItems: "center",
1741
+ gap: "8px",
1742
+ padding: "6px 12px",
1743
+ backgroundColor: showFeedback ? isCorrect ? "#dcfce7" : "#fee2e2" : "#e0e7ff",
1744
+ borderRadius: "6px",
1745
+ fontSize: "14px"
1746
+ }, children: [
1747
+ /* @__PURE__ */ jsx4("span", { children: matchedRight }),
1748
+ !showFeedback && /* @__PURE__ */ jsx4(
1749
+ "button",
1750
+ {
1751
+ onClick: () => handleClearMatch(leftItem),
1752
+ style: {
1753
+ background: "none",
1754
+ border: "none",
1755
+ cursor: "pointer",
1756
+ padding: "2px",
1757
+ display: "flex",
1758
+ color: "#6b7280"
1759
+ },
1760
+ "aria-label": "Remove match",
1761
+ children: /* @__PURE__ */ jsxs4("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1762
+ /* @__PURE__ */ jsx4("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1763
+ /* @__PURE__ */ jsx4("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1764
+ ] })
1765
+ }
1766
+ ),
1767
+ showFeedback && (isCorrect ? /* @__PURE__ */ jsx4("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "#22c55e", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ jsx4("polyline", { points: "20 6 9 17 4 12" }) }) : /* @__PURE__ */ jsxs4("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "#ef4444", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1768
+ /* @__PURE__ */ jsx4("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1769
+ /* @__PURE__ */ jsx4("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1770
+ ] }))
1771
+ ] }) : /* @__PURE__ */ jsx4("span", { style: { color: "#9ca3af", fontSize: "14px", fontStyle: "italic" }, children: "Drop here" }),
1772
+ showFeedback && !isCorrect && correctMatch && /* @__PURE__ */ jsxs4("span", { style: { color: "#166534", fontSize: "13px", marginLeft: "8px" }, children: [
1773
+ "(Correct: ",
1774
+ correctMatch,
1775
+ ")"
1776
+ ] })
1777
+ ]
1778
+ },
1779
+ idx
1780
+ );
1781
+ })
1782
+ ] }),
1783
+ /* @__PURE__ */ jsxs4("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1784
+ /* @__PURE__ */ jsx4("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Drag to match:" }),
1785
+ unmatchedRightItems.length > 0 ? unmatchedRightItems.map((rightItem, idx) => {
1786
+ const isDragging = draggedItem === rightItem;
1787
+ return /* @__PURE__ */ jsxs4(
1788
+ "div",
1789
+ {
1790
+ style: {
1791
+ padding: "12px 16px",
1792
+ border: "2px solid #e5e7eb",
1793
+ borderRadius: "8px",
1794
+ backgroundColor: "#ffffff",
1795
+ cursor: showFeedback ? "default" : "grab",
1796
+ opacity: isDragging ? 0.5 : 1,
1797
+ display: "flex",
1798
+ alignItems: "center",
1799
+ gap: "8px",
1800
+ transition: "all 0.2s ease"
1801
+ },
1802
+ draggable: !showFeedback,
1803
+ onDragStart: (e) => handleDragStart(e, rightItem),
1804
+ onDragEnd: handleDragEnd,
1805
+ "data-testid": `draggable-right-${idx}`,
1806
+ children: [
1807
+ !showFeedback && /* @__PURE__ */ jsx4("div", { style: { color: "#9ca3af", display: "flex" }, children: /* @__PURE__ */ jsxs4("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1808
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "5", r: "1" }),
1809
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "12", r: "1" }),
1810
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "19", r: "1" }),
1811
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "5", r: "1" }),
1812
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "12", r: "1" }),
1813
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "19", r: "1" })
1814
+ ] }) }),
1815
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: rightItem })
1816
+ ]
1817
+ },
1818
+ idx
1819
+ );
1820
+ }) : /* @__PURE__ */ jsx4("div", { style: {
1821
+ padding: "16px",
1822
+ textAlign: "center",
1823
+ color: "#22c55e",
1824
+ fontSize: "14px"
1825
+ }, children: "All items matched!" })
1826
+ ] })
1827
+ ] });
1828
+ }
1339
1829
  function Spinner({ size = 16, color = "#ffffff" }) {
1340
- return /* @__PURE__ */ jsx3(
1830
+ return /* @__PURE__ */ jsx4(
1341
1831
  "span",
1342
1832
  {
1343
1833
  style: {
@@ -1349,7 +1839,7 @@ function Spinner({ size = 16, color = "#ffffff" }) {
1349
1839
  borderRadius: "50%",
1350
1840
  animation: "spin 0.8s linear infinite"
1351
1841
  },
1352
- children: /* @__PURE__ */ jsx3("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1842
+ children: /* @__PURE__ */ jsx4("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1353
1843
  }
1354
1844
  );
1355
1845
  }
@@ -1367,7 +1857,8 @@ function QuizPlayer({
1367
1857
  onProgress,
1368
1858
  onGenerateMoreQuestions,
1369
1859
  className,
1370
- forceNewAttempt = true
1860
+ forceNewAttempt = true,
1861
+ hideScore = false
1371
1862
  }) {
1372
1863
  const [quiz, setQuiz] = useState3(null);
1373
1864
  const [attempt, setAttempt] = useState3(null);
@@ -1378,10 +1869,14 @@ function QuizPlayer({
1378
1869
  const [isNavigating, setIsNavigating] = useState3(false);
1379
1870
  const [isCompleted, setIsCompleted] = useState3(false);
1380
1871
  const [result, setResult] = useState3(null);
1381
- const [error, setError] = useState3(null);
1872
+ const [errorCode, setErrorCode] = useState3(null);
1382
1873
  const [isLoading, setIsLoading] = useState3(true);
1383
1874
  const [elapsedSeconds, setElapsedSeconds] = useState3(0);
1384
1875
  const [showIntro, setShowIntro] = useState3(true);
1876
+ const [showResumeChoice, setShowResumeChoice] = useState3(false);
1877
+ const [hasExistingProgress, setHasExistingProgress] = useState3(false);
1878
+ const [existingProgressCount, setExistingProgressCount] = useState3(0);
1879
+ const [isStartingFresh, setIsStartingFresh] = useState3(false);
1385
1880
  const [timerStarted, setTimerStarted] = useState3(false);
1386
1881
  const [showFeedback, setShowFeedback] = useState3(false);
1387
1882
  const [currentAnswerDetail, setCurrentAnswerDetail] = useState3(null);
@@ -1416,7 +1911,7 @@ function QuizPlayer({
1416
1911
  if (!apiClient.current) return;
1417
1912
  try {
1418
1913
  setIsLoading(true);
1419
- setError(null);
1914
+ setErrorCode(null);
1420
1915
  const quizData = await apiClient.current.getQuiz(quizId);
1421
1916
  setQuiz(quizData);
1422
1917
  const attemptData = await apiClient.current.createAttempt({
@@ -1429,7 +1924,10 @@ function QuizPlayer({
1429
1924
  forceNew: forceNewAttempt
1430
1925
  });
1431
1926
  setAttempt(attemptData);
1432
- if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0) {
1927
+ if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0 && attemptData.status === "in_progress") {
1928
+ setHasExistingProgress(true);
1929
+ setExistingProgressCount(attemptData.answers.length);
1930
+ setShowResumeChoice(true);
1433
1931
  setAnswersDetail(attemptData.answers);
1434
1932
  const answersMap = /* @__PURE__ */ new Map();
1435
1933
  attemptData.answers.forEach((a) => {
@@ -1452,7 +1950,15 @@ function QuizPlayer({
1452
1950
  setIsLoading(false);
1453
1951
  } catch (err) {
1454
1952
  const message = err instanceof Error ? err.message : "Failed to load quiz";
1455
- setError(message);
1953
+ const code = getErrorFromMessage(message, "quiz");
1954
+ setErrorCode(code);
1955
+ apiClient.current?.logError({
1956
+ errorCode: code,
1957
+ context: "quiz",
1958
+ resourceId: quizId,
1959
+ component: "QuizPlayer",
1960
+ message
1961
+ });
1456
1962
  setIsLoading(false);
1457
1963
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1458
1964
  }
@@ -1460,7 +1966,7 @@ function QuizPlayer({
1460
1966
  initialize();
1461
1967
  }, [quizId, lessonId, assignLessonId, courseId, childId, parentId, forceNewAttempt]);
1462
1968
  useEffect3(() => {
1463
- if (timerStarted && !isCompleted && !error) {
1969
+ if (timerStarted && !isCompleted && !errorCode) {
1464
1970
  startTimeRef.current = Date.now();
1465
1971
  timerRef.current = setInterval(() => {
1466
1972
  setElapsedSeconds(Math.floor((Date.now() - startTimeRef.current) / 1e3));
@@ -1471,11 +1977,62 @@ function QuizPlayer({
1471
1977
  clearInterval(timerRef.current);
1472
1978
  }
1473
1979
  };
1474
- }, [timerStarted, isCompleted, error]);
1980
+ }, [timerStarted, isCompleted, errorCode]);
1475
1981
  const handleStart = useCallback2(() => {
1476
1982
  setShowIntro(false);
1983
+ setShowResumeChoice(false);
1477
1984
  setTimerStarted(true);
1478
1985
  }, []);
1986
+ const handleResumePrevious = useCallback2(() => {
1987
+ if (quiz && answers.size > 0) {
1988
+ const answeredIds = new Set(answers.keys());
1989
+ const allQs = [...quiz.questions, ...extraQuestions].filter((q) => !skippedQuestionIds.has(q.id));
1990
+ let resumeIndex = 0;
1991
+ for (let i = 0; i < allQs.length; i++) {
1992
+ if (!answeredIds.has(allQs[i].id)) {
1993
+ resumeIndex = i;
1994
+ break;
1995
+ }
1996
+ resumeIndex = allQs.length - 1;
1997
+ }
1998
+ setCurrentQuestionIndex(resumeIndex);
1999
+ }
2000
+ setShowIntro(false);
2001
+ setShowResumeChoice(false);
2002
+ setTimerStarted(true);
2003
+ }, [quiz, answers, extraQuestions, skippedQuestionIds]);
2004
+ const handleStartFresh = useCallback2(async () => {
2005
+ if (!apiClient.current || !attempt) return;
2006
+ setIsStartingFresh(true);
2007
+ try {
2008
+ await apiClient.current.updateAttempt(attempt.id, {
2009
+ status: "abandoned"
2010
+ });
2011
+ const newAttemptData = await apiClient.current.createAttempt({
2012
+ quizId,
2013
+ lessonId,
2014
+ assignLessonId,
2015
+ courseId,
2016
+ childId,
2017
+ parentId,
2018
+ forceNew: true
2019
+ });
2020
+ setAttempt(newAttemptData);
2021
+ setAnswers(/* @__PURE__ */ new Map());
2022
+ setAnswersDetail([]);
2023
+ setCurrentQuestionIndex(0);
2024
+ setHasExistingProgress(false);
2025
+ setExistingProgressCount(0);
2026
+ setShowResumeChoice(false);
2027
+ setShowIntro(false);
2028
+ setTimerStarted(true);
2029
+ } catch (err) {
2030
+ const message = err instanceof Error ? err.message : "Failed to start fresh";
2031
+ onErrorRef.current?.(new Error(message));
2032
+ } finally {
2033
+ setIsStartingFresh(false);
2034
+ }
2035
+ }, [attempt, quizId, lessonId, assignLessonId, courseId, childId, parentId]);
1479
2036
  useEffect3(() => {
1480
2037
  setShowFeedback(false);
1481
2038
  setCurrentAnswerDetail(null);
@@ -1595,7 +2152,15 @@ function QuizPlayer({
1595
2152
  onCompleteRef.current?.(quizResult);
1596
2153
  } catch (err) {
1597
2154
  const message = err instanceof Error ? err.message : "Failed to submit quiz";
1598
- setError(message);
2155
+ const code = getErrorFromMessage(message, "quiz");
2156
+ setErrorCode(code);
2157
+ apiClient.current?.logError({
2158
+ errorCode: code,
2159
+ context: "quiz",
2160
+ resourceId: quizId,
2161
+ component: "QuizPlayer",
2162
+ message
2163
+ });
1599
2164
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1600
2165
  } finally {
1601
2166
  setIsSubmitting(false);
@@ -1674,13 +2239,10 @@ function QuizPlayer({
1674
2239
  }
1675
2240
  }, [currentQuestion, apiClient, attempt, quiz, childId, parentId, lessonId, courseId, assignLessonId]);
1676
2241
  if (isLoading) {
1677
- return /* @__PURE__ */ jsx3("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx3("div", { style: defaultStyles.loading, children: "Loading quiz..." }) });
2242
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx4("div", { style: defaultStyles.loading, children: "Loading quiz..." }) });
1678
2243
  }
1679
- if (error) {
1680
- return /* @__PURE__ */ jsx3("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx3("div", { style: defaultStyles.error, children: /* @__PURE__ */ jsxs3("p", { children: [
1681
- "Error: ",
1682
- error
1683
- ] }) }) });
2244
+ if (errorCode) {
2245
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx4(MaintenanceScreen, { errorCode }) });
1684
2246
  }
1685
2247
  if (isCompleted && result) {
1686
2248
  const percentage = result.totalQuestions > 0 ? Math.round(result.correctAnswers / result.totalQuestions * 100) : 0;
@@ -1738,7 +2300,7 @@ function QuizPlayer({
1738
2300
  rotation: Math.random() * 360,
1739
2301
  size: 6 + Math.random() * 8
1740
2302
  }));
1741
- const StarIcon = ({ filled, delay }) => /* @__PURE__ */ jsx3(
2303
+ const StarIcon = ({ filled, delay }) => /* @__PURE__ */ jsx4(
1742
2304
  "svg",
1743
2305
  {
1744
2306
  width: "36",
@@ -1749,7 +2311,7 @@ function QuizPlayer({
1749
2311
  animationDelay: `${delay}s`,
1750
2312
  opacity: 0
1751
2313
  },
1752
- children: /* @__PURE__ */ jsx3(
2314
+ children: /* @__PURE__ */ jsx4(
1753
2315
  "path",
1754
2316
  {
1755
2317
  d: "M12 2l3.09 6.26L22 9.27l-5 4.87 1.18 6.88L12 17.77l-6.18 3.25L7 14.14 2 9.27l6.91-1.01L12 2z",
@@ -1760,62 +2322,23 @@ function QuizPlayer({
1760
2322
  )
1761
2323
  }
1762
2324
  );
1763
- const MascotOwl = ({ mood }) => {
1764
- const getEyeExpression = () => {
1765
- switch (mood) {
1766
- case "celebrating":
1767
- return { leftEye: ">", rightEye: "<", pupilY: 42 };
1768
- // Squinting happy
1769
- case "happy":
1770
- return { leftEye: null, rightEye: null, pupilY: 42 };
1771
- // Normal happy
1772
- case "encouraging":
1773
- return { leftEye: null, rightEye: null, pupilY: 44 };
1774
- // Looking down warmly
1775
- default:
1776
- return { leftEye: null, rightEye: null, pupilY: 42 };
1777
- }
1778
- };
1779
- const eyeExpr = getEyeExpression();
1780
- return /* @__PURE__ */ jsxs3(
1781
- "svg",
2325
+ const Mascot = ({ mood }) => {
2326
+ return /* @__PURE__ */ jsx4(
2327
+ "img",
1782
2328
  {
1783
- width: "120",
1784
- height: "120",
1785
- viewBox: "0 0 100 100",
2329
+ src: astronautImage,
2330
+ alt: "Astronaut mascot",
2331
+ width: "180",
2332
+ height: "180",
1786
2333
  style: {
1787
- animation: mood === "celebrating" ? "bounce 0.6s ease-in-out infinite" : "gentleBob 2s ease-in-out infinite"
1788
- },
1789
- children: [
1790
- /* @__PURE__ */ jsx3("ellipse", { cx: "50", cy: "60", rx: "35", ry: "30", fill: "#8b5cf6" }),
1791
- /* @__PURE__ */ jsx3("ellipse", { cx: "50", cy: "65", rx: "25", ry: "20", fill: "#c4b5fd" }),
1792
- /* @__PURE__ */ jsx3("circle", { cx: "50", cy: "35", r: "28", fill: "#a78bfa" }),
1793
- /* @__PURE__ */ jsx3("polygon", { points: "28,15 35,30 22,28", fill: "#7c3aed" }),
1794
- /* @__PURE__ */ jsx3("polygon", { points: "72,15 65,30 78,28", fill: "#7c3aed" }),
1795
- /* @__PURE__ */ jsx3("ellipse", { cx: "38", cy: "38", rx: "10", ry: "12", fill: "white" }),
1796
- /* @__PURE__ */ jsx3("ellipse", { cx: "62", cy: "38", rx: "10", ry: "12", fill: "white" }),
1797
- eyeExpr.leftEye ? /* @__PURE__ */ jsx3("text", { x: "38", y: "44", textAnchor: "middle", fontSize: "16", fill: "#1f2937", children: eyeExpr.leftEye }) : /* @__PURE__ */ jsx3("circle", { cx: "38", cy: eyeExpr.pupilY, r: "5", fill: "#1f2937" }),
1798
- eyeExpr.rightEye ? /* @__PURE__ */ jsx3("text", { x: "62", y: "44", textAnchor: "middle", fontSize: "16", fill: "#1f2937", children: eyeExpr.rightEye }) : /* @__PURE__ */ jsx3("circle", { cx: "62", cy: eyeExpr.pupilY, r: "5", fill: "#1f2937" }),
1799
- /* @__PURE__ */ jsx3("polygon", { points: "50,45 45,52 55,52", fill: "#fbbf24" }),
1800
- (mood === "celebrating" || mood === "happy") && /* @__PURE__ */ jsxs3(Fragment2, { children: [
1801
- /* @__PURE__ */ jsx3("ellipse", { cx: "28", cy: "45", rx: "5", ry: "3", fill: "#fda4af", opacity: "0.6" }),
1802
- /* @__PURE__ */ jsx3("ellipse", { cx: "72", cy: "45", rx: "5", ry: "3", fill: "#fda4af", opacity: "0.6" })
1803
- ] }),
1804
- mood === "celebrating" ? /* @__PURE__ */ jsxs3(Fragment2, { children: [
1805
- /* @__PURE__ */ jsx3("ellipse", { cx: "18", cy: "55", rx: "8", ry: "15", fill: "#7c3aed", transform: "rotate(-30 18 55)" }),
1806
- /* @__PURE__ */ jsx3("ellipse", { cx: "82", cy: "55", rx: "8", ry: "15", fill: "#7c3aed", transform: "rotate(30 82 55)" })
1807
- ] }) : /* @__PURE__ */ jsxs3(Fragment2, { children: [
1808
- /* @__PURE__ */ jsx3("ellipse", { cx: "20", cy: "60", rx: "8", ry: "15", fill: "#7c3aed" }),
1809
- /* @__PURE__ */ jsx3("ellipse", { cx: "80", cy: "60", rx: "8", ry: "15", fill: "#7c3aed" })
1810
- ] }),
1811
- /* @__PURE__ */ jsx3("ellipse", { cx: "40", cy: "88", rx: "8", ry: "4", fill: "#fbbf24" }),
1812
- /* @__PURE__ */ jsx3("ellipse", { cx: "60", cy: "88", rx: "8", ry: "4", fill: "#fbbf24" })
1813
- ]
2334
+ animation: mood === "celebrating" ? "bounce 0.6s ease-in-out infinite" : "gentleBob 2s ease-in-out infinite",
2335
+ objectFit: "contain"
2336
+ }
1814
2337
  }
1815
2338
  );
1816
2339
  };
1817
- return /* @__PURE__ */ jsxs3("div", { className, style: defaultStyles.container, children: [
1818
- /* @__PURE__ */ jsx3("style", { children: `
2340
+ return /* @__PURE__ */ jsxs4("div", { className, style: defaultStyles.container, children: [
2341
+ /* @__PURE__ */ jsx4("style", { children: `
1819
2342
  @keyframes confettiFall {
1820
2343
  0% {
1821
2344
  transform: translateY(-10px) rotate(0deg);
@@ -1879,8 +2402,8 @@ function QuizPlayer({
1879
2402
  }
1880
2403
  }
1881
2404
  ` }),
1882
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.results, children: [
1883
- percentage >= 60 && /* @__PURE__ */ jsx3("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ jsx3(
2405
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.results, children: [
2406
+ percentage >= 60 && /* @__PURE__ */ jsx4("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ jsx4(
1884
2407
  "div",
1885
2408
  {
1886
2409
  style: {
@@ -1897,15 +2420,70 @@ function QuizPlayer({
1897
2420
  },
1898
2421
  piece.id
1899
2422
  )) }),
1900
- /* @__PURE__ */ jsx3("div", { style: { ...defaultStyles.resultsBackground, background: theme.bgGradient } }),
1901
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.resultsContent, children: [
1902
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.resultStars, children: [
1903
- /* @__PURE__ */ jsx3(StarIcon, { filled: theme.stars >= 1, delay: 0.3 }),
1904
- /* @__PURE__ */ jsx3(StarIcon, { filled: theme.stars >= 2, delay: 0.5 }),
1905
- /* @__PURE__ */ jsx3(StarIcon, { filled: theme.stars >= 3, delay: 0.7 })
2423
+ /* @__PURE__ */ jsx4("div", { style: { ...defaultStyles.resultsBackground, background: theme.bgGradient } }),
2424
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.resultsContent, children: hideScore ? /* @__PURE__ */ jsxs4(Fragment2, { children: [
2425
+ /* @__PURE__ */ jsx4("div", { style: { marginBottom: "24px" }, children: /* @__PURE__ */ jsx4(Mascot, { mood: "happy" }) }),
2426
+ /* @__PURE__ */ jsx4(
2427
+ "div",
2428
+ {
2429
+ style: {
2430
+ background: "linear-gradient(135deg, #dcfce7 0%, #bbf7d0 100%)",
2431
+ padding: "12px 28px",
2432
+ borderRadius: "50px",
2433
+ boxShadow: "0 4px 12px rgba(0,0,0,0.15)",
2434
+ marginBottom: "20px",
2435
+ animation: "badgePop 0.6s ease-out 0.2s forwards",
2436
+ opacity: 0,
2437
+ border: "3px solid #22c55e"
2438
+ },
2439
+ children: /* @__PURE__ */ jsx4(
2440
+ "span",
2441
+ {
2442
+ style: {
2443
+ fontSize: "22px",
2444
+ fontWeight: "700",
2445
+ color: "#1f2937",
2446
+ textShadow: "0 1px 2px rgba(255,255,255,0.5)"
2447
+ },
2448
+ children: "All Done!"
2449
+ }
2450
+ )
2451
+ }
2452
+ ),
2453
+ /* @__PURE__ */ jsx4(
2454
+ "div",
2455
+ {
2456
+ style: {
2457
+ animation: "scoreSlideIn 0.5s ease-out 0.4s forwards",
2458
+ opacity: 0
2459
+ },
2460
+ children: /* @__PURE__ */ jsx4(
2461
+ "div",
2462
+ {
2463
+ style: {
2464
+ fontSize: "24px",
2465
+ fontWeight: "600",
2466
+ color: "#22c55e",
2467
+ lineHeight: "1.4",
2468
+ marginBottom: "12px"
2469
+ },
2470
+ children: "The quiz was submitted successfully!"
2471
+ }
2472
+ )
2473
+ }
2474
+ ),
2475
+ /* @__PURE__ */ jsxs4("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2476
+ "Time: ",
2477
+ formatTime(result.timeSpentSeconds)
2478
+ ] })
2479
+ ] }) : /* @__PURE__ */ jsxs4(Fragment2, { children: [
2480
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.resultStars, children: [
2481
+ /* @__PURE__ */ jsx4(StarIcon, { filled: theme.stars >= 1, delay: 0.3 }),
2482
+ /* @__PURE__ */ jsx4(StarIcon, { filled: theme.stars >= 2, delay: 0.5 }),
2483
+ /* @__PURE__ */ jsx4(StarIcon, { filled: theme.stars >= 3, delay: 0.7 })
1906
2484
  ] }),
1907
- /* @__PURE__ */ jsx3("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ jsx3(MascotOwl, { mood: theme.mascotMood }) }),
1908
- /* @__PURE__ */ jsx3(
2485
+ /* @__PURE__ */ jsx4("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ jsx4(Mascot, { mood: theme.mascotMood }) }),
2486
+ /* @__PURE__ */ jsx4(
1909
2487
  "div",
1910
2488
  {
1911
2489
  style: {
@@ -1918,7 +2496,7 @@ function QuizPlayer({
1918
2496
  opacity: 0,
1919
2497
  border: `3px solid ${theme.badgeColor}`
1920
2498
  },
1921
- children: /* @__PURE__ */ jsx3(
2499
+ children: /* @__PURE__ */ jsx4(
1922
2500
  "span",
1923
2501
  {
1924
2502
  style: {
@@ -1932,7 +2510,7 @@ function QuizPlayer({
1932
2510
  )
1933
2511
  }
1934
2512
  ),
1935
- /* @__PURE__ */ jsxs3(
2513
+ /* @__PURE__ */ jsxs4(
1936
2514
  "div",
1937
2515
  {
1938
2516
  style: {
@@ -1940,7 +2518,7 @@ function QuizPlayer({
1940
2518
  opacity: 0
1941
2519
  },
1942
2520
  children: [
1943
- /* @__PURE__ */ jsxs3(
2521
+ /* @__PURE__ */ jsxs4(
1944
2522
  "div",
1945
2523
  {
1946
2524
  style: {
@@ -1957,7 +2535,7 @@ function QuizPlayer({
1957
2535
  ]
1958
2536
  }
1959
2537
  ),
1960
- /* @__PURE__ */ jsx3(
2538
+ /* @__PURE__ */ jsx4(
1961
2539
  "div",
1962
2540
  {
1963
2541
  style: {
@@ -1972,25 +2550,84 @@ function QuizPlayer({
1972
2550
  ]
1973
2551
  }
1974
2552
  ),
1975
- /* @__PURE__ */ jsxs3("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2553
+ /* @__PURE__ */ jsxs4("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
1976
2554
  "Time: ",
1977
2555
  formatTime(result.timeSpentSeconds)
1978
2556
  ] })
1979
- ] })
2557
+ ] }) })
1980
2558
  ] })
1981
2559
  ] });
1982
2560
  }
2561
+ if (quiz && showIntro && showResumeChoice && hasExistingProgress) {
2562
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsxs4("div", { style: defaultStyles.intro, children: [
2563
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introTitle, children: "Welcome Back!" }),
2564
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introSubtitle, children: "You have an unfinished quiz. Would you like to continue or start over?" }),
2565
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.introQuestionCount, children: [
2566
+ existingProgressCount,
2567
+ " of ",
2568
+ quiz.questions.length,
2569
+ " question",
2570
+ quiz.questions.length !== 1 ? "s" : "",
2571
+ " answered"
2572
+ ] }),
2573
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", flexDirection: "column", gap: "12px", marginTop: "8px" }, children: [
2574
+ /* @__PURE__ */ jsx4(
2575
+ "button",
2576
+ {
2577
+ style: defaultStyles.startButton,
2578
+ onClick: handleResumePrevious,
2579
+ onMouseOver: (e) => {
2580
+ e.currentTarget.style.transform = "translateY(-2px)";
2581
+ e.currentTarget.style.boxShadow = "0 6px 20px rgba(124, 58, 237, 0.5)";
2582
+ },
2583
+ onMouseOut: (e) => {
2584
+ e.currentTarget.style.transform = "translateY(0)";
2585
+ e.currentTarget.style.boxShadow = "0 4px 14px rgba(124, 58, 237, 0.4)";
2586
+ },
2587
+ "data-testid": "button-continue-quiz",
2588
+ children: "Continue Where I Left Off"
2589
+ }
2590
+ ),
2591
+ /* @__PURE__ */ jsx4(
2592
+ "button",
2593
+ {
2594
+ style: {
2595
+ ...defaultStyles.startButton,
2596
+ background: "transparent",
2597
+ color: "#7c3aed",
2598
+ border: "2px solid #7c3aed",
2599
+ boxShadow: "none"
2600
+ },
2601
+ onClick: handleStartFresh,
2602
+ disabled: isStartingFresh,
2603
+ onMouseOver: (e) => {
2604
+ if (!isStartingFresh) {
2605
+ e.currentTarget.style.transform = "translateY(-2px)";
2606
+ e.currentTarget.style.background = "rgba(124, 58, 237, 0.1)";
2607
+ }
2608
+ },
2609
+ onMouseOut: (e) => {
2610
+ e.currentTarget.style.transform = "translateY(0)";
2611
+ e.currentTarget.style.background = "transparent";
2612
+ },
2613
+ "data-testid": "button-start-fresh",
2614
+ children: isStartingFresh ? "Starting..." : "Start Fresh"
2615
+ }
2616
+ )
2617
+ ] })
2618
+ ] }) });
2619
+ }
1983
2620
  if (quiz && showIntro) {
1984
- return /* @__PURE__ */ jsx3("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsxs3("div", { style: defaultStyles.intro, children: [
1985
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.introTitle, children: quiz.title }),
1986
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.introSubtitle, children: "Ready to test your knowledge?" }),
1987
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.introQuestionCount, children: [
2621
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsxs4("div", { style: defaultStyles.intro, children: [
2622
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introTitle, children: quiz.title }),
2623
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introSubtitle, children: "Ready to test your knowledge?" }),
2624
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.introQuestionCount, children: [
1988
2625
  quiz.questions.length,
1989
2626
  " question",
1990
2627
  quiz.questions.length !== 1 ? "s" : "",
1991
2628
  " to answer"
1992
2629
  ] }),
1993
- /* @__PURE__ */ jsx3(
2630
+ /* @__PURE__ */ jsx4(
1994
2631
  "button",
1995
2632
  {
1996
2633
  style: defaultStyles.startButton,
@@ -2010,7 +2647,7 @@ function QuizPlayer({
2010
2647
  ] }) });
2011
2648
  }
2012
2649
  if (!quiz || !currentQuestion) {
2013
- return /* @__PURE__ */ jsx3("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx3("div", { style: defaultStyles.error, children: "No quiz data available" }) });
2650
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx4("div", { style: defaultStyles.error, children: "No quiz data available" }) });
2014
2651
  }
2015
2652
  const selectedAnswer = answers.get(currentQuestion.id);
2016
2653
  const isLastQuestion = currentQuestionIndex === totalQuestions - 1;
@@ -2018,21 +2655,21 @@ function QuizPlayer({
2018
2655
  const remainingSlots = maxQuestions - totalQuestions;
2019
2656
  const questionsToAdd = Math.min(5, remainingSlots);
2020
2657
  const canAddMore = onGenerateMoreQuestions && remainingSlots > 0;
2021
- return /* @__PURE__ */ jsx3("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsxs3("div", { style: defaultStyles.mainLayout, children: [
2022
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.quizContent, children: [
2023
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.header, children: [
2024
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.title, children: quiz.title }),
2025
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.progress, children: [
2658
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsxs4("div", { style: defaultStyles.mainLayout, children: [
2659
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.quizContent, children: [
2660
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.header, children: [
2661
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.title, children: quiz.title }),
2662
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.progress, children: [
2026
2663
  "Question ",
2027
2664
  currentQuestionIndex + 1,
2028
2665
  " of ",
2029
2666
  totalQuestions
2030
2667
  ] }),
2031
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ jsx3("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2668
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ jsx4("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2032
2669
  ] }),
2033
- /* @__PURE__ */ jsxs3("div", { style: { ...defaultStyles.question, position: "relative", paddingBottom: "40px" }, children: [
2034
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.questionText, children: /* @__PURE__ */ jsx3(TextToSpeech, { text: currentQuestion.question, inline: true, size: "md" }) }),
2035
- isExtraQuestion && /* @__PURE__ */ jsxs3(
2670
+ /* @__PURE__ */ jsxs4("div", { style: { ...defaultStyles.question, position: "relative", paddingBottom: "40px" }, children: [
2671
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.questionText, children: /* @__PURE__ */ jsx4(TextToSpeech, { text: currentQuestion.question, inline: true, size: "md" }) }),
2672
+ isExtraQuestion && /* @__PURE__ */ jsxs4(
2036
2673
  "button",
2037
2674
  {
2038
2675
  onClick: () => setShowSkipModal(true),
@@ -2065,15 +2702,15 @@ function QuizPlayer({
2065
2702
  },
2066
2703
  "data-testid": "button-skip-question",
2067
2704
  children: [
2068
- /* @__PURE__ */ jsxs3("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2069
- /* @__PURE__ */ jsx3("polygon", { points: "5 4 15 12 5 20 5 4" }),
2070
- /* @__PURE__ */ jsx3("line", { x1: "19", y1: "5", x2: "19", y2: "19" })
2705
+ /* @__PURE__ */ jsxs4("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2706
+ /* @__PURE__ */ jsx4("polygon", { points: "5 4 15 12 5 20 5 4" }),
2707
+ /* @__PURE__ */ jsx4("line", { x1: "19", y1: "5", x2: "19", y2: "19" })
2071
2708
  ] }),
2072
- /* @__PURE__ */ jsx3("span", { children: "Skip" })
2709
+ /* @__PURE__ */ jsx4("span", { children: "Skip" })
2073
2710
  ]
2074
2711
  }
2075
2712
  ),
2076
- !isExtraQuestion && /* @__PURE__ */ jsxs3(
2713
+ !isExtraQuestion && /* @__PURE__ */ jsxs4(
2077
2714
  "button",
2078
2715
  {
2079
2716
  onClick: () => setShowReportModal(true),
@@ -2106,15 +2743,15 @@ function QuizPlayer({
2106
2743
  },
2107
2744
  "data-testid": "button-report-question",
2108
2745
  children: [
2109
- /* @__PURE__ */ jsxs3("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2110
- /* @__PURE__ */ jsx3("path", { d: "M4 15s1-1 4-1 5 2 8 2 4-1 4-1V3s-1 1-4 1-5-2-8-2-4 1-4 1z" }),
2111
- /* @__PURE__ */ jsx3("line", { x1: "4", y1: "22", x2: "4", y2: "15" })
2746
+ /* @__PURE__ */ jsxs4("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2747
+ /* @__PURE__ */ jsx4("path", { d: "M4 15s1-1 4-1 5 2 8 2 4-1 4-1V3s-1 1-4 1-5-2-8-2-4 1-4 1z" }),
2748
+ /* @__PURE__ */ jsx4("line", { x1: "4", y1: "22", x2: "4", y2: "15" })
2112
2749
  ] }),
2113
- /* @__PURE__ */ jsx3("span", { children: "Report" })
2750
+ /* @__PURE__ */ jsx4("span", { children: "Report" })
2114
2751
  ]
2115
2752
  }
2116
2753
  ),
2117
- (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2754
+ (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2118
2755
  const isSelected = selectedAnswer === option;
2119
2756
  const isCorrectOption = currentQuestion.correctAnswer === option;
2120
2757
  let optionStyle = { ...defaultStyles.option };
@@ -2127,7 +2764,7 @@ function QuizPlayer({
2127
2764
  } else if (isSelected) {
2128
2765
  optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2129
2766
  }
2130
- return /* @__PURE__ */ jsxs3(
2767
+ return /* @__PURE__ */ jsxs4(
2131
2768
  "div",
2132
2769
  {
2133
2770
  style: {
@@ -2139,14 +2776,14 @@ function QuizPlayer({
2139
2776
  },
2140
2777
  onClick: () => !showFeedback && handleAnswerChange(option),
2141
2778
  children: [
2142
- /* @__PURE__ */ jsx3("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx3(TextToSpeech, { text: option, size: "sm" }) }),
2143
- /* @__PURE__ */ jsx3("span", { style: { flex: 1 }, children: option })
2779
+ /* @__PURE__ */ jsx4("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx4(TextToSpeech, { text: option, size: "sm" }) }),
2780
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
2144
2781
  ]
2145
2782
  },
2146
2783
  idx
2147
2784
  );
2148
2785
  }) }),
2149
- currentQuestion.type === "multiple" && /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2786
+ currentQuestion.type === "multiple" && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2150
2787
  const selected = Array.isArray(selectedAnswer) && selectedAnswer.includes(option);
2151
2788
  const correctAnswers = Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer : currentQuestion.correctAnswer ? [currentQuestion.correctAnswer] : [];
2152
2789
  const isCorrectOption = correctAnswers.includes(option);
@@ -2160,7 +2797,7 @@ function QuizPlayer({
2160
2797
  } else if (selected) {
2161
2798
  optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2162
2799
  }
2163
- return /* @__PURE__ */ jsxs3(
2800
+ return /* @__PURE__ */ jsxs4(
2164
2801
  "div",
2165
2802
  {
2166
2803
  style: {
@@ -2180,14 +2817,14 @@ function QuizPlayer({
2180
2817
  }
2181
2818
  },
2182
2819
  children: [
2183
- /* @__PURE__ */ jsx3("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx3(TextToSpeech, { text: option, size: "sm" }) }),
2184
- /* @__PURE__ */ jsx3("span", { style: { flex: 1 }, children: option })
2820
+ /* @__PURE__ */ jsx4("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx4(TextToSpeech, { text: option, size: "sm" }) }),
2821
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
2185
2822
  ]
2186
2823
  },
2187
2824
  idx
2188
2825
  );
2189
2826
  }) }),
2190
- (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ jsx3(
2827
+ (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ jsx4(
2191
2828
  "textarea",
2192
2829
  {
2193
2830
  style: { ...defaultStyles.input, minHeight: currentQuestion.type === "essay" ? "150px" : "60px" },
@@ -2197,7 +2834,7 @@ function QuizPlayer({
2197
2834
  disabled: showFeedback
2198
2835
  }
2199
2836
  ),
2200
- currentQuestion.type === "fill" && /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ jsx3(
2837
+ currentQuestion.type === "fill" && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ jsx4(
2201
2838
  "input",
2202
2839
  {
2203
2840
  style: defaultStyles.input,
@@ -2212,18 +2849,127 @@ function QuizPlayer({
2212
2849
  },
2213
2850
  idx
2214
2851
  )) }),
2215
- showFeedback && currentAnswerDetail && /* @__PURE__ */ jsxs3("div", { style: {
2852
+ currentQuestion.type === "sorting" && currentQuestion.items && /* @__PURE__ */ jsx4(
2853
+ SortingDragDrop,
2854
+ {
2855
+ items: currentQuestion.items,
2856
+ currentOrder: Array.isArray(selectedAnswer) ? selectedAnswer : currentQuestion.items.map((_, i) => i),
2857
+ correctOrder: currentQuestion.correctOrder,
2858
+ showFeedback,
2859
+ onOrderChange: handleAnswerChange
2860
+ }
2861
+ ),
2862
+ currentQuestion.type === "matrix" && currentQuestion.leftItems && currentQuestion.rightItems && /* @__PURE__ */ jsx4(
2863
+ MatchingDragDrop,
2864
+ {
2865
+ leftItems: currentQuestion.leftItems,
2866
+ rightItems: currentQuestion.rightItems,
2867
+ currentMatches: typeof selectedAnswer === "object" && selectedAnswer !== null && !Array.isArray(selectedAnswer) ? selectedAnswer : {},
2868
+ correctMatches: currentQuestion.correctMatches,
2869
+ showFeedback,
2870
+ onMatchChange: handleAnswerChange
2871
+ }
2872
+ ),
2873
+ currentQuestion.type === "assessment" && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: (() => {
2874
+ const scaleType = currentQuestion.scaleType || "likert";
2875
+ if (scaleType === "yes-no") {
2876
+ const options = ["Yes", "No"];
2877
+ return options.map((option, idx) => {
2878
+ const isSelected = selectedAnswer === option;
2879
+ let optionStyle = { ...defaultStyles.option };
2880
+ if (isSelected) {
2881
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2882
+ }
2883
+ return /* @__PURE__ */ jsx4(
2884
+ "div",
2885
+ {
2886
+ style: {
2887
+ ...optionStyle,
2888
+ cursor: showFeedback ? "default" : "pointer",
2889
+ display: "flex",
2890
+ alignItems: "center",
2891
+ gap: "8px"
2892
+ },
2893
+ onClick: () => !showFeedback && handleAnswerChange(option),
2894
+ "data-testid": `assessment-option-${option.toLowerCase()}`,
2895
+ children: /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
2896
+ },
2897
+ idx
2898
+ );
2899
+ });
2900
+ }
2901
+ if (scaleType === "rating") {
2902
+ const min = currentQuestion.scaleMin || 1;
2903
+ const max = currentQuestion.scaleMax || 5;
2904
+ const ratings = Array.from({ length: max - min + 1 }, (_, i) => min + i);
2905
+ return /* @__PURE__ */ jsx4("div", { style: { display: "flex", gap: "8px", flexWrap: "wrap", justifyContent: "center" }, children: ratings.map((rating) => {
2906
+ const isSelected = selectedAnswer === rating;
2907
+ return /* @__PURE__ */ jsx4(
2908
+ "button",
2909
+ {
2910
+ onClick: () => !showFeedback && handleAnswerChange(rating),
2911
+ disabled: showFeedback,
2912
+ style: {
2913
+ width: "48px",
2914
+ height: "48px",
2915
+ borderRadius: "50%",
2916
+ border: isSelected ? "2px solid #6721b0" : "2px solid #e5e7eb",
2917
+ backgroundColor: isSelected ? "#f3e8ff" : "#ffffff",
2918
+ cursor: showFeedback ? "not-allowed" : "pointer",
2919
+ fontSize: "18px",
2920
+ fontWeight: "600",
2921
+ color: isSelected ? "#6721b0" : "#374151"
2922
+ },
2923
+ "data-testid": `assessment-rating-${rating}`,
2924
+ children: rating
2925
+ },
2926
+ rating
2927
+ );
2928
+ }) });
2929
+ }
2930
+ const likertOptions = [
2931
+ "Strongly Disagree",
2932
+ "Disagree",
2933
+ "Neutral",
2934
+ "Agree",
2935
+ "Strongly Agree"
2936
+ ];
2937
+ return likertOptions.map((option, idx) => {
2938
+ const isSelected = selectedAnswer === option;
2939
+ let optionStyle = { ...defaultStyles.option };
2940
+ if (isSelected) {
2941
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2942
+ }
2943
+ return /* @__PURE__ */ jsx4(
2944
+ "div",
2945
+ {
2946
+ style: {
2947
+ ...optionStyle,
2948
+ cursor: showFeedback ? "default" : "pointer",
2949
+ display: "flex",
2950
+ alignItems: "center",
2951
+ gap: "8px"
2952
+ },
2953
+ onClick: () => !showFeedback && handleAnswerChange(option),
2954
+ "data-testid": `assessment-likert-${idx}`,
2955
+ children: /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
2956
+ },
2957
+ idx
2958
+ );
2959
+ });
2960
+ })() }),
2961
+ showFeedback && currentAnswerDetail && /* @__PURE__ */ jsxs4("div", { style: {
2216
2962
  ...defaultStyles.feedback,
2217
- ...currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
2963
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
2218
2964
  }, children: [
2219
- /* @__PURE__ */ jsx3("div", { style: {
2965
+ /* @__PURE__ */ jsx4("div", { style: {
2220
2966
  ...defaultStyles.feedbackTitle,
2221
- ...currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
2222
- }, children: currentAnswerDetail.isCorrect ? "\u2713 Correct!" : "\u2717 Incorrect" }),
2223
- currentQuestion.explanation && /* @__PURE__ */ jsx3("div", { style: defaultStyles.feedbackExplanation, children: currentQuestion.explanation })
2967
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackTitleNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
2968
+ }, children: currentQuestion.type === "assessment" ? "\u2713 Response recorded" : currentAnswerDetail.isCorrect ? "\u2713 Correct!" : "\u2717 Incorrect" }),
2969
+ currentQuestion.explanation && /* @__PURE__ */ jsx4("div", { style: defaultStyles.feedbackExplanation, children: currentQuestion.explanation })
2224
2970
  ] })
2225
2971
  ] }),
2226
- showSkipModal && /* @__PURE__ */ jsx3("div", { style: {
2972
+ showSkipModal && /* @__PURE__ */ jsx4("div", { style: {
2227
2973
  position: "fixed",
2228
2974
  top: 0,
2229
2975
  left: 0,
@@ -2234,7 +2980,7 @@ function QuizPlayer({
2234
2980
  alignItems: "center",
2235
2981
  justifyContent: "center",
2236
2982
  zIndex: 1e3
2237
- }, children: /* @__PURE__ */ jsxs3("div", { style: {
2983
+ }, children: /* @__PURE__ */ jsxs4("div", { style: {
2238
2984
  backgroundColor: "#ffffff",
2239
2985
  borderRadius: "12px",
2240
2986
  padding: "24px",
@@ -2242,10 +2988,10 @@ function QuizPlayer({
2242
2988
  width: "90%",
2243
2989
  boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2244
2990
  }, children: [
2245
- /* @__PURE__ */ jsx3("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Skip Question" }),
2246
- /* @__PURE__ */ jsx3("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "Help us improve by telling us why you're skipping this question." }),
2247
- /* @__PURE__ */ jsxs3("div", { style: { display: "flex", flexDirection: "column", gap: "8px", marginBottom: "16px" }, children: [
2248
- /* @__PURE__ */ jsx3(
2991
+ /* @__PURE__ */ jsx4("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Skip Question" }),
2992
+ /* @__PURE__ */ jsx4("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "Help us improve by telling us why you're skipping this question." }),
2993
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", flexDirection: "column", gap: "8px", marginBottom: "16px" }, children: [
2994
+ /* @__PURE__ */ jsx4(
2249
2995
  "button",
2250
2996
  {
2251
2997
  onClick: () => setSelectedSkipReason("question_issue"),
@@ -2266,7 +3012,7 @@ function QuizPlayer({
2266
3012
  children: "Question has an issue"
2267
3013
  }
2268
3014
  ),
2269
- /* @__PURE__ */ jsx3(
3015
+ /* @__PURE__ */ jsx4(
2270
3016
  "button",
2271
3017
  {
2272
3018
  onClick: () => setSelectedSkipReason("dont_know"),
@@ -2288,9 +3034,9 @@ function QuizPlayer({
2288
3034
  }
2289
3035
  )
2290
3036
  ] }),
2291
- /* @__PURE__ */ jsxs3("div", { style: { marginBottom: "16px" }, children: [
2292
- /* @__PURE__ */ jsx3("label", { style: { display: "block", fontSize: "13px", fontWeight: "500", color: "#374151", marginBottom: "6px" }, children: "Additional details (optional)" }),
2293
- /* @__PURE__ */ jsx3(
3037
+ /* @__PURE__ */ jsxs4("div", { style: { marginBottom: "16px" }, children: [
3038
+ /* @__PURE__ */ jsx4("label", { style: { display: "block", fontSize: "13px", fontWeight: "500", color: "#374151", marginBottom: "6px" }, children: "Additional details (optional)" }),
3039
+ /* @__PURE__ */ jsx4(
2294
3040
  "textarea",
2295
3041
  {
2296
3042
  value: skipComment,
@@ -2311,13 +3057,13 @@ function QuizPlayer({
2311
3057
  "data-testid": "input-skip-comment"
2312
3058
  }
2313
3059
  ),
2314
- /* @__PURE__ */ jsxs3("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3060
+ /* @__PURE__ */ jsxs4("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2315
3061
  skipComment.length,
2316
3062
  "/200"
2317
3063
  ] })
2318
3064
  ] }),
2319
- /* @__PURE__ */ jsxs3("div", { style: { display: "flex", gap: "10px" }, children: [
2320
- /* @__PURE__ */ jsx3(
3065
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", gap: "10px" }, children: [
3066
+ /* @__PURE__ */ jsx4(
2321
3067
  "button",
2322
3068
  {
2323
3069
  onClick: () => {
@@ -2340,7 +3086,7 @@ function QuizPlayer({
2340
3086
  children: "Cancel"
2341
3087
  }
2342
3088
  ),
2343
- /* @__PURE__ */ jsx3(
3089
+ /* @__PURE__ */ jsx4(
2344
3090
  "button",
2345
3091
  {
2346
3092
  onClick: () => selectedSkipReason && handleSkipQuestion(selectedSkipReason, skipComment),
@@ -2363,7 +3109,7 @@ function QuizPlayer({
2363
3109
  )
2364
3110
  ] })
2365
3111
  ] }) }),
2366
- showReportModal && /* @__PURE__ */ jsx3("div", { style: {
3112
+ showReportModal && /* @__PURE__ */ jsx4("div", { style: {
2367
3113
  position: "fixed",
2368
3114
  top: 0,
2369
3115
  left: 0,
@@ -2374,7 +3120,7 @@ function QuizPlayer({
2374
3120
  alignItems: "center",
2375
3121
  justifyContent: "center",
2376
3122
  zIndex: 1e3
2377
- }, children: /* @__PURE__ */ jsxs3("div", { style: {
3123
+ }, children: /* @__PURE__ */ jsxs4("div", { style: {
2378
3124
  backgroundColor: "#ffffff",
2379
3125
  borderRadius: "12px",
2380
3126
  padding: "24px",
@@ -2382,10 +3128,10 @@ function QuizPlayer({
2382
3128
  width: "90%",
2383
3129
  boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2384
3130
  }, children: [
2385
- /* @__PURE__ */ jsx3("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Report Question" }),
2386
- /* @__PURE__ */ jsx3("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "What's wrong with this question?" }),
2387
- /* @__PURE__ */ jsxs3("div", { style: { marginBottom: "16px" }, children: [
2388
- /* @__PURE__ */ jsx3(
3131
+ /* @__PURE__ */ jsx4("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Report Question" }),
3132
+ /* @__PURE__ */ jsx4("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "What's wrong with this question?" }),
3133
+ /* @__PURE__ */ jsxs4("div", { style: { marginBottom: "16px" }, children: [
3134
+ /* @__PURE__ */ jsx4(
2389
3135
  "textarea",
2390
3136
  {
2391
3137
  value: reportComment,
@@ -2406,13 +3152,13 @@ function QuizPlayer({
2406
3152
  "data-testid": "input-report-comment"
2407
3153
  }
2408
3154
  ),
2409
- /* @__PURE__ */ jsxs3("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3155
+ /* @__PURE__ */ jsxs4("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2410
3156
  reportComment.length,
2411
3157
  "/300"
2412
3158
  ] })
2413
3159
  ] }),
2414
- /* @__PURE__ */ jsxs3("div", { style: { display: "flex", gap: "10px" }, children: [
2415
- /* @__PURE__ */ jsx3(
3160
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", gap: "10px" }, children: [
3161
+ /* @__PURE__ */ jsx4(
2416
3162
  "button",
2417
3163
  {
2418
3164
  onClick: () => {
@@ -2434,7 +3180,7 @@ function QuizPlayer({
2434
3180
  children: "Cancel"
2435
3181
  }
2436
3182
  ),
2437
- /* @__PURE__ */ jsx3(
3183
+ /* @__PURE__ */ jsx4(
2438
3184
  "button",
2439
3185
  {
2440
3186
  onClick: () => handleReportQuestion(reportComment),
@@ -2457,8 +3203,8 @@ function QuizPlayer({
2457
3203
  )
2458
3204
  ] })
2459
3205
  ] }) }),
2460
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.buttonsColumn, children: [
2461
- showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ jsx3(
3206
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.buttonsColumn, children: [
3207
+ showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ jsx4(
2462
3208
  "button",
2463
3209
  {
2464
3210
  style: {
@@ -2468,10 +3214,10 @@ function QuizPlayer({
2468
3214
  onClick: handleAddMoreQuestions,
2469
3215
  disabled: isGeneratingExtra,
2470
3216
  "data-testid": "button-add-more-questions",
2471
- children: isGeneratingExtra ? /* @__PURE__ */ jsxs3(Fragment2, { children: [
2472
- /* @__PURE__ */ jsx3(Spinner, { size: 16, color: "#9ca3af" }),
3217
+ children: isGeneratingExtra ? /* @__PURE__ */ jsxs4(Fragment2, { children: [
3218
+ /* @__PURE__ */ jsx4(Spinner, { size: 16, color: "#9ca3af" }),
2473
3219
  "Generating Questions..."
2474
- ] }) : /* @__PURE__ */ jsxs3(Fragment2, { children: [
3220
+ ] }) : /* @__PURE__ */ jsxs4(Fragment2, { children: [
2475
3221
  "+ Add ",
2476
3222
  questionsToAdd,
2477
3223
  " More Question",
@@ -2479,9 +3225,9 @@ function QuizPlayer({
2479
3225
  ] })
2480
3226
  }
2481
3227
  ),
2482
- /* @__PURE__ */ jsx3("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
3228
+ /* @__PURE__ */ jsx4("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
2483
3229
  // After viewing feedback
2484
- isLastQuestion ? /* @__PURE__ */ jsx3(
3230
+ isLastQuestion ? /* @__PURE__ */ jsx4(
2485
3231
  "button",
2486
3232
  {
2487
3233
  style: {
@@ -2491,9 +3237,9 @@ function QuizPlayer({
2491
3237
  onClick: handleSubmit,
2492
3238
  disabled: isSubmitting || isGeneratingExtra,
2493
3239
  "data-testid": "button-submit-quiz",
2494
- children: isSubmitting ? /* @__PURE__ */ jsx3(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
3240
+ children: isSubmitting ? /* @__PURE__ */ jsx4(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
2495
3241
  }
2496
- ) : /* @__PURE__ */ jsx3(
3242
+ ) : /* @__PURE__ */ jsx4(
2497
3243
  "button",
2498
3244
  {
2499
3245
  style: {
@@ -2507,7 +3253,7 @@ function QuizPlayer({
2507
3253
  )
2508
3254
  ) : (
2509
3255
  // Before checking answer
2510
- /* @__PURE__ */ jsx3(
3256
+ /* @__PURE__ */ jsx4(
2511
3257
  "button",
2512
3258
  {
2513
3259
  style: {
@@ -2517,13 +3263,13 @@ function QuizPlayer({
2517
3263
  onClick: handleCheckAnswer,
2518
3264
  disabled: isNavigating || selectedAnswer === void 0,
2519
3265
  "data-testid": "button-check-answer",
2520
- children: isNavigating ? /* @__PURE__ */ jsx3(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
3266
+ children: isNavigating ? /* @__PURE__ */ jsx4(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
2521
3267
  }
2522
3268
  )
2523
3269
  ) })
2524
3270
  ] })
2525
3271
  ] }),
2526
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ jsx3(
3272
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ jsx4(
2527
3273
  QuestionChatPanel,
2528
3274
  {
2529
3275
  apiClient: apiClient.current,
@@ -2541,7 +3287,7 @@ function QuizPlayer({
2541
3287
  lessonId,
2542
3288
  courseId,
2543
3289
  answerResult: showFeedback && currentAnswerDetail ? {
2544
- wasIncorrect: !currentAnswerDetail.isCorrect,
3290
+ wasIncorrect: currentQuestion.type !== "assessment" && !currentAnswerDetail.isCorrect,
2545
3291
  selectedAnswer: typeof selectedAnswer === "string" ? selectedAnswer : Array.isArray(selectedAnswer) ? selectedAnswer.join(", ") : void 0,
2546
3292
  correctAnswer: typeof currentQuestion.correctAnswer === "string" ? currentQuestion.correctAnswer : Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer.join(", ") : void 0,
2547
3293
  explanation: currentQuestion.explanation
@@ -2552,8 +3298,8 @@ function QuizPlayer({
2552
3298
  }
2553
3299
 
2554
3300
  // src/AttemptViewer.tsx
2555
- import { useState as useState4, useEffect as useEffect4 } from "react";
2556
- import { jsx as jsx4, jsxs as jsxs4 } from "react/jsx-runtime";
3301
+ import { useState as useState4, useEffect as useEffect4, useRef as useRef4 } from "react";
3302
+ import { jsx as jsx5, jsxs as jsxs5 } from "react/jsx-runtime";
2557
3303
  var defaultStyles2 = {
2558
3304
  container: {
2559
3305
  fontFamily: "system-ui, -apple-system, sans-serif",
@@ -2749,10 +3495,31 @@ var spinnerKeyframes = `
2749
3495
  to { transform: rotate(360deg); }
2750
3496
  }
2751
3497
  `;
2752
- function formatAnswer(answer) {
3498
+ function formatAnswer(answer, questionType, items, leftItems) {
2753
3499
  if (answer === null || answer === void 0) {
2754
3500
  return "No answer";
2755
3501
  }
3502
+ if (questionType === "sorting" && Array.isArray(answer)) {
3503
+ const indices = answer;
3504
+ if (items && items.length > 0) {
3505
+ return indices.map((idx, pos) => `${pos + 1}. ${items[idx] ?? `Item ${idx + 1}`}`).join(", ");
3506
+ }
3507
+ return indices.map((idx, pos) => `Position ${pos + 1}: Item ${idx + 1}`).join(", ");
3508
+ }
3509
+ if (questionType === "matrix" && typeof answer === "object" && !Array.isArray(answer)) {
3510
+ const matches = answer;
3511
+ if (leftItems && leftItems.length > 0) {
3512
+ return leftItems.map((left) => {
3513
+ const right = matches[left];
3514
+ return `${left} \u2192 ${right || "No answer"}`;
3515
+ }).join(", ");
3516
+ }
3517
+ const entries = Object.entries(matches);
3518
+ if (entries.length === 0) {
3519
+ return "No answer";
3520
+ }
3521
+ return entries.map(([left, right]) => `${left} \u2192 ${right || "No answer"}`).join(", ");
3522
+ }
2756
3523
  if (typeof answer === "string") {
2757
3524
  return answer;
2758
3525
  }
@@ -2764,6 +3531,15 @@ function formatAnswer(answer) {
2764
3531
  }
2765
3532
  return String(answer);
2766
3533
  }
3534
+ function formatCorrectSortingAnswer(items, correctOrder) {
3535
+ return correctOrder.map((idx, pos) => `${pos + 1}. ${items[idx] ?? `Item ${idx + 1}`}`).join(", ");
3536
+ }
3537
+ function formatCorrectMatrixAnswer(correctMatches, leftItems) {
3538
+ if (leftItems && leftItems.length > 0) {
3539
+ return leftItems.map((left) => `${left} \u2192 ${correctMatches[left] || "N/A"}`).join(", ");
3540
+ }
3541
+ return Object.entries(correctMatches).map(([left, right]) => `${left} \u2192 ${right}`).join(", ");
3542
+ }
2767
3543
  function AttemptViewer({
2768
3544
  attemptId,
2769
3545
  apiBaseUrl,
@@ -2776,23 +3552,39 @@ function AttemptViewer({
2776
3552
  }) {
2777
3553
  const [attempt, setAttempt] = useState4(null);
2778
3554
  const [loading, setLoading] = useState4(true);
2779
- const [error, setError] = useState4(null);
3555
+ const [errorCode, setErrorCode] = useState4(null);
2780
3556
  const [chatHistories, setChatHistories] = useState4({});
2781
3557
  const [expandedChats, setExpandedChats] = useState4(/* @__PURE__ */ new Set());
3558
+ const [fetchedAttemptId, setFetchedAttemptId] = useState4(null);
3559
+ const onErrorRef = useRef4(onError);
3560
+ onErrorRef.current = onError;
2782
3561
  useEffect4(() => {
3562
+ if (fetchedAttemptId === attemptId) return;
2783
3563
  const apiClient = new QuizApiClient({
2784
3564
  baseUrl: apiBaseUrl,
2785
3565
  authToken
2786
3566
  });
2787
3567
  async function fetchAttempt() {
2788
3568
  setLoading(true);
2789
- setError(null);
3569
+ setErrorCode(null);
2790
3570
  try {
2791
3571
  const response = await fetch(`${apiBaseUrl}/api/external/quiz-attempts/${attemptId}`, {
2792
3572
  headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
2793
3573
  });
2794
3574
  if (!response.ok) {
2795
- throw new Error(`Failed to fetch attempt: ${response.statusText}`);
3575
+ const code = getErrorFromHttpStatus(response.status, "attempt");
3576
+ setErrorCode(code);
3577
+ apiClient.logError({
3578
+ errorCode: code,
3579
+ context: "attempt",
3580
+ resourceId: attemptId,
3581
+ component: "AttemptViewer",
3582
+ message: `HTTP ${response.status}: ${response.statusText}`
3583
+ });
3584
+ onErrorRef.current?.(new Error(`Failed to fetch attempt: ${response.statusText}`));
3585
+ setLoading(false);
3586
+ setFetchedAttemptId(attemptId);
3587
+ return;
2796
3588
  }
2797
3589
  const data = await response.json();
2798
3590
  setAttempt(data);
@@ -2806,14 +3598,23 @@ function AttemptViewer({
2806
3598
  }
2807
3599
  } catch (err) {
2808
3600
  const errorMessage = err instanceof Error ? err.message : "Failed to load attempt";
2809
- setError(errorMessage);
2810
- onError?.(err instanceof Error ? err : new Error(errorMessage));
3601
+ const code = getErrorFromMessage(errorMessage, "attempt");
3602
+ setErrorCode(code);
3603
+ apiClient.logError({
3604
+ errorCode: code,
3605
+ context: "attempt",
3606
+ resourceId: attemptId,
3607
+ component: "AttemptViewer",
3608
+ message: errorMessage
3609
+ });
3610
+ onErrorRef.current?.(err instanceof Error ? err : new Error(errorMessage));
2811
3611
  } finally {
2812
3612
  setLoading(false);
3613
+ setFetchedAttemptId(attemptId);
2813
3614
  }
2814
3615
  }
2815
3616
  fetchAttempt();
2816
- }, [attemptId, apiBaseUrl, authToken, onError, showConversation]);
3617
+ }, [attemptId, apiBaseUrl, authToken, showConversation, fetchedAttemptId]);
2817
3618
  const toggleChatExpanded = (questionId) => {
2818
3619
  setExpandedChats((prev) => {
2819
3620
  const newSet = new Set(prev);
@@ -2827,53 +3628,49 @@ function AttemptViewer({
2827
3628
  };
2828
3629
  const handleRetry = () => {
2829
3630
  setLoading(true);
2830
- setError(null);
3631
+ setErrorCode(null);
2831
3632
  window.location.reload();
2832
3633
  };
2833
3634
  if (loading) {
2834
- return /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.container, className, children: [
2835
- /* @__PURE__ */ jsx4("style", { children: spinnerKeyframes }),
2836
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.loading, children: [
2837
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.spinner }),
2838
- /* @__PURE__ */ jsx4("p", { style: { marginTop: "16px", color: "#6b7280" }, children: "Loading attempt..." })
3635
+ return /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.container, className, children: [
3636
+ /* @__PURE__ */ jsx5("style", { children: spinnerKeyframes }),
3637
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.loading, children: [
3638
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.spinner }),
3639
+ /* @__PURE__ */ jsx5("p", { style: { marginTop: "16px", color: "#6b7280" }, children: "Loading attempt..." })
2839
3640
  ] })
2840
3641
  ] });
2841
3642
  }
2842
- if (error || !attempt) {
2843
- return /* @__PURE__ */ jsx4("div", { style: defaultStyles2.container, className, children: /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.error, children: [
2844
- /* @__PURE__ */ jsx4("p", { style: { fontSize: "18px", fontWeight: "500" }, children: "Failed to load attempt" }),
2845
- /* @__PURE__ */ jsx4("p", { style: { marginTop: "8px", color: "#6b7280" }, children: error }),
2846
- /* @__PURE__ */ jsx4("button", { style: defaultStyles2.retryButton, onClick: handleRetry, children: "Try Again" })
2847
- ] }) });
3643
+ if (errorCode || !attempt) {
3644
+ return /* @__PURE__ */ jsx5("div", { style: defaultStyles2.container, className, children: /* @__PURE__ */ jsx5(MaintenanceScreen, { errorCode: errorCode || "ATTEMPT_NOT_FOUND" }) });
2848
3645
  }
2849
3646
  const scorePercentage = attempt.score ?? 0;
2850
3647
  const correctCount = attempt.correctAnswers ?? 0;
2851
3648
  const totalQuestions = attempt.totalQuestions;
2852
3649
  const timeSpent = attempt.timeSpentSeconds ?? 0;
2853
- return /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.container, className, children: [
2854
- /* @__PURE__ */ jsx4("style", { children: spinnerKeyframes }),
2855
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.header, children: /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryGrid, children: [
2856
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryCard, children: [
2857
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryValue, children: [
3650
+ return /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.container, className, children: [
3651
+ /* @__PURE__ */ jsx5("style", { children: spinnerKeyframes }),
3652
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.header, children: /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryGrid, children: [
3653
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryCard, children: [
3654
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryValue, children: [
2858
3655
  scorePercentage,
2859
3656
  "%"
2860
3657
  ] }),
2861
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.summaryLabel, children: "Score" })
3658
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryLabel, children: "Score" })
2862
3659
  ] }),
2863
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryCard, children: [
2864
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryValue, children: [
3660
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryCard, children: [
3661
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryValue, children: [
2865
3662
  correctCount,
2866
3663
  "/",
2867
3664
  totalQuestions
2868
3665
  ] }),
2869
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
3666
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
2870
3667
  ] }),
2871
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryCard, children: [
2872
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.summaryValue, children: formatTime(timeSpent) }),
2873
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.summaryLabel, children: "Time" })
3668
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryCard, children: [
3669
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryValue, children: formatTime(timeSpent) }),
3670
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryLabel, children: "Time" })
2874
3671
  ] })
2875
3672
  ] }) }),
2876
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ jsxs4(
3673
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ jsxs5(
2877
3674
  "div",
2878
3675
  {
2879
3676
  style: {
@@ -2881,12 +3678,12 @@ function AttemptViewer({
2881
3678
  ...answer.isCorrect ? defaultStyles2.questionCardCorrect : defaultStyles2.questionCardIncorrect
2882
3679
  },
2883
3680
  children: [
2884
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.questionHeader, children: [
2885
- /* @__PURE__ */ jsxs4("span", { style: defaultStyles2.questionNumber, children: [
3681
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.questionHeader, children: [
3682
+ /* @__PURE__ */ jsxs5("span", { style: defaultStyles2.questionNumber, children: [
2886
3683
  "Question ",
2887
3684
  index + 1
2888
3685
  ] }),
2889
- /* @__PURE__ */ jsx4(
3686
+ /* @__PURE__ */ jsx5(
2890
3687
  "span",
2891
3688
  {
2892
3689
  style: {
@@ -2897,35 +3694,35 @@ function AttemptViewer({
2897
3694
  }
2898
3695
  )
2899
3696
  ] }),
2900
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.questionText, children: answer.questionText }),
2901
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.answerSection, children: [
2902
- /* @__PURE__ */ jsx4("span", { style: defaultStyles2.answerLabel, children: "Your answer:" }),
2903
- /* @__PURE__ */ jsx4("span", { style: defaultStyles2.studentAnswer, children: formatAnswer(answer.selectedAnswer) })
3697
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.questionText, children: answer.questionText }),
3698
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.answerSection, children: [
3699
+ /* @__PURE__ */ jsx5("span", { style: defaultStyles2.answerLabel, children: "Your answer:" }),
3700
+ /* @__PURE__ */ jsx5("span", { style: defaultStyles2.studentAnswer, children: formatAnswer(answer.selectedAnswer, answer.questionType, answer.items, answer.leftItems) })
2904
3701
  ] }),
2905
- !answer.isCorrect && answer.correctAnswer && /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.answerSection, children: [
2906
- /* @__PURE__ */ jsx4("span", { style: defaultStyles2.answerLabel, children: "Correct answer:" }),
2907
- /* @__PURE__ */ jsx4("span", { style: defaultStyles2.correctAnswer, children: formatAnswer(answer.correctAnswer) })
3702
+ !answer.isCorrect && answer.correctAnswer && /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.answerSection, children: [
3703
+ /* @__PURE__ */ jsx5("span", { style: defaultStyles2.answerLabel, children: "Correct answer:" }),
3704
+ /* @__PURE__ */ jsx5("span", { style: defaultStyles2.correctAnswer, children: answer.questionType === "sorting" && answer.items && answer.correctOrder ? formatCorrectSortingAnswer(answer.items, answer.correctOrder) : answer.questionType === "matrix" && answer.correctAnswer && typeof answer.correctAnswer === "object" && !Array.isArray(answer.correctAnswer) ? formatCorrectMatrixAnswer(answer.correctAnswer, answer.leftItems) : formatAnswer(answer.correctAnswer, answer.questionType, answer.items, answer.leftItems) })
2908
3705
  ] }),
2909
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.points, children: [
3706
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.points, children: [
2910
3707
  answer.pointsEarned,
2911
3708
  " / ",
2912
3709
  answer.points,
2913
3710
  " points"
2914
3711
  ] }),
2915
- showExplanations && answer.explanation && /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.explanation, children: [
2916
- /* @__PURE__ */ jsx4("strong", { children: "Explanation:" }),
3712
+ showExplanations && answer.explanation && /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.explanation, children: [
3713
+ /* @__PURE__ */ jsx5("strong", { children: "Explanation:" }),
2917
3714
  " ",
2918
3715
  answer.explanation
2919
3716
  ] }),
2920
- showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.chatHistorySection, children: [
2921
- /* @__PURE__ */ jsxs4(
3717
+ showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.chatHistorySection, children: [
3718
+ /* @__PURE__ */ jsxs5(
2922
3719
  "button",
2923
3720
  {
2924
3721
  style: defaultStyles2.chatToggleButton,
2925
3722
  onClick: () => toggleChatExpanded(answer.questionId),
2926
3723
  "data-testid": `button-toggle-chat-${answer.questionId}`,
2927
3724
  children: [
2928
- /* @__PURE__ */ jsx4("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ jsx4("path", { d: "M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" }) }),
3725
+ /* @__PURE__ */ jsx5("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ jsx5("path", { d: "M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" }) }),
2929
3726
  expandedChats.has(answer.questionId) ? "Hide" : "View",
2930
3727
  " Chat History (",
2931
3728
  chatHistories[answer.questionId].messages.length,
@@ -2933,7 +3730,7 @@ function AttemptViewer({
2933
3730
  ]
2934
3731
  }
2935
3732
  ),
2936
- expandedChats.has(answer.questionId) && /* @__PURE__ */ jsx4("div", { style: defaultStyles2.chatMessages, children: chatHistories[answer.questionId].messages.map((msg, msgIndex) => /* @__PURE__ */ jsx4(
3733
+ expandedChats.has(answer.questionId) && /* @__PURE__ */ jsx5("div", { style: defaultStyles2.chatMessages, children: chatHistories[answer.questionId].messages.map((msg, msgIndex) => /* @__PURE__ */ jsx5(
2937
3734
  "div",
2938
3735
  {
2939
3736
  style: {
@@ -2951,14 +3748,620 @@ function AttemptViewer({
2951
3748
  )) })
2952
3749
  ] });
2953
3750
  }
3751
+
3752
+ // src/ErrorTypesPanel.tsx
3753
+ import { jsx as jsx6, jsxs as jsxs6 } from "react/jsx-runtime";
3754
+ var panelStyles2 = {
3755
+ container: {
3756
+ fontFamily: "system-ui, -apple-system, sans-serif",
3757
+ padding: "24px",
3758
+ backgroundColor: "#ffffff",
3759
+ borderRadius: "12px",
3760
+ maxWidth: "800px"
3761
+ },
3762
+ header: {
3763
+ marginBottom: "24px"
3764
+ },
3765
+ title: {
3766
+ fontSize: "20px",
3767
+ fontWeight: "600",
3768
+ color: "#111827",
3769
+ marginBottom: "8px"
3770
+ },
3771
+ subtitle: {
3772
+ fontSize: "14px",
3773
+ color: "#6b7280"
3774
+ },
3775
+ section: {
3776
+ marginBottom: "24px"
3777
+ },
3778
+ sectionTitle: {
3779
+ fontSize: "14px",
3780
+ fontWeight: "600",
3781
+ color: "#374151",
3782
+ marginBottom: "12px",
3783
+ textTransform: "uppercase",
3784
+ letterSpacing: "0.05em"
3785
+ },
3786
+ errorList: {
3787
+ display: "flex",
3788
+ flexDirection: "column",
3789
+ gap: "12px"
3790
+ },
3791
+ errorCard: {
3792
+ padding: "16px",
3793
+ backgroundColor: "#f9fafb",
3794
+ borderRadius: "8px",
3795
+ border: "1px solid #e5e7eb"
3796
+ },
3797
+ errorCardBlocking: {
3798
+ borderLeft: "4px solid #ef4444"
3799
+ },
3800
+ errorCardNonBlocking: {
3801
+ borderLeft: "4px solid #f59e0b"
3802
+ },
3803
+ errorHeader: {
3804
+ display: "flex",
3805
+ justifyContent: "space-between",
3806
+ alignItems: "flex-start",
3807
+ marginBottom: "8px"
3808
+ },
3809
+ errorCode: {
3810
+ fontSize: "13px",
3811
+ fontWeight: "600",
3812
+ color: "#1f2937",
3813
+ fontFamily: "monospace",
3814
+ backgroundColor: "#e5e7eb",
3815
+ padding: "2px 8px",
3816
+ borderRadius: "4px"
3817
+ },
3818
+ errorBadge: {
3819
+ fontSize: "11px",
3820
+ fontWeight: "500",
3821
+ padding: "2px 8px",
3822
+ borderRadius: "12px"
3823
+ },
3824
+ blockingBadge: {
3825
+ backgroundColor: "#fee2e2",
3826
+ color: "#dc2626"
3827
+ },
3828
+ nonBlockingBadge: {
3829
+ backgroundColor: "#fef3c7",
3830
+ color: "#d97706"
3831
+ },
3832
+ userMessage: {
3833
+ fontSize: "15px",
3834
+ fontWeight: "500",
3835
+ color: "#111827",
3836
+ marginBottom: "4px"
3837
+ },
3838
+ subMessage: {
3839
+ fontSize: "13px",
3840
+ color: "#6b7280",
3841
+ marginBottom: "8px"
3842
+ },
3843
+ causeLabel: {
3844
+ fontSize: "11px",
3845
+ fontWeight: "600",
3846
+ color: "#9ca3af",
3847
+ textTransform: "uppercase",
3848
+ marginBottom: "4px"
3849
+ },
3850
+ causeText: {
3851
+ fontSize: "13px",
3852
+ color: "#4b5563",
3853
+ fontStyle: "italic"
3854
+ }
3855
+ };
3856
+ function ErrorTypesPanel({ className }) {
3857
+ const blockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => e.isBlocking);
3858
+ const nonBlockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => !e.isBlocking);
3859
+ const renderErrorCard = (error) => /* @__PURE__ */ jsxs6(
3860
+ "div",
3861
+ {
3862
+ style: {
3863
+ ...panelStyles2.errorCard,
3864
+ ...error.isBlocking ? panelStyles2.errorCardBlocking : panelStyles2.errorCardNonBlocking
3865
+ },
3866
+ "data-testid": `error-card-${error.code}`,
3867
+ children: [
3868
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.errorHeader, children: [
3869
+ /* @__PURE__ */ jsx6("code", { style: panelStyles2.errorCode, children: error.code }),
3870
+ /* @__PURE__ */ jsx6(
3871
+ "span",
3872
+ {
3873
+ style: {
3874
+ ...panelStyles2.errorBadge,
3875
+ ...error.isBlocking ? panelStyles2.blockingBadge : panelStyles2.nonBlockingBadge
3876
+ },
3877
+ children: error.isBlocking ? "Blocking" : "Non-Blocking"
3878
+ }
3879
+ )
3880
+ ] }),
3881
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.userMessage, children: error.userMessage }),
3882
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.subMessage, children: error.subMessage }),
3883
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.causeLabel, children: "Why this happens:" }),
3884
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.causeText, children: error.cause })
3885
+ ]
3886
+ },
3887
+ error.code
3888
+ );
3889
+ return /* @__PURE__ */ jsxs6("div", { style: panelStyles2.container, className, "data-testid": "error-types-panel", children: [
3890
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.header, children: [
3891
+ /* @__PURE__ */ jsx6("h2", { style: panelStyles2.title, children: "Error Types Reference" }),
3892
+ /* @__PURE__ */ jsx6("p", { style: panelStyles2.subtitle, children: "List of all error types that can occur in the quiz components, with user-facing messages and technical causes." })
3893
+ ] }),
3894
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.section, children: [
3895
+ /* @__PURE__ */ jsxs6("h3", { style: panelStyles2.sectionTitle, children: [
3896
+ "Blocking Errors (",
3897
+ blockingErrors.length,
3898
+ ")"
3899
+ ] }),
3900
+ /* @__PURE__ */ jsx6("p", { style: { ...panelStyles2.subtitle, marginBottom: "12px" }, children: "These errors prevent the quiz from loading or continuing. Users see a full-screen error message." }),
3901
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.errorList, children: blockingErrors.map(renderErrorCard) })
3902
+ ] }),
3903
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.section, children: [
3904
+ /* @__PURE__ */ jsxs6("h3", { style: panelStyles2.sectionTitle, children: [
3905
+ "Non-Blocking Errors (",
3906
+ nonBlockingErrors.length,
3907
+ ")"
3908
+ ] }),
3909
+ /* @__PURE__ */ jsx6("p", { style: { ...panelStyles2.subtitle, marginBottom: "12px" }, children: "These errors affect specific features but allow the quiz to continue. Users may see a toast notification." }),
3910
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.errorList, children: nonBlockingErrors.map(renderErrorCard) })
3911
+ ] })
3912
+ ] });
3913
+ }
3914
+
3915
+ // src/ErrorLogsPanel.tsx
3916
+ import { useState as useState5, useEffect as useEffect5 } from "react";
3917
+ import { jsx as jsx7, jsxs as jsxs7 } from "react/jsx-runtime";
3918
+ var panelStyles3 = {
3919
+ container: {
3920
+ fontFamily: "system-ui, -apple-system, sans-serif",
3921
+ padding: "24px",
3922
+ backgroundColor: "#ffffff",
3923
+ borderRadius: "12px",
3924
+ maxWidth: "1000px"
3925
+ },
3926
+ header: {
3927
+ marginBottom: "24px"
3928
+ },
3929
+ title: {
3930
+ fontSize: "20px",
3931
+ fontWeight: "600",
3932
+ color: "#111827",
3933
+ marginBottom: "8px"
3934
+ },
3935
+ subtitle: {
3936
+ fontSize: "14px",
3937
+ color: "#6b7280"
3938
+ },
3939
+ tabs: {
3940
+ display: "flex",
3941
+ gap: "8px",
3942
+ marginBottom: "16px",
3943
+ borderBottom: "1px solid #e5e7eb",
3944
+ paddingBottom: "12px"
3945
+ },
3946
+ tab: {
3947
+ padding: "8px 16px",
3948
+ fontSize: "14px",
3949
+ fontWeight: "500",
3950
+ borderRadius: "6px",
3951
+ cursor: "pointer",
3952
+ border: "none",
3953
+ backgroundColor: "transparent",
3954
+ color: "#6b7280"
3955
+ },
3956
+ tabActive: {
3957
+ backgroundColor: "#6721b0",
3958
+ color: "#ffffff"
3959
+ },
3960
+ stats: {
3961
+ display: "flex",
3962
+ gap: "16px",
3963
+ marginBottom: "20px"
3964
+ },
3965
+ statCard: {
3966
+ padding: "12px 16px",
3967
+ backgroundColor: "#f9fafb",
3968
+ borderRadius: "8px",
3969
+ flex: 1
3970
+ },
3971
+ statValue: {
3972
+ fontSize: "24px",
3973
+ fontWeight: "700",
3974
+ color: "#111827"
3975
+ },
3976
+ statLabel: {
3977
+ fontSize: "12px",
3978
+ color: "#6b7280",
3979
+ textTransform: "uppercase",
3980
+ letterSpacing: "0.05em"
3981
+ },
3982
+ errorList: {
3983
+ display: "flex",
3984
+ flexDirection: "column",
3985
+ gap: "12px"
3986
+ },
3987
+ errorCard: {
3988
+ padding: "16px",
3989
+ backgroundColor: "#f9fafb",
3990
+ borderRadius: "8px",
3991
+ border: "1px solid #e5e7eb"
3992
+ },
3993
+ errorCardRecent: {
3994
+ borderLeft: "4px solid #ef4444",
3995
+ backgroundColor: "#fef2f2"
3996
+ },
3997
+ errorCardOld: {
3998
+ borderLeft: "4px solid #9ca3af",
3999
+ opacity: 0.7
4000
+ },
4001
+ errorHeader: {
4002
+ display: "flex",
4003
+ justifyContent: "space-between",
4004
+ alignItems: "flex-start",
4005
+ marginBottom: "8px"
4006
+ },
4007
+ errorLeft: {
4008
+ flex: 1
4009
+ },
4010
+ errorCode: {
4011
+ fontSize: "13px",
4012
+ fontWeight: "600",
4013
+ color: "#1f2937",
4014
+ fontFamily: "monospace",
4015
+ backgroundColor: "#e5e7eb",
4016
+ padding: "2px 8px",
4017
+ borderRadius: "4px",
4018
+ display: "inline-block"
4019
+ },
4020
+ countBadge: {
4021
+ fontSize: "14px",
4022
+ fontWeight: "700",
4023
+ padding: "4px 12px",
4024
+ borderRadius: "16px",
4025
+ backgroundColor: "#ef4444",
4026
+ color: "#ffffff"
4027
+ },
4028
+ countBadgeLow: {
4029
+ backgroundColor: "#f59e0b"
4030
+ },
4031
+ contextBadge: {
4032
+ fontSize: "11px",
4033
+ fontWeight: "500",
4034
+ padding: "2px 8px",
4035
+ borderRadius: "4px",
4036
+ marginLeft: "8px",
4037
+ backgroundColor: "#dbeafe",
4038
+ color: "#1d4ed8"
4039
+ },
4040
+ userMessage: {
4041
+ fontSize: "15px",
4042
+ fontWeight: "500",
4043
+ color: "#111827",
4044
+ marginTop: "8px",
4045
+ marginBottom: "4px"
4046
+ },
4047
+ resourceId: {
4048
+ fontSize: "12px",
4049
+ color: "#6b7280",
4050
+ fontFamily: "monospace",
4051
+ marginTop: "4px"
4052
+ },
4053
+ resourceLink: {
4054
+ color: "#6721b0",
4055
+ textDecoration: "underline",
4056
+ cursor: "pointer"
4057
+ },
4058
+ timestamps: {
4059
+ display: "flex",
4060
+ gap: "16px",
4061
+ marginTop: "8px",
4062
+ fontSize: "12px",
4063
+ color: "#9ca3af"
4064
+ },
4065
+ dismissButton: {
4066
+ padding: "6px 12px",
4067
+ fontSize: "12px",
4068
+ fontWeight: "500",
4069
+ borderRadius: "4px",
4070
+ cursor: "pointer",
4071
+ border: "1px solid #e5e7eb",
4072
+ backgroundColor: "#ffffff",
4073
+ color: "#6b7280",
4074
+ marginTop: "8px"
4075
+ },
4076
+ dismissButtonHover: {
4077
+ backgroundColor: "#f3f4f6"
4078
+ },
4079
+ emptyState: {
4080
+ textAlign: "center",
4081
+ padding: "40px",
4082
+ color: "#9ca3af",
4083
+ fontSize: "14px"
4084
+ },
4085
+ loading: {
4086
+ textAlign: "center",
4087
+ padding: "40px",
4088
+ color: "#6b7280",
4089
+ fontSize: "14px"
4090
+ },
4091
+ pagination: {
4092
+ display: "flex",
4093
+ justifyContent: "center",
4094
+ gap: "8px",
4095
+ marginTop: "20px"
4096
+ },
4097
+ pageButton: {
4098
+ padding: "8px 12px",
4099
+ fontSize: "14px",
4100
+ borderRadius: "4px",
4101
+ cursor: "pointer",
4102
+ border: "1px solid #e5e7eb",
4103
+ backgroundColor: "#ffffff",
4104
+ color: "#374151"
4105
+ },
4106
+ pageButtonDisabled: {
4107
+ opacity: 0.5,
4108
+ cursor: "not-allowed"
4109
+ }
4110
+ };
4111
+ function formatRelativeTime(dateStr) {
4112
+ const date = new Date(dateStr);
4113
+ const now = /* @__PURE__ */ new Date();
4114
+ const diffMs = now.getTime() - date.getTime();
4115
+ const diffMins = Math.floor(diffMs / 6e4);
4116
+ const diffHours = Math.floor(diffMs / 36e5);
4117
+ const diffDays = Math.floor(diffMs / 864e5);
4118
+ if (diffMins < 1) return "Just now";
4119
+ if (diffMins < 60) return `${diffMins}m ago`;
4120
+ if (diffHours < 24) return `${diffHours}h ago`;
4121
+ if (diffDays < 7) return `${diffDays}d ago`;
4122
+ return date.toLocaleDateString();
4123
+ }
4124
+ function isRecent(dateStr) {
4125
+ const date = new Date(dateStr);
4126
+ const now = /* @__PURE__ */ new Date();
4127
+ const diffMs = now.getTime() - date.getTime();
4128
+ const diffHours = diffMs / 36e5;
4129
+ return diffHours < 24;
4130
+ }
4131
+ function ErrorLogsPanel({ apiBaseUrl, authToken, onResourceClick }) {
4132
+ const [logs, setLogs] = useState5([]);
4133
+ const [loading, setLoading] = useState5(true);
4134
+ const [showDismissed, setShowDismissed] = useState5(false);
4135
+ const [page, setPage] = useState5(1);
4136
+ const [totalPages, setTotalPages] = useState5(1);
4137
+ const [total, setTotal] = useState5(0);
4138
+ useEffect5(() => {
4139
+ fetchLogs();
4140
+ }, [showDismissed, page]);
4141
+ async function fetchLogs() {
4142
+ setLoading(true);
4143
+ try {
4144
+ const params = new URLSearchParams();
4145
+ params.set("dismissed", showDismissed ? "true" : "false");
4146
+ params.set("page", page.toString());
4147
+ params.set("limit", "20");
4148
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs?${params}`, {
4149
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4150
+ });
4151
+ if (response.ok) {
4152
+ const data = await response.json();
4153
+ setLogs(data.items || []);
4154
+ setTotalPages(data.totalPages || 1);
4155
+ setTotal(data.total || 0);
4156
+ }
4157
+ } catch (err) {
4158
+ console.error("Failed to fetch error logs:", err);
4159
+ } finally {
4160
+ setLoading(false);
4161
+ }
4162
+ }
4163
+ async function handleDismiss(id) {
4164
+ try {
4165
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/dismiss`, {
4166
+ method: "PATCH",
4167
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4168
+ });
4169
+ if (response.ok) {
4170
+ fetchLogs();
4171
+ }
4172
+ } catch (err) {
4173
+ console.error("Failed to dismiss error:", err);
4174
+ }
4175
+ }
4176
+ async function handleUndismiss(id) {
4177
+ try {
4178
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/undismiss`, {
4179
+ method: "PATCH",
4180
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4181
+ });
4182
+ if (response.ok) {
4183
+ fetchLogs();
4184
+ }
4185
+ } catch (err) {
4186
+ console.error("Failed to undismiss error:", err);
4187
+ }
4188
+ }
4189
+ function getErrorDefinition(code) {
4190
+ return ERROR_DEFINITIONS[code];
4191
+ }
4192
+ const totalErrors = logs.reduce((sum, log) => sum + log.count, 0);
4193
+ const recentCount = logs.filter((log) => isRecent(log.lastSeenAt)).length;
4194
+ return /* @__PURE__ */ jsxs7("div", { style: panelStyles3.container, "data-testid": "error-logs-panel", children: [
4195
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.header, children: [
4196
+ /* @__PURE__ */ jsx7("h2", { style: panelStyles3.title, children: "Error Tracking" }),
4197
+ /* @__PURE__ */ jsx7("p", { style: panelStyles3.subtitle, children: "Aggregated errors from QuizPlayer and AttemptViewer components, sorted by occurrence count" })
4198
+ ] }),
4199
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.tabs, children: [
4200
+ /* @__PURE__ */ jsx7(
4201
+ "button",
4202
+ {
4203
+ style: {
4204
+ ...panelStyles3.tab,
4205
+ ...showDismissed ? {} : panelStyles3.tabActive
4206
+ },
4207
+ onClick: () => {
4208
+ setShowDismissed(false);
4209
+ setPage(1);
4210
+ },
4211
+ "data-testid": "tab-active-errors",
4212
+ children: "Active Errors"
4213
+ }
4214
+ ),
4215
+ /* @__PURE__ */ jsx7(
4216
+ "button",
4217
+ {
4218
+ style: {
4219
+ ...panelStyles3.tab,
4220
+ ...showDismissed ? panelStyles3.tabActive : {}
4221
+ },
4222
+ onClick: () => {
4223
+ setShowDismissed(true);
4224
+ setPage(1);
4225
+ },
4226
+ "data-testid": "tab-dismissed-errors",
4227
+ children: "Dismissed"
4228
+ }
4229
+ )
4230
+ ] }),
4231
+ !showDismissed && /* @__PURE__ */ jsxs7("div", { style: panelStyles3.stats, children: [
4232
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.statCard, children: [
4233
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statValue, children: total }),
4234
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statLabel, children: "Unique Errors" })
4235
+ ] }),
4236
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.statCard, children: [
4237
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statValue, children: totalErrors }),
4238
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statLabel, children: "Total Occurrences" })
4239
+ ] }),
4240
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.statCard, children: [
4241
+ /* @__PURE__ */ jsx7("div", { style: { ...panelStyles3.statValue, color: recentCount > 0 ? "#ef4444" : "#22c55e" }, children: recentCount }),
4242
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statLabel, children: "Last 24h" })
4243
+ ] })
4244
+ ] }),
4245
+ loading ? /* @__PURE__ */ jsx7("div", { style: panelStyles3.loading, children: "Loading..." }) : logs.length === 0 ? /* @__PURE__ */ jsx7("div", { style: panelStyles3.emptyState, children: showDismissed ? "No dismissed errors" : "No active errors" }) : /* @__PURE__ */ jsx7("div", { style: panelStyles3.errorList, children: logs.map((log) => {
4246
+ const def = getErrorDefinition(log.errorCode);
4247
+ const recent = isRecent(log.lastSeenAt);
4248
+ return /* @__PURE__ */ jsxs7(
4249
+ "div",
4250
+ {
4251
+ style: {
4252
+ ...panelStyles3.errorCard,
4253
+ ...recent ? panelStyles3.errorCardRecent : panelStyles3.errorCardOld
4254
+ },
4255
+ "data-testid": `error-log-${log.id}`,
4256
+ children: [
4257
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.errorHeader, children: [
4258
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.errorLeft, children: [
4259
+ /* @__PURE__ */ jsx7("span", { style: panelStyles3.errorCode, children: log.errorCode }),
4260
+ /* @__PURE__ */ jsx7("span", { style: panelStyles3.contextBadge, children: log.context }),
4261
+ /* @__PURE__ */ jsx7("span", { style: { ...panelStyles3.contextBadge, backgroundColor: "#f3e8ff", color: "#7c3aed" }, children: log.component })
4262
+ ] }),
4263
+ /* @__PURE__ */ jsxs7(
4264
+ "span",
4265
+ {
4266
+ style: {
4267
+ ...panelStyles3.countBadge,
4268
+ ...log.count < 10 ? panelStyles3.countBadgeLow : {}
4269
+ },
4270
+ children: [
4271
+ log.count,
4272
+ "x"
4273
+ ]
4274
+ }
4275
+ )
4276
+ ] }),
4277
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.userMessage, children: def?.userMessage || log.lastMessage || "Unknown error" }),
4278
+ log.resourceId && /* @__PURE__ */ jsxs7("div", { style: panelStyles3.resourceId, children: [
4279
+ "Resource:",
4280
+ " ",
4281
+ onResourceClick ? /* @__PURE__ */ jsx7(
4282
+ "span",
4283
+ {
4284
+ style: panelStyles3.resourceLink,
4285
+ onClick: () => onResourceClick(log.resourceId, log.context),
4286
+ "data-testid": `link-resource-${log.id}`,
4287
+ children: log.resourceId
4288
+ }
4289
+ ) : log.resourceId
4290
+ ] }),
4291
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.timestamps, children: [
4292
+ /* @__PURE__ */ jsxs7("span", { children: [
4293
+ "First: ",
4294
+ formatRelativeTime(log.firstSeenAt)
4295
+ ] }),
4296
+ /* @__PURE__ */ jsxs7("span", { children: [
4297
+ "Last: ",
4298
+ formatRelativeTime(log.lastSeenAt)
4299
+ ] })
4300
+ ] }),
4301
+ /* @__PURE__ */ jsx7(
4302
+ "button",
4303
+ {
4304
+ style: panelStyles3.dismissButton,
4305
+ onClick: () => showDismissed ? handleUndismiss(log.id) : handleDismiss(log.id),
4306
+ "data-testid": `button-dismiss-${log.id}`,
4307
+ children: showDismissed ? "Restore" : "Dismiss"
4308
+ }
4309
+ )
4310
+ ]
4311
+ },
4312
+ log.id
4313
+ );
4314
+ }) }),
4315
+ totalPages > 1 && /* @__PURE__ */ jsxs7("div", { style: panelStyles3.pagination, children: [
4316
+ /* @__PURE__ */ jsx7(
4317
+ "button",
4318
+ {
4319
+ style: {
4320
+ ...panelStyles3.pageButton,
4321
+ ...page <= 1 ? panelStyles3.pageButtonDisabled : {}
4322
+ },
4323
+ onClick: () => setPage((p) => Math.max(1, p - 1)),
4324
+ disabled: page <= 1,
4325
+ "data-testid": "button-prev-page",
4326
+ children: "Previous"
4327
+ }
4328
+ ),
4329
+ /* @__PURE__ */ jsxs7("span", { style: { padding: "8px 12px", color: "#6b7280" }, children: [
4330
+ "Page ",
4331
+ page,
4332
+ " of ",
4333
+ totalPages
4334
+ ] }),
4335
+ /* @__PURE__ */ jsx7(
4336
+ "button",
4337
+ {
4338
+ style: {
4339
+ ...panelStyles3.pageButton,
4340
+ ...page >= totalPages ? panelStyles3.pageButtonDisabled : {}
4341
+ },
4342
+ onClick: () => setPage((p) => Math.min(totalPages, p + 1)),
4343
+ disabled: page >= totalPages,
4344
+ "data-testid": "button-next-page",
4345
+ children: "Next"
4346
+ }
4347
+ )
4348
+ ] })
4349
+ ] });
4350
+ }
2954
4351
  export {
2955
4352
  AttemptViewer,
4353
+ ERROR_DEFINITIONS,
4354
+ ErrorLogsPanel,
4355
+ ErrorTypesPanel,
4356
+ MaintenanceScreen,
2956
4357
  QuizApiClient,
2957
4358
  QuizPlayer,
2958
4359
  TextToSpeech,
2959
4360
  calculateScore,
2960
4361
  checkAnswer,
2961
4362
  createAnswerDetail,
2962
- formatTime
4363
+ formatTime,
4364
+ getErrorFromHttpStatus,
4365
+ getErrorFromMessage
2963
4366
  };
2964
4367
  //# sourceMappingURL=index.mjs.map