@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.js CHANGED
@@ -21,13 +21,19 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
21
21
  var index_exports = {};
22
22
  __export(index_exports, {
23
23
  AttemptViewer: () => AttemptViewer,
24
+ ERROR_DEFINITIONS: () => ERROR_DEFINITIONS,
25
+ ErrorLogsPanel: () => ErrorLogsPanel,
26
+ ErrorTypesPanel: () => ErrorTypesPanel,
27
+ MaintenanceScreen: () => MaintenanceScreen,
24
28
  QuizApiClient: () => QuizApiClient,
25
29
  QuizPlayer: () => QuizPlayer,
26
30
  TextToSpeech: () => TextToSpeech,
27
31
  calculateScore: () => calculateScore,
28
32
  checkAnswer: () => checkAnswer,
29
33
  createAnswerDetail: () => createAnswerDetail,
30
- formatTime: () => formatTime
34
+ formatTime: () => formatTime,
35
+ getErrorFromHttpStatus: () => getErrorFromHttpStatus,
36
+ getErrorFromMessage: () => getErrorFromMessage
31
37
  });
32
38
  module.exports = __toCommonJS(index_exports);
33
39
 
@@ -136,6 +142,13 @@ var QuizApiClient = class {
136
142
  }
137
143
  return response.blob();
138
144
  }
145
+ async logError(params) {
146
+ try {
147
+ await this.request("POST", "/api/external/log-error", params);
148
+ } catch (e) {
149
+ console.warn("[QuizEngine] Failed to log error:", e);
150
+ }
151
+ }
139
152
  };
140
153
 
141
154
  // src/utils.ts
@@ -191,15 +204,16 @@ function checkAnswer(question, selectedAnswer) {
191
204
  return { isCorrect, pointsEarned: isCorrect ? points : 0 };
192
205
  }
193
206
  case "essay":
194
- case "assessment":
195
207
  return { isCorrect: false, pointsEarned: 0 };
208
+ case "assessment":
209
+ return { isCorrect: true, pointsEarned: points };
196
210
  default:
197
211
  return { isCorrect: false, pointsEarned: 0 };
198
212
  }
199
213
  }
200
214
  function createAnswerDetail(question, selectedAnswer) {
201
215
  const { isCorrect, pointsEarned } = checkAnswer(question, selectedAnswer);
202
- return {
216
+ const detail = {
203
217
  questionId: question.id,
204
218
  questionText: question.question,
205
219
  questionType: question.type,
@@ -211,6 +225,14 @@ function createAnswerDetail(question, selectedAnswer) {
211
225
  explanation: question.explanation,
212
226
  hint: question.hint
213
227
  };
228
+ if (question.type === "sorting") {
229
+ detail.items = question.items;
230
+ detail.correctOrder = question.correctOrder;
231
+ } else if (question.type === "matrix") {
232
+ detail.leftItems = question.leftItems;
233
+ detail.rightItems = question.rightItems;
234
+ }
235
+ return detail;
214
236
  }
215
237
  function calculateScore(answers) {
216
238
  const totalPoints = answers.reduce((sum, a) => sum + a.points, 0);
@@ -1047,8 +1069,191 @@ function QuestionChatPanel({
1047
1069
  ] });
1048
1070
  }
1049
1071
 
1050
- // src/QuizPlayer.tsx
1072
+ // src/errors.ts
1073
+ var ERROR_DEFINITIONS = {
1074
+ QUIZ_NOT_FOUND: {
1075
+ code: "QUIZ_NOT_FOUND",
1076
+ userMessage: "We couldn't find this quiz",
1077
+ subMessage: "The quiz may have been removed or the link is incorrect.",
1078
+ cause: "The quiz ID does not exist in the database, or the quiz has been deleted.",
1079
+ isBlocking: true
1080
+ },
1081
+ ATTEMPT_NOT_FOUND: {
1082
+ code: "ATTEMPT_NOT_FOUND",
1083
+ userMessage: "We couldn't find this quiz attempt",
1084
+ subMessage: "The attempt may have expired or the link is incorrect.",
1085
+ cause: "The attempt ID does not exist, or the attempt has been deleted/archived.",
1086
+ isBlocking: true
1087
+ },
1088
+ QUIZ_EXPIRED: {
1089
+ code: "QUIZ_EXPIRED",
1090
+ userMessage: "This quiz has expired",
1091
+ subMessage: "The deadline for this quiz has passed.",
1092
+ cause: "The quiz end date/time has passed and submissions are no longer accepted.",
1093
+ isBlocking: true
1094
+ },
1095
+ QUIZ_NOT_STARTED: {
1096
+ code: "QUIZ_NOT_STARTED",
1097
+ userMessage: "This quiz is not available yet",
1098
+ subMessage: "Please check back when the quiz opens.",
1099
+ cause: "The quiz start date/time has not yet been reached.",
1100
+ isBlocking: true
1101
+ },
1102
+ NETWORK_ERROR: {
1103
+ code: "NETWORK_ERROR",
1104
+ userMessage: "Connection problem",
1105
+ subMessage: "Please check your internet connection and try again.",
1106
+ cause: "Unable to reach the server due to network connectivity issues.",
1107
+ isBlocking: true
1108
+ },
1109
+ SERVER_ERROR: {
1110
+ code: "SERVER_ERROR",
1111
+ userMessage: "Something went wrong on our end",
1112
+ subMessage: "Our team has been notified. Please try again later.",
1113
+ cause: "The server encountered an internal error (HTTP 500+).",
1114
+ isBlocking: true
1115
+ },
1116
+ UNAUTHORIZED: {
1117
+ code: "UNAUTHORIZED",
1118
+ userMessage: "Please sign in to continue",
1119
+ subMessage: "You need to be logged in to access this quiz.",
1120
+ cause: "The user is not authenticated (HTTP 401).",
1121
+ isBlocking: true
1122
+ },
1123
+ FORBIDDEN: {
1124
+ code: "FORBIDDEN",
1125
+ userMessage: "You don't have access to this quiz",
1126
+ subMessage: "Contact your instructor if you believe this is a mistake.",
1127
+ cause: "The user does not have permission to access this resource (HTTP 403).",
1128
+ isBlocking: true
1129
+ },
1130
+ TTS_FAILED: {
1131
+ code: "TTS_FAILED",
1132
+ userMessage: "Text-to-speech unavailable",
1133
+ subMessage: "Audio features are temporarily unavailable.",
1134
+ cause: "The text-to-speech service failed or is unreachable.",
1135
+ isBlocking: false
1136
+ },
1137
+ CHAT_FAILED: {
1138
+ code: "CHAT_FAILED",
1139
+ userMessage: "Chat assistance unavailable",
1140
+ subMessage: "The AI helper is temporarily unavailable.",
1141
+ cause: "The chat/AI service failed to initialize or respond.",
1142
+ isBlocking: false
1143
+ },
1144
+ SUBMISSION_FAILED: {
1145
+ code: "SUBMISSION_FAILED",
1146
+ userMessage: "Failed to submit your answer",
1147
+ subMessage: "Please try again. Your progress has been saved.",
1148
+ cause: "The answer submission request failed due to network or server issues.",
1149
+ isBlocking: false
1150
+ },
1151
+ UNKNOWN_ERROR: {
1152
+ code: "UNKNOWN_ERROR",
1153
+ userMessage: "Something unexpected happened",
1154
+ subMessage: "Please try refreshing the page.",
1155
+ cause: "An unclassified error occurred.",
1156
+ isBlocking: true
1157
+ }
1158
+ };
1159
+ function getErrorFromHttpStatus(status, context) {
1160
+ switch (status) {
1161
+ case 401:
1162
+ return "UNAUTHORIZED";
1163
+ case 403:
1164
+ return "FORBIDDEN";
1165
+ case 404:
1166
+ return context === "attempt" ? "ATTEMPT_NOT_FOUND" : "QUIZ_NOT_FOUND";
1167
+ case 410:
1168
+ return "QUIZ_EXPIRED";
1169
+ case 500:
1170
+ case 502:
1171
+ case 503:
1172
+ case 504:
1173
+ return "SERVER_ERROR";
1174
+ default:
1175
+ if (status >= 400 && status < 500) {
1176
+ return context === "attempt" ? "ATTEMPT_NOT_FOUND" : "QUIZ_NOT_FOUND";
1177
+ }
1178
+ return "UNKNOWN_ERROR";
1179
+ }
1180
+ }
1181
+ function getErrorFromMessage(message, context) {
1182
+ const lowerMessage = message.toLowerCase();
1183
+ if (lowerMessage.includes("network") || lowerMessage.includes("fetch") || lowerMessage.includes("connection")) {
1184
+ return "NETWORK_ERROR";
1185
+ }
1186
+ if (lowerMessage.includes("not found") || lowerMessage.includes("404")) {
1187
+ return context === "attempt" ? "ATTEMPT_NOT_FOUND" : "QUIZ_NOT_FOUND";
1188
+ }
1189
+ if (lowerMessage.includes("unauthorized") || lowerMessage.includes("401")) {
1190
+ return "UNAUTHORIZED";
1191
+ }
1192
+ if (lowerMessage.includes("forbidden") || lowerMessage.includes("403")) {
1193
+ return "FORBIDDEN";
1194
+ }
1195
+ if (lowerMessage.includes("expired")) {
1196
+ return "QUIZ_EXPIRED";
1197
+ }
1198
+ if (lowerMessage.includes("tts") || lowerMessage.includes("speech")) {
1199
+ return "TTS_FAILED";
1200
+ }
1201
+ if (lowerMessage.includes("chat")) {
1202
+ return "CHAT_FAILED";
1203
+ }
1204
+ return "UNKNOWN_ERROR";
1205
+ }
1206
+
1207
+ // src/MaintenanceScreen.tsx
1051
1208
  var import_jsx_runtime3 = require("react/jsx-runtime");
1209
+ function MaintenanceScreen({ errorCode }) {
1210
+ const errorDef = errorCode ? ERROR_DEFINITIONS[errorCode] : null;
1211
+ const message = errorDef?.userMessage || "Your quiz is on its way...";
1212
+ const subMessage = errorDef?.subMessage || "Please check back soon!";
1213
+ const containerStyle = {
1214
+ display: "flex",
1215
+ flexDirection: "column",
1216
+ alignItems: "center",
1217
+ justifyContent: "center",
1218
+ minHeight: "300px",
1219
+ padding: "40px",
1220
+ background: "linear-gradient(135deg, #f8f7ff 0%, #e8e4f8 50%, #f0ebff 100%)",
1221
+ borderRadius: "16px"
1222
+ };
1223
+ const iconStyle = {
1224
+ fontSize: "48px",
1225
+ marginBottom: "24px",
1226
+ color: "#8b5cf6"
1227
+ };
1228
+ const messageStyle = {
1229
+ fontSize: "20px",
1230
+ fontWeight: "600",
1231
+ color: "#4c1d95",
1232
+ textAlign: "center",
1233
+ marginBottom: "12px"
1234
+ };
1235
+ const submessageStyle = {
1236
+ fontSize: "14px",
1237
+ color: "#7c3aed",
1238
+ textAlign: "center",
1239
+ opacity: 0.8
1240
+ };
1241
+ return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: containerStyle, "data-testid": "maintenance-screen", "data-error-code": errorCode || "none", children: [
1242
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: iconStyle, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "64", height: "64", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1243
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "12", cy: "12", r: "10" }),
1244
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("path", { d: "M12 8v4" }),
1245
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("path", { d: "M12 16h.01" })
1246
+ ] }) }),
1247
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: messageStyle, "data-testid": "text-error-message", children: message }),
1248
+ /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: submessageStyle, "data-testid": "text-error-submessage", children: subMessage })
1249
+ ] });
1250
+ }
1251
+
1252
+ // src/assets/astronautData.ts
1253
+ var astronautImage = "";
1254
+
1255
+ // src/QuizPlayer.tsx
1256
+ var import_jsx_runtime4 = require("react/jsx-runtime");
1052
1257
  var defaultStyles = {
1053
1258
  container: {
1054
1259
  fontFamily: "system-ui, -apple-system, sans-serif",
@@ -1349,6 +1554,13 @@ var defaultStyles = {
1349
1554
  backgroundColor: "#fef2f2",
1350
1555
  borderColor: "#ef4444"
1351
1556
  },
1557
+ feedbackNeutral: {
1558
+ backgroundColor: "#f0f9ff",
1559
+ borderColor: "#0ea5e9"
1560
+ },
1561
+ feedbackTitleNeutral: {
1562
+ color: "#0369a1"
1563
+ },
1352
1564
  feedbackTitle: {
1353
1565
  fontSize: "16px",
1354
1566
  fontWeight: "600",
@@ -1369,8 +1581,292 @@ var defaultStyles = {
1369
1581
  lineHeight: "1.5"
1370
1582
  }
1371
1583
  };
1584
+ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOrderChange }) {
1585
+ const [draggedIndex, setDraggedIndex] = (0, import_react3.useState)(null);
1586
+ const [dragOverIndex, setDragOverIndex] = (0, import_react3.useState)(null);
1587
+ const handleDragStart = (e, position) => {
1588
+ if (showFeedback) return;
1589
+ setDraggedIndex(position);
1590
+ e.dataTransfer.effectAllowed = "move";
1591
+ e.dataTransfer.setData("text/plain", position.toString());
1592
+ };
1593
+ const handleDragOver = (e, position) => {
1594
+ e.preventDefault();
1595
+ if (showFeedback) return;
1596
+ e.dataTransfer.dropEffect = "move";
1597
+ setDragOverIndex(position);
1598
+ };
1599
+ const handleDragLeave = () => {
1600
+ setDragOverIndex(null);
1601
+ };
1602
+ const handleDrop = (e, toPosition) => {
1603
+ e.preventDefault();
1604
+ if (showFeedback) return;
1605
+ const fromPosition = parseInt(e.dataTransfer.getData("text/plain"), 10);
1606
+ if (fromPosition !== toPosition) {
1607
+ const newOrder = [...currentOrder];
1608
+ const [movedItem] = newOrder.splice(fromPosition, 1);
1609
+ newOrder.splice(toPosition, 0, movedItem);
1610
+ onOrderChange(newOrder);
1611
+ }
1612
+ setDraggedIndex(null);
1613
+ setDragOverIndex(null);
1614
+ };
1615
+ const handleDragEnd = () => {
1616
+ setDraggedIndex(null);
1617
+ setDragOverIndex(null);
1618
+ };
1619
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentOrder.map((itemIndex, position) => {
1620
+ const isCorrectPosition = correctOrder?.[position] === itemIndex;
1621
+ const isDragging = draggedIndex === position;
1622
+ const isDragOver = dragOverIndex === position;
1623
+ let itemStyle = {
1624
+ ...defaultStyles.option,
1625
+ display: "flex",
1626
+ alignItems: "center",
1627
+ gap: "12px",
1628
+ cursor: showFeedback ? "default" : "grab",
1629
+ opacity: isDragging ? 0.5 : 1,
1630
+ transition: "all 0.2s ease",
1631
+ transform: isDragOver && !showFeedback ? "scale(1.02)" : "scale(1)"
1632
+ };
1633
+ if (showFeedback) {
1634
+ if (isCorrectPosition) {
1635
+ itemStyle = { ...itemStyle, ...defaultStyles.optionCorrect };
1636
+ } else {
1637
+ itemStyle = { ...itemStyle, ...defaultStyles.optionIncorrect };
1638
+ }
1639
+ } else if (isDragOver) {
1640
+ itemStyle = { ...itemStyle, borderColor: "#6366f1", backgroundColor: "#eef2ff" };
1641
+ }
1642
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1643
+ "div",
1644
+ {
1645
+ style: itemStyle,
1646
+ "data-testid": `sorting-item-${position}`,
1647
+ draggable: !showFeedback,
1648
+ onDragStart: (e) => handleDragStart(e, position),
1649
+ onDragOver: (e) => handleDragOver(e, position),
1650
+ onDragLeave: handleDragLeave,
1651
+ onDrop: (e) => handleDrop(e, position),
1652
+ onDragEnd: handleDragEnd,
1653
+ children: [
1654
+ !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
1655
+ cursor: "grab",
1656
+ padding: "4px",
1657
+ display: "flex",
1658
+ alignItems: "center",
1659
+ color: "#9ca3af"
1660
+ }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1661
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "5", r: "1" }),
1662
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "12", r: "1" }),
1663
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "19", r: "1" }),
1664
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "5", r: "1" }),
1665
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "12", r: "1" }),
1666
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "19", r: "1" })
1667
+ ] }) }),
1668
+ showFeedback && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
1669
+ display: "flex",
1670
+ alignItems: "center",
1671
+ color: isCorrectPosition ? "#22c55e" : "#ef4444"
1672
+ }, children: isCorrectPosition ? /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1673
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("path", { d: "M22 11.08V12a10 10 0 1 1-5.93-9.14" }),
1674
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("polyline", { points: "22 4 12 14.01 9 11.01" })
1675
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1676
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "12", cy: "12", r: "10" }),
1677
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "15", y1: "9", x2: "9", y2: "15" }),
1678
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "9", y1: "9", x2: "15", y2: "15" })
1679
+ ] }) }),
1680
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: items[itemIndex] }),
1681
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(TextToSpeech, { text: items[itemIndex], size: "sm" }) })
1682
+ ]
1683
+ },
1684
+ itemIndex
1685
+ );
1686
+ }) });
1687
+ }
1688
+ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatches, showFeedback, onMatchChange }) {
1689
+ const [draggedItem, setDraggedItem] = (0, import_react3.useState)(null);
1690
+ const [dragOverLeft, setDragOverLeft] = (0, import_react3.useState)(null);
1691
+ const matchedRightItems = Object.values(currentMatches);
1692
+ const unmatchedRightItems = rightItems.filter((item) => !matchedRightItems.includes(item));
1693
+ const handleDragStart = (e, rightItem) => {
1694
+ if (showFeedback) return;
1695
+ setDraggedItem(rightItem);
1696
+ e.dataTransfer.effectAllowed = "move";
1697
+ e.dataTransfer.setData("text/plain", rightItem);
1698
+ };
1699
+ const handleDragOver = (e, leftItem) => {
1700
+ e.preventDefault();
1701
+ if (showFeedback) return;
1702
+ e.dataTransfer.dropEffect = "move";
1703
+ setDragOverLeft(leftItem);
1704
+ };
1705
+ const handleDragLeave = () => {
1706
+ setDragOverLeft(null);
1707
+ };
1708
+ const handleDrop = (e, leftItem) => {
1709
+ e.preventDefault();
1710
+ if (showFeedback) return;
1711
+ const rightItem = e.dataTransfer.getData("text/plain");
1712
+ if (rightItem) {
1713
+ const newMatches = { ...currentMatches };
1714
+ Object.keys(newMatches).forEach((key) => {
1715
+ if (newMatches[key] === rightItem) {
1716
+ delete newMatches[key];
1717
+ }
1718
+ });
1719
+ newMatches[leftItem] = rightItem;
1720
+ onMatchChange(newMatches);
1721
+ }
1722
+ setDraggedItem(null);
1723
+ setDragOverLeft(null);
1724
+ };
1725
+ const handleDragEnd = () => {
1726
+ setDraggedItem(null);
1727
+ setDragOverLeft(null);
1728
+ };
1729
+ const handleClearMatch = (leftItem) => {
1730
+ if (showFeedback) return;
1731
+ const newMatches = { ...currentMatches };
1732
+ delete newMatches[leftItem];
1733
+ onMatchChange(newMatches);
1734
+ };
1735
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", gap: "24px", flexWrap: "wrap" }, children: [
1736
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1737
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Match these items:" }),
1738
+ leftItems.map((leftItem, idx) => {
1739
+ const matchedRight = currentMatches[leftItem];
1740
+ const correctMatch = correctMatches?.[leftItem];
1741
+ const isCorrect = matchedRight === correctMatch;
1742
+ const isDragOver = dragOverLeft === leftItem;
1743
+ let rowStyle = {
1744
+ display: "flex",
1745
+ alignItems: "center",
1746
+ gap: "12px",
1747
+ padding: "12px 16px",
1748
+ border: "2px dashed #e5e7eb",
1749
+ borderRadius: "8px",
1750
+ backgroundColor: "#ffffff",
1751
+ minHeight: "56px",
1752
+ transition: "all 0.2s ease"
1753
+ };
1754
+ if (showFeedback) {
1755
+ rowStyle.borderStyle = "solid";
1756
+ if (isCorrect) {
1757
+ rowStyle = { ...rowStyle, borderColor: "#22c55e", backgroundColor: "#f0fdf4" };
1758
+ } else {
1759
+ rowStyle = { ...rowStyle, borderColor: "#ef4444", backgroundColor: "#fef2f2" };
1760
+ }
1761
+ } else if (isDragOver) {
1762
+ rowStyle = { ...rowStyle, borderColor: "#6366f1", backgroundColor: "#eef2ff", borderStyle: "solid" };
1763
+ } else if (matchedRight) {
1764
+ rowStyle = { ...rowStyle, borderStyle: "solid", borderColor: "#22c55e" };
1765
+ }
1766
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1767
+ "div",
1768
+ {
1769
+ style: rowStyle,
1770
+ "data-testid": `matrix-row-${idx}`,
1771
+ onDragOver: (e) => handleDragOver(e, leftItem),
1772
+ onDragLeave: handleDragLeave,
1773
+ onDrop: (e) => handleDrop(e, leftItem),
1774
+ children: [
1775
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1, fontWeight: "500" }, children: leftItem }),
1776
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { color: "#6b7280" }, children: "\u2192" }),
1777
+ matchedRight ? /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
1778
+ display: "flex",
1779
+ alignItems: "center",
1780
+ gap: "8px",
1781
+ padding: "6px 12px",
1782
+ backgroundColor: showFeedback ? isCorrect ? "#dcfce7" : "#fee2e2" : "#e0e7ff",
1783
+ borderRadius: "6px",
1784
+ fontSize: "14px"
1785
+ }, children: [
1786
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { children: matchedRight }),
1787
+ !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1788
+ "button",
1789
+ {
1790
+ onClick: () => handleClearMatch(leftItem),
1791
+ style: {
1792
+ background: "none",
1793
+ border: "none",
1794
+ cursor: "pointer",
1795
+ padding: "2px",
1796
+ display: "flex",
1797
+ color: "#6b7280"
1798
+ },
1799
+ "aria-label": "Remove match",
1800
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1801
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1802
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1803
+ ] })
1804
+ }
1805
+ ),
1806
+ showFeedback && (isCorrect ? /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "#22c55e", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("polyline", { points: "20 6 9 17 4 12" }) }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "#ef4444", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1807
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1808
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1809
+ ] }))
1810
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { color: "#9ca3af", fontSize: "14px", fontStyle: "italic" }, children: "Drop here" }),
1811
+ showFeedback && !isCorrect && correctMatch && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("span", { style: { color: "#166534", fontSize: "13px", marginLeft: "8px" }, children: [
1812
+ "(Correct: ",
1813
+ correctMatch,
1814
+ ")"
1815
+ ] })
1816
+ ]
1817
+ },
1818
+ idx
1819
+ );
1820
+ })
1821
+ ] }),
1822
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1823
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Drag to match:" }),
1824
+ unmatchedRightItems.length > 0 ? unmatchedRightItems.map((rightItem, idx) => {
1825
+ const isDragging = draggedItem === rightItem;
1826
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1827
+ "div",
1828
+ {
1829
+ style: {
1830
+ padding: "12px 16px",
1831
+ border: "2px solid #e5e7eb",
1832
+ borderRadius: "8px",
1833
+ backgroundColor: "#ffffff",
1834
+ cursor: showFeedback ? "default" : "grab",
1835
+ opacity: isDragging ? 0.5 : 1,
1836
+ display: "flex",
1837
+ alignItems: "center",
1838
+ gap: "8px",
1839
+ transition: "all 0.2s ease"
1840
+ },
1841
+ draggable: !showFeedback,
1842
+ onDragStart: (e) => handleDragStart(e, rightItem),
1843
+ onDragEnd: handleDragEnd,
1844
+ "data-testid": `draggable-right-${idx}`,
1845
+ children: [
1846
+ !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { color: "#9ca3af", display: "flex" }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1847
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "5", r: "1" }),
1848
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "12", r: "1" }),
1849
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "19", r: "1" }),
1850
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "5", r: "1" }),
1851
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "12", r: "1" }),
1852
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "19", r: "1" })
1853
+ ] }) }),
1854
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: rightItem })
1855
+ ]
1856
+ },
1857
+ idx
1858
+ );
1859
+ }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
1860
+ padding: "16px",
1861
+ textAlign: "center",
1862
+ color: "#22c55e",
1863
+ fontSize: "14px"
1864
+ }, children: "All items matched!" })
1865
+ ] })
1866
+ ] });
1867
+ }
1372
1868
  function Spinner({ size = 16, color = "#ffffff" }) {
1373
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
1869
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1374
1870
  "span",
1375
1871
  {
1376
1872
  style: {
@@ -1382,7 +1878,7 @@ function Spinner({ size = 16, color = "#ffffff" }) {
1382
1878
  borderRadius: "50%",
1383
1879
  animation: "spin 0.8s linear infinite"
1384
1880
  },
1385
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1881
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1386
1882
  }
1387
1883
  );
1388
1884
  }
@@ -1400,7 +1896,8 @@ function QuizPlayer({
1400
1896
  onProgress,
1401
1897
  onGenerateMoreQuestions,
1402
1898
  className,
1403
- forceNewAttempt = true
1899
+ forceNewAttempt = true,
1900
+ hideScore = false
1404
1901
  }) {
1405
1902
  const [quiz, setQuiz] = (0, import_react3.useState)(null);
1406
1903
  const [attempt, setAttempt] = (0, import_react3.useState)(null);
@@ -1411,10 +1908,14 @@ function QuizPlayer({
1411
1908
  const [isNavigating, setIsNavigating] = (0, import_react3.useState)(false);
1412
1909
  const [isCompleted, setIsCompleted] = (0, import_react3.useState)(false);
1413
1910
  const [result, setResult] = (0, import_react3.useState)(null);
1414
- const [error, setError] = (0, import_react3.useState)(null);
1911
+ const [errorCode, setErrorCode] = (0, import_react3.useState)(null);
1415
1912
  const [isLoading, setIsLoading] = (0, import_react3.useState)(true);
1416
1913
  const [elapsedSeconds, setElapsedSeconds] = (0, import_react3.useState)(0);
1417
1914
  const [showIntro, setShowIntro] = (0, import_react3.useState)(true);
1915
+ const [showResumeChoice, setShowResumeChoice] = (0, import_react3.useState)(false);
1916
+ const [hasExistingProgress, setHasExistingProgress] = (0, import_react3.useState)(false);
1917
+ const [existingProgressCount, setExistingProgressCount] = (0, import_react3.useState)(0);
1918
+ const [isStartingFresh, setIsStartingFresh] = (0, import_react3.useState)(false);
1418
1919
  const [timerStarted, setTimerStarted] = (0, import_react3.useState)(false);
1419
1920
  const [showFeedback, setShowFeedback] = (0, import_react3.useState)(false);
1420
1921
  const [currentAnswerDetail, setCurrentAnswerDetail] = (0, import_react3.useState)(null);
@@ -1449,7 +1950,7 @@ function QuizPlayer({
1449
1950
  if (!apiClient.current) return;
1450
1951
  try {
1451
1952
  setIsLoading(true);
1452
- setError(null);
1953
+ setErrorCode(null);
1453
1954
  const quizData = await apiClient.current.getQuiz(quizId);
1454
1955
  setQuiz(quizData);
1455
1956
  const attemptData = await apiClient.current.createAttempt({
@@ -1462,7 +1963,10 @@ function QuizPlayer({
1462
1963
  forceNew: forceNewAttempt
1463
1964
  });
1464
1965
  setAttempt(attemptData);
1465
- if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0) {
1966
+ if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0 && attemptData.status === "in_progress") {
1967
+ setHasExistingProgress(true);
1968
+ setExistingProgressCount(attemptData.answers.length);
1969
+ setShowResumeChoice(true);
1466
1970
  setAnswersDetail(attemptData.answers);
1467
1971
  const answersMap = /* @__PURE__ */ new Map();
1468
1972
  attemptData.answers.forEach((a) => {
@@ -1485,7 +1989,15 @@ function QuizPlayer({
1485
1989
  setIsLoading(false);
1486
1990
  } catch (err) {
1487
1991
  const message = err instanceof Error ? err.message : "Failed to load quiz";
1488
- setError(message);
1992
+ const code = getErrorFromMessage(message, "quiz");
1993
+ setErrorCode(code);
1994
+ apiClient.current?.logError({
1995
+ errorCode: code,
1996
+ context: "quiz",
1997
+ resourceId: quizId,
1998
+ component: "QuizPlayer",
1999
+ message
2000
+ });
1489
2001
  setIsLoading(false);
1490
2002
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1491
2003
  }
@@ -1493,7 +2005,7 @@ function QuizPlayer({
1493
2005
  initialize();
1494
2006
  }, [quizId, lessonId, assignLessonId, courseId, childId, parentId, forceNewAttempt]);
1495
2007
  (0, import_react3.useEffect)(() => {
1496
- if (timerStarted && !isCompleted && !error) {
2008
+ if (timerStarted && !isCompleted && !errorCode) {
1497
2009
  startTimeRef.current = Date.now();
1498
2010
  timerRef.current = setInterval(() => {
1499
2011
  setElapsedSeconds(Math.floor((Date.now() - startTimeRef.current) / 1e3));
@@ -1504,11 +2016,62 @@ function QuizPlayer({
1504
2016
  clearInterval(timerRef.current);
1505
2017
  }
1506
2018
  };
1507
- }, [timerStarted, isCompleted, error]);
2019
+ }, [timerStarted, isCompleted, errorCode]);
1508
2020
  const handleStart = (0, import_react3.useCallback)(() => {
1509
2021
  setShowIntro(false);
2022
+ setShowResumeChoice(false);
1510
2023
  setTimerStarted(true);
1511
2024
  }, []);
2025
+ const handleResumePrevious = (0, import_react3.useCallback)(() => {
2026
+ if (quiz && answers.size > 0) {
2027
+ const answeredIds = new Set(answers.keys());
2028
+ const allQs = [...quiz.questions, ...extraQuestions].filter((q) => !skippedQuestionIds.has(q.id));
2029
+ let resumeIndex = 0;
2030
+ for (let i = 0; i < allQs.length; i++) {
2031
+ if (!answeredIds.has(allQs[i].id)) {
2032
+ resumeIndex = i;
2033
+ break;
2034
+ }
2035
+ resumeIndex = allQs.length - 1;
2036
+ }
2037
+ setCurrentQuestionIndex(resumeIndex);
2038
+ }
2039
+ setShowIntro(false);
2040
+ setShowResumeChoice(false);
2041
+ setTimerStarted(true);
2042
+ }, [quiz, answers, extraQuestions, skippedQuestionIds]);
2043
+ const handleStartFresh = (0, import_react3.useCallback)(async () => {
2044
+ if (!apiClient.current || !attempt) return;
2045
+ setIsStartingFresh(true);
2046
+ try {
2047
+ await apiClient.current.updateAttempt(attempt.id, {
2048
+ status: "abandoned"
2049
+ });
2050
+ const newAttemptData = await apiClient.current.createAttempt({
2051
+ quizId,
2052
+ lessonId,
2053
+ assignLessonId,
2054
+ courseId,
2055
+ childId,
2056
+ parentId,
2057
+ forceNew: true
2058
+ });
2059
+ setAttempt(newAttemptData);
2060
+ setAnswers(/* @__PURE__ */ new Map());
2061
+ setAnswersDetail([]);
2062
+ setCurrentQuestionIndex(0);
2063
+ setHasExistingProgress(false);
2064
+ setExistingProgressCount(0);
2065
+ setShowResumeChoice(false);
2066
+ setShowIntro(false);
2067
+ setTimerStarted(true);
2068
+ } catch (err) {
2069
+ const message = err instanceof Error ? err.message : "Failed to start fresh";
2070
+ onErrorRef.current?.(new Error(message));
2071
+ } finally {
2072
+ setIsStartingFresh(false);
2073
+ }
2074
+ }, [attempt, quizId, lessonId, assignLessonId, courseId, childId, parentId]);
1512
2075
  (0, import_react3.useEffect)(() => {
1513
2076
  setShowFeedback(false);
1514
2077
  setCurrentAnswerDetail(null);
@@ -1628,7 +2191,15 @@ function QuizPlayer({
1628
2191
  onCompleteRef.current?.(quizResult);
1629
2192
  } catch (err) {
1630
2193
  const message = err instanceof Error ? err.message : "Failed to submit quiz";
1631
- setError(message);
2194
+ const code = getErrorFromMessage(message, "quiz");
2195
+ setErrorCode(code);
2196
+ apiClient.current?.logError({
2197
+ errorCode: code,
2198
+ context: "quiz",
2199
+ resourceId: quizId,
2200
+ component: "QuizPlayer",
2201
+ message
2202
+ });
1632
2203
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1633
2204
  } finally {
1634
2205
  setIsSubmitting(false);
@@ -1707,13 +2278,10 @@ function QuizPlayer({
1707
2278
  }
1708
2279
  }, [currentQuestion, apiClient, attempt, quiz, childId, parentId, lessonId, courseId, assignLessonId]);
1709
2280
  if (isLoading) {
1710
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.loading, children: "Loading quiz..." }) });
2281
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.loading, children: "Loading quiz..." }) });
1711
2282
  }
1712
- if (error) {
1713
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.error, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("p", { children: [
1714
- "Error: ",
1715
- error
1716
- ] }) }) });
2283
+ if (errorCode) {
2284
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(MaintenanceScreen, { errorCode }) });
1717
2285
  }
1718
2286
  if (isCompleted && result) {
1719
2287
  const percentage = result.totalQuestions > 0 ? Math.round(result.correctAnswers / result.totalQuestions * 100) : 0;
@@ -1771,7 +2339,7 @@ function QuizPlayer({
1771
2339
  rotation: Math.random() * 360,
1772
2340
  size: 6 + Math.random() * 8
1773
2341
  }));
1774
- const StarIcon = ({ filled, delay }) => /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2342
+ const StarIcon = ({ filled, delay }) => /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1775
2343
  "svg",
1776
2344
  {
1777
2345
  width: "36",
@@ -1782,7 +2350,7 @@ function QuizPlayer({
1782
2350
  animationDelay: `${delay}s`,
1783
2351
  opacity: 0
1784
2352
  },
1785
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2353
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1786
2354
  "path",
1787
2355
  {
1788
2356
  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",
@@ -1793,62 +2361,23 @@ function QuizPlayer({
1793
2361
  )
1794
2362
  }
1795
2363
  );
1796
- const MascotOwl = ({ mood }) => {
1797
- const getEyeExpression = () => {
1798
- switch (mood) {
1799
- case "celebrating":
1800
- return { leftEye: ">", rightEye: "<", pupilY: 42 };
1801
- // Squinting happy
1802
- case "happy":
1803
- return { leftEye: null, rightEye: null, pupilY: 42 };
1804
- // Normal happy
1805
- case "encouraging":
1806
- return { leftEye: null, rightEye: null, pupilY: 44 };
1807
- // Looking down warmly
1808
- default:
1809
- return { leftEye: null, rightEye: null, pupilY: 42 };
1810
- }
1811
- };
1812
- const eyeExpr = getEyeExpression();
1813
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
1814
- "svg",
2364
+ const Mascot = ({ mood }) => {
2365
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2366
+ "img",
1815
2367
  {
1816
- width: "120",
1817
- height: "120",
1818
- viewBox: "0 0 100 100",
2368
+ src: astronautImage,
2369
+ alt: "Astronaut mascot",
2370
+ width: "180",
2371
+ height: "180",
1819
2372
  style: {
1820
- animation: mood === "celebrating" ? "bounce 0.6s ease-in-out infinite" : "gentleBob 2s ease-in-out infinite"
1821
- },
1822
- children: [
1823
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "50", cy: "60", rx: "35", ry: "30", fill: "#8b5cf6" }),
1824
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "50", cy: "65", rx: "25", ry: "20", fill: "#c4b5fd" }),
1825
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "50", cy: "35", r: "28", fill: "#a78bfa" }),
1826
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "28,15 35,30 22,28", fill: "#7c3aed" }),
1827
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "72,15 65,30 78,28", fill: "#7c3aed" }),
1828
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "38", cy: "38", rx: "10", ry: "12", fill: "white" }),
1829
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "62", cy: "38", rx: "10", ry: "12", fill: "white" }),
1830
- eyeExpr.leftEye ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("text", { x: "38", y: "44", textAnchor: "middle", fontSize: "16", fill: "#1f2937", children: eyeExpr.leftEye }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "38", cy: eyeExpr.pupilY, r: "5", fill: "#1f2937" }),
1831
- eyeExpr.rightEye ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("text", { x: "62", y: "44", textAnchor: "middle", fontSize: "16", fill: "#1f2937", children: eyeExpr.rightEye }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "62", cy: eyeExpr.pupilY, r: "5", fill: "#1f2937" }),
1832
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "50,45 45,52 55,52", fill: "#fbbf24" }),
1833
- (mood === "celebrating" || mood === "happy") && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
1834
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "28", cy: "45", rx: "5", ry: "3", fill: "#fda4af", opacity: "0.6" }),
1835
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "72", cy: "45", rx: "5", ry: "3", fill: "#fda4af", opacity: "0.6" })
1836
- ] }),
1837
- mood === "celebrating" ? /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
1838
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "18", cy: "55", rx: "8", ry: "15", fill: "#7c3aed", transform: "rotate(-30 18 55)" }),
1839
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "82", cy: "55", rx: "8", ry: "15", fill: "#7c3aed", transform: "rotate(30 82 55)" })
1840
- ] }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
1841
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "20", cy: "60", rx: "8", ry: "15", fill: "#7c3aed" }),
1842
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "80", cy: "60", rx: "8", ry: "15", fill: "#7c3aed" })
1843
- ] }),
1844
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "40", cy: "88", rx: "8", ry: "4", fill: "#fbbf24" }),
1845
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "60", cy: "88", rx: "8", ry: "4", fill: "#fbbf24" })
1846
- ]
2373
+ animation: mood === "celebrating" ? "bounce 0.6s ease-in-out infinite" : "gentleBob 2s ease-in-out infinite",
2374
+ objectFit: "contain"
2375
+ }
1847
2376
  }
1848
2377
  );
1849
2378
  };
1850
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { className, style: defaultStyles.container, children: [
1851
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("style", { children: `
2379
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { className, style: defaultStyles.container, children: [
2380
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: `
1852
2381
  @keyframes confettiFall {
1853
2382
  0% {
1854
2383
  transform: translateY(-10px) rotate(0deg);
@@ -1912,8 +2441,8 @@ function QuizPlayer({
1912
2441
  }
1913
2442
  }
1914
2443
  ` }),
1915
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.results, children: [
1916
- percentage >= 60 && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2444
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.results, children: [
2445
+ percentage >= 60 && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1917
2446
  "div",
1918
2447
  {
1919
2448
  style: {
@@ -1930,15 +2459,70 @@ function QuizPlayer({
1930
2459
  },
1931
2460
  piece.id
1932
2461
  )) }),
1933
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { ...defaultStyles.resultsBackground, background: theme.bgGradient } }),
1934
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.resultsContent, children: [
1935
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.resultStars, children: [
1936
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(StarIcon, { filled: theme.stars >= 1, delay: 0.3 }),
1937
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(StarIcon, { filled: theme.stars >= 2, delay: 0.5 }),
1938
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(StarIcon, { filled: theme.stars >= 3, delay: 0.7 })
2462
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { ...defaultStyles.resultsBackground, background: theme.bgGradient } }),
2463
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.resultsContent, children: hideScore ? /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
2464
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { marginBottom: "24px" }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Mascot, { mood: "happy" }) }),
2465
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2466
+ "div",
2467
+ {
2468
+ style: {
2469
+ background: "linear-gradient(135deg, #dcfce7 0%, #bbf7d0 100%)",
2470
+ padding: "12px 28px",
2471
+ borderRadius: "50px",
2472
+ boxShadow: "0 4px 12px rgba(0,0,0,0.15)",
2473
+ marginBottom: "20px",
2474
+ animation: "badgePop 0.6s ease-out 0.2s forwards",
2475
+ opacity: 0,
2476
+ border: "3px solid #22c55e"
2477
+ },
2478
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2479
+ "span",
2480
+ {
2481
+ style: {
2482
+ fontSize: "22px",
2483
+ fontWeight: "700",
2484
+ color: "#1f2937",
2485
+ textShadow: "0 1px 2px rgba(255,255,255,0.5)"
2486
+ },
2487
+ children: "All Done!"
2488
+ }
2489
+ )
2490
+ }
2491
+ ),
2492
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2493
+ "div",
2494
+ {
2495
+ style: {
2496
+ animation: "scoreSlideIn 0.5s ease-out 0.4s forwards",
2497
+ opacity: 0
2498
+ },
2499
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2500
+ "div",
2501
+ {
2502
+ style: {
2503
+ fontSize: "24px",
2504
+ fontWeight: "600",
2505
+ color: "#22c55e",
2506
+ lineHeight: "1.4",
2507
+ marginBottom: "12px"
2508
+ },
2509
+ children: "The quiz was submitted successfully!"
2510
+ }
2511
+ )
2512
+ }
2513
+ ),
2514
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2515
+ "Time: ",
2516
+ formatTime(result.timeSpentSeconds)
2517
+ ] })
2518
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
2519
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.resultStars, children: [
2520
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(StarIcon, { filled: theme.stars >= 1, delay: 0.3 }),
2521
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(StarIcon, { filled: theme.stars >= 2, delay: 0.5 }),
2522
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(StarIcon, { filled: theme.stars >= 3, delay: 0.7 })
1939
2523
  ] }),
1940
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(MascotOwl, { mood: theme.mascotMood }) }),
1941
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2524
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Mascot, { mood: theme.mascotMood }) }),
2525
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1942
2526
  "div",
1943
2527
  {
1944
2528
  style: {
@@ -1951,7 +2535,7 @@ function QuizPlayer({
1951
2535
  opacity: 0,
1952
2536
  border: `3px solid ${theme.badgeColor}`
1953
2537
  },
1954
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2538
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1955
2539
  "span",
1956
2540
  {
1957
2541
  style: {
@@ -1965,7 +2549,7 @@ function QuizPlayer({
1965
2549
  )
1966
2550
  }
1967
2551
  ),
1968
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2552
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1969
2553
  "div",
1970
2554
  {
1971
2555
  style: {
@@ -1973,7 +2557,7 @@ function QuizPlayer({
1973
2557
  opacity: 0
1974
2558
  },
1975
2559
  children: [
1976
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2560
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1977
2561
  "div",
1978
2562
  {
1979
2563
  style: {
@@ -1990,7 +2574,7 @@ function QuizPlayer({
1990
2574
  ]
1991
2575
  }
1992
2576
  ),
1993
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2577
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1994
2578
  "div",
1995
2579
  {
1996
2580
  style: {
@@ -2005,25 +2589,84 @@ function QuizPlayer({
2005
2589
  ]
2006
2590
  }
2007
2591
  ),
2008
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2592
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2009
2593
  "Time: ",
2010
2594
  formatTime(result.timeSpentSeconds)
2011
2595
  ] })
2012
- ] })
2596
+ ] }) })
2013
2597
  ] })
2014
2598
  ] });
2015
2599
  }
2600
+ if (quiz && showIntro && showResumeChoice && hasExistingProgress) {
2601
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.intro, children: [
2602
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introTitle, children: "Welcome Back!" }),
2603
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introSubtitle, children: "You have an unfinished quiz. Would you like to continue or start over?" }),
2604
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.introQuestionCount, children: [
2605
+ existingProgressCount,
2606
+ " of ",
2607
+ quiz.questions.length,
2608
+ " question",
2609
+ quiz.questions.length !== 1 ? "s" : "",
2610
+ " answered"
2611
+ ] }),
2612
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", flexDirection: "column", gap: "12px", marginTop: "8px" }, children: [
2613
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2614
+ "button",
2615
+ {
2616
+ style: defaultStyles.startButton,
2617
+ onClick: handleResumePrevious,
2618
+ onMouseOver: (e) => {
2619
+ e.currentTarget.style.transform = "translateY(-2px)";
2620
+ e.currentTarget.style.boxShadow = "0 6px 20px rgba(124, 58, 237, 0.5)";
2621
+ },
2622
+ onMouseOut: (e) => {
2623
+ e.currentTarget.style.transform = "translateY(0)";
2624
+ e.currentTarget.style.boxShadow = "0 4px 14px rgba(124, 58, 237, 0.4)";
2625
+ },
2626
+ "data-testid": "button-continue-quiz",
2627
+ children: "Continue Where I Left Off"
2628
+ }
2629
+ ),
2630
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2631
+ "button",
2632
+ {
2633
+ style: {
2634
+ ...defaultStyles.startButton,
2635
+ background: "transparent",
2636
+ color: "#7c3aed",
2637
+ border: "2px solid #7c3aed",
2638
+ boxShadow: "none"
2639
+ },
2640
+ onClick: handleStartFresh,
2641
+ disabled: isStartingFresh,
2642
+ onMouseOver: (e) => {
2643
+ if (!isStartingFresh) {
2644
+ e.currentTarget.style.transform = "translateY(-2px)";
2645
+ e.currentTarget.style.background = "rgba(124, 58, 237, 0.1)";
2646
+ }
2647
+ },
2648
+ onMouseOut: (e) => {
2649
+ e.currentTarget.style.transform = "translateY(0)";
2650
+ e.currentTarget.style.background = "transparent";
2651
+ },
2652
+ "data-testid": "button-start-fresh",
2653
+ children: isStartingFresh ? "Starting..." : "Start Fresh"
2654
+ }
2655
+ )
2656
+ ] })
2657
+ ] }) });
2658
+ }
2016
2659
  if (quiz && showIntro) {
2017
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.intro, children: [
2018
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.introTitle, children: quiz.title }),
2019
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.introSubtitle, children: "Ready to test your knowledge?" }),
2020
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.introQuestionCount, children: [
2660
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.intro, children: [
2661
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introTitle, children: quiz.title }),
2662
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introSubtitle, children: "Ready to test your knowledge?" }),
2663
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.introQuestionCount, children: [
2021
2664
  quiz.questions.length,
2022
2665
  " question",
2023
2666
  quiz.questions.length !== 1 ? "s" : "",
2024
2667
  " to answer"
2025
2668
  ] }),
2026
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2669
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2027
2670
  "button",
2028
2671
  {
2029
2672
  style: defaultStyles.startButton,
@@ -2043,7 +2686,7 @@ function QuizPlayer({
2043
2686
  ] }) });
2044
2687
  }
2045
2688
  if (!quiz || !currentQuestion) {
2046
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.error, children: "No quiz data available" }) });
2689
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.error, children: "No quiz data available" }) });
2047
2690
  }
2048
2691
  const selectedAnswer = answers.get(currentQuestion.id);
2049
2692
  const isLastQuestion = currentQuestionIndex === totalQuestions - 1;
@@ -2051,21 +2694,21 @@ function QuizPlayer({
2051
2694
  const remainingSlots = maxQuestions - totalQuestions;
2052
2695
  const questionsToAdd = Math.min(5, remainingSlots);
2053
2696
  const canAddMore = onGenerateMoreQuestions && remainingSlots > 0;
2054
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.mainLayout, children: [
2055
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.quizContent, children: [
2056
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.header, children: [
2057
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.title, children: quiz.title }),
2058
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.progress, children: [
2697
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.mainLayout, children: [
2698
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.quizContent, children: [
2699
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.header, children: [
2700
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.title, children: quiz.title }),
2701
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.progress, children: [
2059
2702
  "Question ",
2060
2703
  currentQuestionIndex + 1,
2061
2704
  " of ",
2062
2705
  totalQuestions
2063
2706
  ] }),
2064
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2707
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2065
2708
  ] }),
2066
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { ...defaultStyles.question, position: "relative", paddingBottom: "40px" }, children: [
2067
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.questionText, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(TextToSpeech, { text: currentQuestion.question, inline: true, size: "md" }) }),
2068
- isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2709
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { ...defaultStyles.question, position: "relative", paddingBottom: "40px" }, children: [
2710
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.questionText, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(TextToSpeech, { text: currentQuestion.question, inline: true, size: "md" }) }),
2711
+ isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2069
2712
  "button",
2070
2713
  {
2071
2714
  onClick: () => setShowSkipModal(true),
@@ -2098,15 +2741,15 @@ function QuizPlayer({
2098
2741
  },
2099
2742
  "data-testid": "button-skip-question",
2100
2743
  children: [
2101
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2102
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "5 4 15 12 5 20 5 4" }),
2103
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "19", y1: "5", x2: "19", y2: "19" })
2744
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2745
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("polygon", { points: "5 4 15 12 5 20 5 4" }),
2746
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "19", y1: "5", x2: "19", y2: "19" })
2104
2747
  ] }),
2105
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { children: "Skip" })
2748
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { children: "Skip" })
2106
2749
  ]
2107
2750
  }
2108
2751
  ),
2109
- !isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2752
+ !isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2110
2753
  "button",
2111
2754
  {
2112
2755
  onClick: () => setShowReportModal(true),
@@ -2139,15 +2782,15 @@ function QuizPlayer({
2139
2782
  },
2140
2783
  "data-testid": "button-report-question",
2141
2784
  children: [
2142
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2143
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("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" }),
2144
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "4", y1: "22", x2: "4", y2: "15" })
2785
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2786
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("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" }),
2787
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "4", y1: "22", x2: "4", y2: "15" })
2145
2788
  ] }),
2146
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { children: "Report" })
2789
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { children: "Report" })
2147
2790
  ]
2148
2791
  }
2149
2792
  ),
2150
- (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2793
+ (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2151
2794
  const isSelected = selectedAnswer === option;
2152
2795
  const isCorrectOption = currentQuestion.correctAnswer === option;
2153
2796
  let optionStyle = { ...defaultStyles.option };
@@ -2160,7 +2803,7 @@ function QuizPlayer({
2160
2803
  } else if (isSelected) {
2161
2804
  optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2162
2805
  }
2163
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2806
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2164
2807
  "div",
2165
2808
  {
2166
2809
  style: {
@@ -2172,14 +2815,14 @@ function QuizPlayer({
2172
2815
  },
2173
2816
  onClick: () => !showFeedback && handleAnswerChange(option),
2174
2817
  children: [
2175
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2176
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: option })
2818
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2819
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
2177
2820
  ]
2178
2821
  },
2179
2822
  idx
2180
2823
  );
2181
2824
  }) }),
2182
- currentQuestion.type === "multiple" && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2825
+ currentQuestion.type === "multiple" && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2183
2826
  const selected = Array.isArray(selectedAnswer) && selectedAnswer.includes(option);
2184
2827
  const correctAnswers = Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer : currentQuestion.correctAnswer ? [currentQuestion.correctAnswer] : [];
2185
2828
  const isCorrectOption = correctAnswers.includes(option);
@@ -2193,7 +2836,7 @@ function QuizPlayer({
2193
2836
  } else if (selected) {
2194
2837
  optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2195
2838
  }
2196
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2839
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2197
2840
  "div",
2198
2841
  {
2199
2842
  style: {
@@ -2213,14 +2856,14 @@ function QuizPlayer({
2213
2856
  }
2214
2857
  },
2215
2858
  children: [
2216
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2217
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: option })
2859
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2860
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
2218
2861
  ]
2219
2862
  },
2220
2863
  idx
2221
2864
  );
2222
2865
  }) }),
2223
- (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2866
+ (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2224
2867
  "textarea",
2225
2868
  {
2226
2869
  style: { ...defaultStyles.input, minHeight: currentQuestion.type === "essay" ? "150px" : "60px" },
@@ -2230,7 +2873,7 @@ function QuizPlayer({
2230
2873
  disabled: showFeedback
2231
2874
  }
2232
2875
  ),
2233
- currentQuestion.type === "fill" && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2876
+ currentQuestion.type === "fill" && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2234
2877
  "input",
2235
2878
  {
2236
2879
  style: defaultStyles.input,
@@ -2245,18 +2888,127 @@ function QuizPlayer({
2245
2888
  },
2246
2889
  idx
2247
2890
  )) }),
2248
- showFeedback && currentAnswerDetail && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: {
2891
+ currentQuestion.type === "sorting" && currentQuestion.items && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2892
+ SortingDragDrop,
2893
+ {
2894
+ items: currentQuestion.items,
2895
+ currentOrder: Array.isArray(selectedAnswer) ? selectedAnswer : currentQuestion.items.map((_, i) => i),
2896
+ correctOrder: currentQuestion.correctOrder,
2897
+ showFeedback,
2898
+ onOrderChange: handleAnswerChange
2899
+ }
2900
+ ),
2901
+ currentQuestion.type === "matrix" && currentQuestion.leftItems && currentQuestion.rightItems && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2902
+ MatchingDragDrop,
2903
+ {
2904
+ leftItems: currentQuestion.leftItems,
2905
+ rightItems: currentQuestion.rightItems,
2906
+ currentMatches: typeof selectedAnswer === "object" && selectedAnswer !== null && !Array.isArray(selectedAnswer) ? selectedAnswer : {},
2907
+ correctMatches: currentQuestion.correctMatches,
2908
+ showFeedback,
2909
+ onMatchChange: handleAnswerChange
2910
+ }
2911
+ ),
2912
+ currentQuestion.type === "assessment" && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: (() => {
2913
+ const scaleType = currentQuestion.scaleType || "likert";
2914
+ if (scaleType === "yes-no") {
2915
+ const options = ["Yes", "No"];
2916
+ return options.map((option, idx) => {
2917
+ const isSelected = selectedAnswer === option;
2918
+ let optionStyle = { ...defaultStyles.option };
2919
+ if (isSelected) {
2920
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2921
+ }
2922
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2923
+ "div",
2924
+ {
2925
+ style: {
2926
+ ...optionStyle,
2927
+ cursor: showFeedback ? "default" : "pointer",
2928
+ display: "flex",
2929
+ alignItems: "center",
2930
+ gap: "8px"
2931
+ },
2932
+ onClick: () => !showFeedback && handleAnswerChange(option),
2933
+ "data-testid": `assessment-option-${option.toLowerCase()}`,
2934
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
2935
+ },
2936
+ idx
2937
+ );
2938
+ });
2939
+ }
2940
+ if (scaleType === "rating") {
2941
+ const min = currentQuestion.scaleMin || 1;
2942
+ const max = currentQuestion.scaleMax || 5;
2943
+ const ratings = Array.from({ length: max - min + 1 }, (_, i) => min + i);
2944
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { display: "flex", gap: "8px", flexWrap: "wrap", justifyContent: "center" }, children: ratings.map((rating) => {
2945
+ const isSelected = selectedAnswer === rating;
2946
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2947
+ "button",
2948
+ {
2949
+ onClick: () => !showFeedback && handleAnswerChange(rating),
2950
+ disabled: showFeedback,
2951
+ style: {
2952
+ width: "48px",
2953
+ height: "48px",
2954
+ borderRadius: "50%",
2955
+ border: isSelected ? "2px solid #6721b0" : "2px solid #e5e7eb",
2956
+ backgroundColor: isSelected ? "#f3e8ff" : "#ffffff",
2957
+ cursor: showFeedback ? "not-allowed" : "pointer",
2958
+ fontSize: "18px",
2959
+ fontWeight: "600",
2960
+ color: isSelected ? "#6721b0" : "#374151"
2961
+ },
2962
+ "data-testid": `assessment-rating-${rating}`,
2963
+ children: rating
2964
+ },
2965
+ rating
2966
+ );
2967
+ }) });
2968
+ }
2969
+ const likertOptions = [
2970
+ "Strongly Disagree",
2971
+ "Disagree",
2972
+ "Neutral",
2973
+ "Agree",
2974
+ "Strongly Agree"
2975
+ ];
2976
+ return likertOptions.map((option, idx) => {
2977
+ const isSelected = selectedAnswer === option;
2978
+ let optionStyle = { ...defaultStyles.option };
2979
+ if (isSelected) {
2980
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2981
+ }
2982
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2983
+ "div",
2984
+ {
2985
+ style: {
2986
+ ...optionStyle,
2987
+ cursor: showFeedback ? "default" : "pointer",
2988
+ display: "flex",
2989
+ alignItems: "center",
2990
+ gap: "8px"
2991
+ },
2992
+ onClick: () => !showFeedback && handleAnswerChange(option),
2993
+ "data-testid": `assessment-likert-${idx}`,
2994
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
2995
+ },
2996
+ idx
2997
+ );
2998
+ });
2999
+ })() }),
3000
+ showFeedback && currentAnswerDetail && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
2249
3001
  ...defaultStyles.feedback,
2250
- ...currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
3002
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
2251
3003
  }, children: [
2252
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
3004
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
2253
3005
  ...defaultStyles.feedbackTitle,
2254
- ...currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
2255
- }, children: currentAnswerDetail.isCorrect ? "\u2713 Correct!" : "\u2717 Incorrect" }),
2256
- currentQuestion.explanation && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.feedbackExplanation, children: currentQuestion.explanation })
3006
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackTitleNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
3007
+ }, children: currentQuestion.type === "assessment" ? "\u2713 Response recorded" : currentAnswerDetail.isCorrect ? "\u2713 Correct!" : "\u2717 Incorrect" }),
3008
+ currentQuestion.explanation && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.feedbackExplanation, children: currentQuestion.explanation })
2257
3009
  ] })
2258
3010
  ] }),
2259
- showSkipModal && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
3011
+ showSkipModal && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
2260
3012
  position: "fixed",
2261
3013
  top: 0,
2262
3014
  left: 0,
@@ -2267,7 +3019,7 @@ function QuizPlayer({
2267
3019
  alignItems: "center",
2268
3020
  justifyContent: "center",
2269
3021
  zIndex: 1e3
2270
- }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: {
3022
+ }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
2271
3023
  backgroundColor: "#ffffff",
2272
3024
  borderRadius: "12px",
2273
3025
  padding: "24px",
@@ -2275,10 +3027,10 @@ function QuizPlayer({
2275
3027
  width: "90%",
2276
3028
  boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2277
3029
  }, children: [
2278
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Skip Question" }),
2279
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "Help us improve by telling us why you're skipping this question." }),
2280
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { display: "flex", flexDirection: "column", gap: "8px", marginBottom: "16px" }, children: [
2281
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3030
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Skip Question" }),
3031
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "Help us improve by telling us why you're skipping this question." }),
3032
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", flexDirection: "column", gap: "8px", marginBottom: "16px" }, children: [
3033
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2282
3034
  "button",
2283
3035
  {
2284
3036
  onClick: () => setSelectedSkipReason("question_issue"),
@@ -2299,7 +3051,7 @@ function QuizPlayer({
2299
3051
  children: "Question has an issue"
2300
3052
  }
2301
3053
  ),
2302
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3054
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2303
3055
  "button",
2304
3056
  {
2305
3057
  onClick: () => setSelectedSkipReason("dont_know"),
@@ -2321,9 +3073,9 @@ function QuizPlayer({
2321
3073
  }
2322
3074
  )
2323
3075
  ] }),
2324
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { marginBottom: "16px" }, children: [
2325
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("label", { style: { display: "block", fontSize: "13px", fontWeight: "500", color: "#374151", marginBottom: "6px" }, children: "Additional details (optional)" }),
2326
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3076
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { marginBottom: "16px" }, children: [
3077
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("label", { style: { display: "block", fontSize: "13px", fontWeight: "500", color: "#374151", marginBottom: "6px" }, children: "Additional details (optional)" }),
3078
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2327
3079
  "textarea",
2328
3080
  {
2329
3081
  value: skipComment,
@@ -2344,13 +3096,13 @@ function QuizPlayer({
2344
3096
  "data-testid": "input-skip-comment"
2345
3097
  }
2346
3098
  ),
2347
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3099
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2348
3100
  skipComment.length,
2349
3101
  "/200"
2350
3102
  ] })
2351
3103
  ] }),
2352
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
2353
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3104
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
3105
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2354
3106
  "button",
2355
3107
  {
2356
3108
  onClick: () => {
@@ -2373,7 +3125,7 @@ function QuizPlayer({
2373
3125
  children: "Cancel"
2374
3126
  }
2375
3127
  ),
2376
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3128
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2377
3129
  "button",
2378
3130
  {
2379
3131
  onClick: () => selectedSkipReason && handleSkipQuestion(selectedSkipReason, skipComment),
@@ -2396,7 +3148,7 @@ function QuizPlayer({
2396
3148
  )
2397
3149
  ] })
2398
3150
  ] }) }),
2399
- showReportModal && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
3151
+ showReportModal && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
2400
3152
  position: "fixed",
2401
3153
  top: 0,
2402
3154
  left: 0,
@@ -2407,7 +3159,7 @@ function QuizPlayer({
2407
3159
  alignItems: "center",
2408
3160
  justifyContent: "center",
2409
3161
  zIndex: 1e3
2410
- }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: {
3162
+ }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
2411
3163
  backgroundColor: "#ffffff",
2412
3164
  borderRadius: "12px",
2413
3165
  padding: "24px",
@@ -2415,10 +3167,10 @@ function QuizPlayer({
2415
3167
  width: "90%",
2416
3168
  boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2417
3169
  }, children: [
2418
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Report Question" }),
2419
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "What's wrong with this question?" }),
2420
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { marginBottom: "16px" }, children: [
2421
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3170
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Report Question" }),
3171
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "What's wrong with this question?" }),
3172
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { marginBottom: "16px" }, children: [
3173
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2422
3174
  "textarea",
2423
3175
  {
2424
3176
  value: reportComment,
@@ -2439,13 +3191,13 @@ function QuizPlayer({
2439
3191
  "data-testid": "input-report-comment"
2440
3192
  }
2441
3193
  ),
2442
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3194
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2443
3195
  reportComment.length,
2444
3196
  "/300"
2445
3197
  ] })
2446
3198
  ] }),
2447
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
2448
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3199
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
3200
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2449
3201
  "button",
2450
3202
  {
2451
3203
  onClick: () => {
@@ -2467,7 +3219,7 @@ function QuizPlayer({
2467
3219
  children: "Cancel"
2468
3220
  }
2469
3221
  ),
2470
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3222
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2471
3223
  "button",
2472
3224
  {
2473
3225
  onClick: () => handleReportQuestion(reportComment),
@@ -2490,8 +3242,8 @@ function QuizPlayer({
2490
3242
  )
2491
3243
  ] })
2492
3244
  ] }) }),
2493
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.buttonsColumn, children: [
2494
- showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3245
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.buttonsColumn, children: [
3246
+ showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2495
3247
  "button",
2496
3248
  {
2497
3249
  style: {
@@ -2501,10 +3253,10 @@ function QuizPlayer({
2501
3253
  onClick: handleAddMoreQuestions,
2502
3254
  disabled: isGeneratingExtra,
2503
3255
  "data-testid": "button-add-more-questions",
2504
- children: isGeneratingExtra ? /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
2505
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(Spinner, { size: 16, color: "#9ca3af" }),
3256
+ children: isGeneratingExtra ? /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
3257
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Spinner, { size: 16, color: "#9ca3af" }),
2506
3258
  "Generating Questions..."
2507
- ] }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
3259
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
2508
3260
  "+ Add ",
2509
3261
  questionsToAdd,
2510
3262
  " More Question",
@@ -2512,9 +3264,9 @@ function QuizPlayer({
2512
3264
  ] })
2513
3265
  }
2514
3266
  ),
2515
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
3267
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
2516
3268
  // After viewing feedback
2517
- isLastQuestion ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3269
+ isLastQuestion ? /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2518
3270
  "button",
2519
3271
  {
2520
3272
  style: {
@@ -2524,9 +3276,9 @@ function QuizPlayer({
2524
3276
  onClick: handleSubmit,
2525
3277
  disabled: isSubmitting || isGeneratingExtra,
2526
3278
  "data-testid": "button-submit-quiz",
2527
- children: isSubmitting ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
3279
+ children: isSubmitting ? /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
2528
3280
  }
2529
- ) : /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3281
+ ) : /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2530
3282
  "button",
2531
3283
  {
2532
3284
  style: {
@@ -2540,7 +3292,7 @@ function QuizPlayer({
2540
3292
  )
2541
3293
  ) : (
2542
3294
  // Before checking answer
2543
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3295
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2544
3296
  "button",
2545
3297
  {
2546
3298
  style: {
@@ -2550,13 +3302,13 @@ function QuizPlayer({
2550
3302
  onClick: handleCheckAnswer,
2551
3303
  disabled: isNavigating || selectedAnswer === void 0,
2552
3304
  "data-testid": "button-check-answer",
2553
- children: isNavigating ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
3305
+ children: isNavigating ? /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
2554
3306
  }
2555
3307
  )
2556
3308
  ) })
2557
3309
  ] })
2558
3310
  ] }),
2559
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3311
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2560
3312
  QuestionChatPanel,
2561
3313
  {
2562
3314
  apiClient: apiClient.current,
@@ -2574,7 +3326,7 @@ function QuizPlayer({
2574
3326
  lessonId,
2575
3327
  courseId,
2576
3328
  answerResult: showFeedback && currentAnswerDetail ? {
2577
- wasIncorrect: !currentAnswerDetail.isCorrect,
3329
+ wasIncorrect: currentQuestion.type !== "assessment" && !currentAnswerDetail.isCorrect,
2578
3330
  selectedAnswer: typeof selectedAnswer === "string" ? selectedAnswer : Array.isArray(selectedAnswer) ? selectedAnswer.join(", ") : void 0,
2579
3331
  correctAnswer: typeof currentQuestion.correctAnswer === "string" ? currentQuestion.correctAnswer : Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer.join(", ") : void 0,
2580
3332
  explanation: currentQuestion.explanation
@@ -2586,7 +3338,7 @@ function QuizPlayer({
2586
3338
 
2587
3339
  // src/AttemptViewer.tsx
2588
3340
  var import_react4 = require("react");
2589
- var import_jsx_runtime4 = require("react/jsx-runtime");
3341
+ var import_jsx_runtime5 = require("react/jsx-runtime");
2590
3342
  var defaultStyles2 = {
2591
3343
  container: {
2592
3344
  fontFamily: "system-ui, -apple-system, sans-serif",
@@ -2782,10 +3534,31 @@ var spinnerKeyframes = `
2782
3534
  to { transform: rotate(360deg); }
2783
3535
  }
2784
3536
  `;
2785
- function formatAnswer(answer) {
3537
+ function formatAnswer(answer, questionType, items, leftItems) {
2786
3538
  if (answer === null || answer === void 0) {
2787
3539
  return "No answer";
2788
3540
  }
3541
+ if (questionType === "sorting" && Array.isArray(answer)) {
3542
+ const indices = answer;
3543
+ if (items && items.length > 0) {
3544
+ return indices.map((idx, pos) => `${pos + 1}. ${items[idx] ?? `Item ${idx + 1}`}`).join(", ");
3545
+ }
3546
+ return indices.map((idx, pos) => `Position ${pos + 1}: Item ${idx + 1}`).join(", ");
3547
+ }
3548
+ if (questionType === "matrix" && typeof answer === "object" && !Array.isArray(answer)) {
3549
+ const matches = answer;
3550
+ if (leftItems && leftItems.length > 0) {
3551
+ return leftItems.map((left) => {
3552
+ const right = matches[left];
3553
+ return `${left} \u2192 ${right || "No answer"}`;
3554
+ }).join(", ");
3555
+ }
3556
+ const entries = Object.entries(matches);
3557
+ if (entries.length === 0) {
3558
+ return "No answer";
3559
+ }
3560
+ return entries.map(([left, right]) => `${left} \u2192 ${right || "No answer"}`).join(", ");
3561
+ }
2789
3562
  if (typeof answer === "string") {
2790
3563
  return answer;
2791
3564
  }
@@ -2797,6 +3570,15 @@ function formatAnswer(answer) {
2797
3570
  }
2798
3571
  return String(answer);
2799
3572
  }
3573
+ function formatCorrectSortingAnswer(items, correctOrder) {
3574
+ return correctOrder.map((idx, pos) => `${pos + 1}. ${items[idx] ?? `Item ${idx + 1}`}`).join(", ");
3575
+ }
3576
+ function formatCorrectMatrixAnswer(correctMatches, leftItems) {
3577
+ if (leftItems && leftItems.length > 0) {
3578
+ return leftItems.map((left) => `${left} \u2192 ${correctMatches[left] || "N/A"}`).join(", ");
3579
+ }
3580
+ return Object.entries(correctMatches).map(([left, right]) => `${left} \u2192 ${right}`).join(", ");
3581
+ }
2800
3582
  function AttemptViewer({
2801
3583
  attemptId,
2802
3584
  apiBaseUrl,
@@ -2809,23 +3591,39 @@ function AttemptViewer({
2809
3591
  }) {
2810
3592
  const [attempt, setAttempt] = (0, import_react4.useState)(null);
2811
3593
  const [loading, setLoading] = (0, import_react4.useState)(true);
2812
- const [error, setError] = (0, import_react4.useState)(null);
3594
+ const [errorCode, setErrorCode] = (0, import_react4.useState)(null);
2813
3595
  const [chatHistories, setChatHistories] = (0, import_react4.useState)({});
2814
3596
  const [expandedChats, setExpandedChats] = (0, import_react4.useState)(/* @__PURE__ */ new Set());
3597
+ const [fetchedAttemptId, setFetchedAttemptId] = (0, import_react4.useState)(null);
3598
+ const onErrorRef = (0, import_react4.useRef)(onError);
3599
+ onErrorRef.current = onError;
2815
3600
  (0, import_react4.useEffect)(() => {
3601
+ if (fetchedAttemptId === attemptId) return;
2816
3602
  const apiClient = new QuizApiClient({
2817
3603
  baseUrl: apiBaseUrl,
2818
3604
  authToken
2819
3605
  });
2820
3606
  async function fetchAttempt() {
2821
3607
  setLoading(true);
2822
- setError(null);
3608
+ setErrorCode(null);
2823
3609
  try {
2824
3610
  const response = await fetch(`${apiBaseUrl}/api/external/quiz-attempts/${attemptId}`, {
2825
3611
  headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
2826
3612
  });
2827
3613
  if (!response.ok) {
2828
- throw new Error(`Failed to fetch attempt: ${response.statusText}`);
3614
+ const code = getErrorFromHttpStatus(response.status, "attempt");
3615
+ setErrorCode(code);
3616
+ apiClient.logError({
3617
+ errorCode: code,
3618
+ context: "attempt",
3619
+ resourceId: attemptId,
3620
+ component: "AttemptViewer",
3621
+ message: `HTTP ${response.status}: ${response.statusText}`
3622
+ });
3623
+ onErrorRef.current?.(new Error(`Failed to fetch attempt: ${response.statusText}`));
3624
+ setLoading(false);
3625
+ setFetchedAttemptId(attemptId);
3626
+ return;
2829
3627
  }
2830
3628
  const data = await response.json();
2831
3629
  setAttempt(data);
@@ -2839,14 +3637,23 @@ function AttemptViewer({
2839
3637
  }
2840
3638
  } catch (err) {
2841
3639
  const errorMessage = err instanceof Error ? err.message : "Failed to load attempt";
2842
- setError(errorMessage);
2843
- onError?.(err instanceof Error ? err : new Error(errorMessage));
3640
+ const code = getErrorFromMessage(errorMessage, "attempt");
3641
+ setErrorCode(code);
3642
+ apiClient.logError({
3643
+ errorCode: code,
3644
+ context: "attempt",
3645
+ resourceId: attemptId,
3646
+ component: "AttemptViewer",
3647
+ message: errorMessage
3648
+ });
3649
+ onErrorRef.current?.(err instanceof Error ? err : new Error(errorMessage));
2844
3650
  } finally {
2845
3651
  setLoading(false);
3652
+ setFetchedAttemptId(attemptId);
2846
3653
  }
2847
3654
  }
2848
3655
  fetchAttempt();
2849
- }, [attemptId, apiBaseUrl, authToken, onError, showConversation]);
3656
+ }, [attemptId, apiBaseUrl, authToken, showConversation, fetchedAttemptId]);
2850
3657
  const toggleChatExpanded = (questionId) => {
2851
3658
  setExpandedChats((prev) => {
2852
3659
  const newSet = new Set(prev);
@@ -2860,53 +3667,49 @@ function AttemptViewer({
2860
3667
  };
2861
3668
  const handleRetry = () => {
2862
3669
  setLoading(true);
2863
- setError(null);
3670
+ setErrorCode(null);
2864
3671
  window.location.reload();
2865
3672
  };
2866
3673
  if (loading) {
2867
- return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.container, className, children: [
2868
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: spinnerKeyframes }),
2869
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.loading, children: [
2870
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.spinner }),
2871
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { marginTop: "16px", color: "#6b7280" }, children: "Loading attempt..." })
3674
+ return /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.container, className, children: [
3675
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("style", { children: spinnerKeyframes }),
3676
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.loading, children: [
3677
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.spinner }),
3678
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("p", { style: { marginTop: "16px", color: "#6b7280" }, children: "Loading attempt..." })
2872
3679
  ] })
2873
3680
  ] });
2874
3681
  }
2875
- if (error || !attempt) {
2876
- return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.container, className, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.error, children: [
2877
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { fontSize: "18px", fontWeight: "500" }, children: "Failed to load attempt" }),
2878
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { marginTop: "8px", color: "#6b7280" }, children: error }),
2879
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("button", { style: defaultStyles2.retryButton, onClick: handleRetry, children: "Try Again" })
2880
- ] }) });
3682
+ if (errorCode || !attempt) {
3683
+ return /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.container, className, children: /* @__PURE__ */ (0, import_jsx_runtime5.jsx)(MaintenanceScreen, { errorCode: errorCode || "ATTEMPT_NOT_FOUND" }) });
2881
3684
  }
2882
3685
  const scorePercentage = attempt.score ?? 0;
2883
3686
  const correctCount = attempt.correctAnswers ?? 0;
2884
3687
  const totalQuestions = attempt.totalQuestions;
2885
3688
  const timeSpent = attempt.timeSpentSeconds ?? 0;
2886
- return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.container, className, children: [
2887
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: spinnerKeyframes }),
2888
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.header, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryGrid, children: [
2889
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
2890
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
3689
+ return /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.container, className, children: [
3690
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("style", { children: spinnerKeyframes }),
3691
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.header, children: /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryGrid, children: [
3692
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3693
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
2891
3694
  scorePercentage,
2892
3695
  "%"
2893
3696
  ] }),
2894
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Score" })
3697
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Score" })
2895
3698
  ] }),
2896
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
2897
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
3699
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3700
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
2898
3701
  correctCount,
2899
3702
  "/",
2900
3703
  totalQuestions
2901
3704
  ] }),
2902
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
3705
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
2903
3706
  ] }),
2904
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
2905
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryValue, children: formatTime(timeSpent) }),
2906
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Time" })
3707
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3708
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryValue, children: formatTime(timeSpent) }),
3709
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Time" })
2907
3710
  ] })
2908
3711
  ] }) }),
2909
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
3712
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)(
2910
3713
  "div",
2911
3714
  {
2912
3715
  style: {
@@ -2914,12 +3717,12 @@ function AttemptViewer({
2914
3717
  ...answer.isCorrect ? defaultStyles2.questionCardCorrect : defaultStyles2.questionCardIncorrect
2915
3718
  },
2916
3719
  children: [
2917
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.questionHeader, children: [
2918
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("span", { style: defaultStyles2.questionNumber, children: [
3720
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.questionHeader, children: [
3721
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("span", { style: defaultStyles2.questionNumber, children: [
2919
3722
  "Question ",
2920
3723
  index + 1
2921
3724
  ] }),
2922
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3725
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)(
2923
3726
  "span",
2924
3727
  {
2925
3728
  style: {
@@ -2930,35 +3733,35 @@ function AttemptViewer({
2930
3733
  }
2931
3734
  )
2932
3735
  ] }),
2933
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.questionText, children: answer.questionText }),
2934
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.answerSection, children: [
2935
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: defaultStyles2.answerLabel, children: "Your answer:" }),
2936
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: defaultStyles2.studentAnswer, children: formatAnswer(answer.selectedAnswer) })
3736
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.questionText, children: answer.questionText }),
3737
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.answerSection, children: [
3738
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("span", { style: defaultStyles2.answerLabel, children: "Your answer:" }),
3739
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("span", { style: defaultStyles2.studentAnswer, children: formatAnswer(answer.selectedAnswer, answer.questionType, answer.items, answer.leftItems) })
2937
3740
  ] }),
2938
- !answer.isCorrect && answer.correctAnswer && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.answerSection, children: [
2939
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: defaultStyles2.answerLabel, children: "Correct answer:" }),
2940
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: defaultStyles2.correctAnswer, children: formatAnswer(answer.correctAnswer) })
3741
+ !answer.isCorrect && answer.correctAnswer && /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.answerSection, children: [
3742
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("span", { style: defaultStyles2.answerLabel, children: "Correct answer:" }),
3743
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("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) })
2941
3744
  ] }),
2942
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.points, children: [
3745
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.points, children: [
2943
3746
  answer.pointsEarned,
2944
3747
  " / ",
2945
3748
  answer.points,
2946
3749
  " points"
2947
3750
  ] }),
2948
- showExplanations && answer.explanation && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.explanation, children: [
2949
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("strong", { children: "Explanation:" }),
3751
+ showExplanations && answer.explanation && /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.explanation, children: [
3752
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("strong", { children: "Explanation:" }),
2950
3753
  " ",
2951
3754
  answer.explanation
2952
3755
  ] }),
2953
- showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.chatHistorySection, children: [
2954
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
3756
+ showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.chatHistorySection, children: [
3757
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)(
2955
3758
  "button",
2956
3759
  {
2957
3760
  style: defaultStyles2.chatToggleButton,
2958
3761
  onClick: () => toggleChatExpanded(answer.questionId),
2959
3762
  "data-testid": `button-toggle-chat-${answer.questionId}`,
2960
3763
  children: [
2961
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("path", { d: "M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" }) }),
3764
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("path", { d: "M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" }) }),
2962
3765
  expandedChats.has(answer.questionId) ? "Hide" : "View",
2963
3766
  " Chat History (",
2964
3767
  chatHistories[answer.questionId].messages.length,
@@ -2966,7 +3769,7 @@ function AttemptViewer({
2966
3769
  ]
2967
3770
  }
2968
3771
  ),
2969
- expandedChats.has(answer.questionId) && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.chatMessages, children: chatHistories[answer.questionId].messages.map((msg, msgIndex) => /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3772
+ expandedChats.has(answer.questionId) && /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.chatMessages, children: chatHistories[answer.questionId].messages.map((msg, msgIndex) => /* @__PURE__ */ (0, import_jsx_runtime5.jsx)(
2970
3773
  "div",
2971
3774
  {
2972
3775
  style: {
@@ -2984,15 +3787,621 @@ function AttemptViewer({
2984
3787
  )) })
2985
3788
  ] });
2986
3789
  }
3790
+
3791
+ // src/ErrorTypesPanel.tsx
3792
+ var import_jsx_runtime6 = require("react/jsx-runtime");
3793
+ var panelStyles2 = {
3794
+ container: {
3795
+ fontFamily: "system-ui, -apple-system, sans-serif",
3796
+ padding: "24px",
3797
+ backgroundColor: "#ffffff",
3798
+ borderRadius: "12px",
3799
+ maxWidth: "800px"
3800
+ },
3801
+ header: {
3802
+ marginBottom: "24px"
3803
+ },
3804
+ title: {
3805
+ fontSize: "20px",
3806
+ fontWeight: "600",
3807
+ color: "#111827",
3808
+ marginBottom: "8px"
3809
+ },
3810
+ subtitle: {
3811
+ fontSize: "14px",
3812
+ color: "#6b7280"
3813
+ },
3814
+ section: {
3815
+ marginBottom: "24px"
3816
+ },
3817
+ sectionTitle: {
3818
+ fontSize: "14px",
3819
+ fontWeight: "600",
3820
+ color: "#374151",
3821
+ marginBottom: "12px",
3822
+ textTransform: "uppercase",
3823
+ letterSpacing: "0.05em"
3824
+ },
3825
+ errorList: {
3826
+ display: "flex",
3827
+ flexDirection: "column",
3828
+ gap: "12px"
3829
+ },
3830
+ errorCard: {
3831
+ padding: "16px",
3832
+ backgroundColor: "#f9fafb",
3833
+ borderRadius: "8px",
3834
+ border: "1px solid #e5e7eb"
3835
+ },
3836
+ errorCardBlocking: {
3837
+ borderLeft: "4px solid #ef4444"
3838
+ },
3839
+ errorCardNonBlocking: {
3840
+ borderLeft: "4px solid #f59e0b"
3841
+ },
3842
+ errorHeader: {
3843
+ display: "flex",
3844
+ justifyContent: "space-between",
3845
+ alignItems: "flex-start",
3846
+ marginBottom: "8px"
3847
+ },
3848
+ errorCode: {
3849
+ fontSize: "13px",
3850
+ fontWeight: "600",
3851
+ color: "#1f2937",
3852
+ fontFamily: "monospace",
3853
+ backgroundColor: "#e5e7eb",
3854
+ padding: "2px 8px",
3855
+ borderRadius: "4px"
3856
+ },
3857
+ errorBadge: {
3858
+ fontSize: "11px",
3859
+ fontWeight: "500",
3860
+ padding: "2px 8px",
3861
+ borderRadius: "12px"
3862
+ },
3863
+ blockingBadge: {
3864
+ backgroundColor: "#fee2e2",
3865
+ color: "#dc2626"
3866
+ },
3867
+ nonBlockingBadge: {
3868
+ backgroundColor: "#fef3c7",
3869
+ color: "#d97706"
3870
+ },
3871
+ userMessage: {
3872
+ fontSize: "15px",
3873
+ fontWeight: "500",
3874
+ color: "#111827",
3875
+ marginBottom: "4px"
3876
+ },
3877
+ subMessage: {
3878
+ fontSize: "13px",
3879
+ color: "#6b7280",
3880
+ marginBottom: "8px"
3881
+ },
3882
+ causeLabel: {
3883
+ fontSize: "11px",
3884
+ fontWeight: "600",
3885
+ color: "#9ca3af",
3886
+ textTransform: "uppercase",
3887
+ marginBottom: "4px"
3888
+ },
3889
+ causeText: {
3890
+ fontSize: "13px",
3891
+ color: "#4b5563",
3892
+ fontStyle: "italic"
3893
+ }
3894
+ };
3895
+ function ErrorTypesPanel({ className }) {
3896
+ const blockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => e.isBlocking);
3897
+ const nonBlockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => !e.isBlocking);
3898
+ const renderErrorCard = (error) => /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)(
3899
+ "div",
3900
+ {
3901
+ style: {
3902
+ ...panelStyles2.errorCard,
3903
+ ...error.isBlocking ? panelStyles2.errorCardBlocking : panelStyles2.errorCardNonBlocking
3904
+ },
3905
+ "data-testid": `error-card-${error.code}`,
3906
+ children: [
3907
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.errorHeader, children: [
3908
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("code", { style: panelStyles2.errorCode, children: error.code }),
3909
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)(
3910
+ "span",
3911
+ {
3912
+ style: {
3913
+ ...panelStyles2.errorBadge,
3914
+ ...error.isBlocking ? panelStyles2.blockingBadge : panelStyles2.nonBlockingBadge
3915
+ },
3916
+ children: error.isBlocking ? "Blocking" : "Non-Blocking"
3917
+ }
3918
+ )
3919
+ ] }),
3920
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.userMessage, children: error.userMessage }),
3921
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.subMessage, children: error.subMessage }),
3922
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.causeLabel, children: "Why this happens:" }),
3923
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.causeText, children: error.cause })
3924
+ ]
3925
+ },
3926
+ error.code
3927
+ );
3928
+ return /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.container, className, "data-testid": "error-types-panel", children: [
3929
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.header, children: [
3930
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("h2", { style: panelStyles2.title, children: "Error Types Reference" }),
3931
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("p", { style: panelStyles2.subtitle, children: "List of all error types that can occur in the quiz components, with user-facing messages and technical causes." })
3932
+ ] }),
3933
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.section, children: [
3934
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("h3", { style: panelStyles2.sectionTitle, children: [
3935
+ "Blocking Errors (",
3936
+ blockingErrors.length,
3937
+ ")"
3938
+ ] }),
3939
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("p", { style: { ...panelStyles2.subtitle, marginBottom: "12px" }, children: "These errors prevent the quiz from loading or continuing. Users see a full-screen error message." }),
3940
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.errorList, children: blockingErrors.map(renderErrorCard) })
3941
+ ] }),
3942
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.section, children: [
3943
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("h3", { style: panelStyles2.sectionTitle, children: [
3944
+ "Non-Blocking Errors (",
3945
+ nonBlockingErrors.length,
3946
+ ")"
3947
+ ] }),
3948
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("p", { style: { ...panelStyles2.subtitle, marginBottom: "12px" }, children: "These errors affect specific features but allow the quiz to continue. Users may see a toast notification." }),
3949
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.errorList, children: nonBlockingErrors.map(renderErrorCard) })
3950
+ ] })
3951
+ ] });
3952
+ }
3953
+
3954
+ // src/ErrorLogsPanel.tsx
3955
+ var import_react5 = require("react");
3956
+ var import_jsx_runtime7 = require("react/jsx-runtime");
3957
+ var panelStyles3 = {
3958
+ container: {
3959
+ fontFamily: "system-ui, -apple-system, sans-serif",
3960
+ padding: "24px",
3961
+ backgroundColor: "#ffffff",
3962
+ borderRadius: "12px",
3963
+ maxWidth: "1000px"
3964
+ },
3965
+ header: {
3966
+ marginBottom: "24px"
3967
+ },
3968
+ title: {
3969
+ fontSize: "20px",
3970
+ fontWeight: "600",
3971
+ color: "#111827",
3972
+ marginBottom: "8px"
3973
+ },
3974
+ subtitle: {
3975
+ fontSize: "14px",
3976
+ color: "#6b7280"
3977
+ },
3978
+ tabs: {
3979
+ display: "flex",
3980
+ gap: "8px",
3981
+ marginBottom: "16px",
3982
+ borderBottom: "1px solid #e5e7eb",
3983
+ paddingBottom: "12px"
3984
+ },
3985
+ tab: {
3986
+ padding: "8px 16px",
3987
+ fontSize: "14px",
3988
+ fontWeight: "500",
3989
+ borderRadius: "6px",
3990
+ cursor: "pointer",
3991
+ border: "none",
3992
+ backgroundColor: "transparent",
3993
+ color: "#6b7280"
3994
+ },
3995
+ tabActive: {
3996
+ backgroundColor: "#6721b0",
3997
+ color: "#ffffff"
3998
+ },
3999
+ stats: {
4000
+ display: "flex",
4001
+ gap: "16px",
4002
+ marginBottom: "20px"
4003
+ },
4004
+ statCard: {
4005
+ padding: "12px 16px",
4006
+ backgroundColor: "#f9fafb",
4007
+ borderRadius: "8px",
4008
+ flex: 1
4009
+ },
4010
+ statValue: {
4011
+ fontSize: "24px",
4012
+ fontWeight: "700",
4013
+ color: "#111827"
4014
+ },
4015
+ statLabel: {
4016
+ fontSize: "12px",
4017
+ color: "#6b7280",
4018
+ textTransform: "uppercase",
4019
+ letterSpacing: "0.05em"
4020
+ },
4021
+ errorList: {
4022
+ display: "flex",
4023
+ flexDirection: "column",
4024
+ gap: "12px"
4025
+ },
4026
+ errorCard: {
4027
+ padding: "16px",
4028
+ backgroundColor: "#f9fafb",
4029
+ borderRadius: "8px",
4030
+ border: "1px solid #e5e7eb"
4031
+ },
4032
+ errorCardRecent: {
4033
+ borderLeft: "4px solid #ef4444",
4034
+ backgroundColor: "#fef2f2"
4035
+ },
4036
+ errorCardOld: {
4037
+ borderLeft: "4px solid #9ca3af",
4038
+ opacity: 0.7
4039
+ },
4040
+ errorHeader: {
4041
+ display: "flex",
4042
+ justifyContent: "space-between",
4043
+ alignItems: "flex-start",
4044
+ marginBottom: "8px"
4045
+ },
4046
+ errorLeft: {
4047
+ flex: 1
4048
+ },
4049
+ errorCode: {
4050
+ fontSize: "13px",
4051
+ fontWeight: "600",
4052
+ color: "#1f2937",
4053
+ fontFamily: "monospace",
4054
+ backgroundColor: "#e5e7eb",
4055
+ padding: "2px 8px",
4056
+ borderRadius: "4px",
4057
+ display: "inline-block"
4058
+ },
4059
+ countBadge: {
4060
+ fontSize: "14px",
4061
+ fontWeight: "700",
4062
+ padding: "4px 12px",
4063
+ borderRadius: "16px",
4064
+ backgroundColor: "#ef4444",
4065
+ color: "#ffffff"
4066
+ },
4067
+ countBadgeLow: {
4068
+ backgroundColor: "#f59e0b"
4069
+ },
4070
+ contextBadge: {
4071
+ fontSize: "11px",
4072
+ fontWeight: "500",
4073
+ padding: "2px 8px",
4074
+ borderRadius: "4px",
4075
+ marginLeft: "8px",
4076
+ backgroundColor: "#dbeafe",
4077
+ color: "#1d4ed8"
4078
+ },
4079
+ userMessage: {
4080
+ fontSize: "15px",
4081
+ fontWeight: "500",
4082
+ color: "#111827",
4083
+ marginTop: "8px",
4084
+ marginBottom: "4px"
4085
+ },
4086
+ resourceId: {
4087
+ fontSize: "12px",
4088
+ color: "#6b7280",
4089
+ fontFamily: "monospace",
4090
+ marginTop: "4px"
4091
+ },
4092
+ resourceLink: {
4093
+ color: "#6721b0",
4094
+ textDecoration: "underline",
4095
+ cursor: "pointer"
4096
+ },
4097
+ timestamps: {
4098
+ display: "flex",
4099
+ gap: "16px",
4100
+ marginTop: "8px",
4101
+ fontSize: "12px",
4102
+ color: "#9ca3af"
4103
+ },
4104
+ dismissButton: {
4105
+ padding: "6px 12px",
4106
+ fontSize: "12px",
4107
+ fontWeight: "500",
4108
+ borderRadius: "4px",
4109
+ cursor: "pointer",
4110
+ border: "1px solid #e5e7eb",
4111
+ backgroundColor: "#ffffff",
4112
+ color: "#6b7280",
4113
+ marginTop: "8px"
4114
+ },
4115
+ dismissButtonHover: {
4116
+ backgroundColor: "#f3f4f6"
4117
+ },
4118
+ emptyState: {
4119
+ textAlign: "center",
4120
+ padding: "40px",
4121
+ color: "#9ca3af",
4122
+ fontSize: "14px"
4123
+ },
4124
+ loading: {
4125
+ textAlign: "center",
4126
+ padding: "40px",
4127
+ color: "#6b7280",
4128
+ fontSize: "14px"
4129
+ },
4130
+ pagination: {
4131
+ display: "flex",
4132
+ justifyContent: "center",
4133
+ gap: "8px",
4134
+ marginTop: "20px"
4135
+ },
4136
+ pageButton: {
4137
+ padding: "8px 12px",
4138
+ fontSize: "14px",
4139
+ borderRadius: "4px",
4140
+ cursor: "pointer",
4141
+ border: "1px solid #e5e7eb",
4142
+ backgroundColor: "#ffffff",
4143
+ color: "#374151"
4144
+ },
4145
+ pageButtonDisabled: {
4146
+ opacity: 0.5,
4147
+ cursor: "not-allowed"
4148
+ }
4149
+ };
4150
+ function formatRelativeTime(dateStr) {
4151
+ const date = new Date(dateStr);
4152
+ const now = /* @__PURE__ */ new Date();
4153
+ const diffMs = now.getTime() - date.getTime();
4154
+ const diffMins = Math.floor(diffMs / 6e4);
4155
+ const diffHours = Math.floor(diffMs / 36e5);
4156
+ const diffDays = Math.floor(diffMs / 864e5);
4157
+ if (diffMins < 1) return "Just now";
4158
+ if (diffMins < 60) return `${diffMins}m ago`;
4159
+ if (diffHours < 24) return `${diffHours}h ago`;
4160
+ if (diffDays < 7) return `${diffDays}d ago`;
4161
+ return date.toLocaleDateString();
4162
+ }
4163
+ function isRecent(dateStr) {
4164
+ const date = new Date(dateStr);
4165
+ const now = /* @__PURE__ */ new Date();
4166
+ const diffMs = now.getTime() - date.getTime();
4167
+ const diffHours = diffMs / 36e5;
4168
+ return diffHours < 24;
4169
+ }
4170
+ function ErrorLogsPanel({ apiBaseUrl, authToken, onResourceClick }) {
4171
+ const [logs, setLogs] = (0, import_react5.useState)([]);
4172
+ const [loading, setLoading] = (0, import_react5.useState)(true);
4173
+ const [showDismissed, setShowDismissed] = (0, import_react5.useState)(false);
4174
+ const [page, setPage] = (0, import_react5.useState)(1);
4175
+ const [totalPages, setTotalPages] = (0, import_react5.useState)(1);
4176
+ const [total, setTotal] = (0, import_react5.useState)(0);
4177
+ (0, import_react5.useEffect)(() => {
4178
+ fetchLogs();
4179
+ }, [showDismissed, page]);
4180
+ async function fetchLogs() {
4181
+ setLoading(true);
4182
+ try {
4183
+ const params = new URLSearchParams();
4184
+ params.set("dismissed", showDismissed ? "true" : "false");
4185
+ params.set("page", page.toString());
4186
+ params.set("limit", "20");
4187
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs?${params}`, {
4188
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4189
+ });
4190
+ if (response.ok) {
4191
+ const data = await response.json();
4192
+ setLogs(data.items || []);
4193
+ setTotalPages(data.totalPages || 1);
4194
+ setTotal(data.total || 0);
4195
+ }
4196
+ } catch (err) {
4197
+ console.error("Failed to fetch error logs:", err);
4198
+ } finally {
4199
+ setLoading(false);
4200
+ }
4201
+ }
4202
+ async function handleDismiss(id) {
4203
+ try {
4204
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/dismiss`, {
4205
+ method: "PATCH",
4206
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4207
+ });
4208
+ if (response.ok) {
4209
+ fetchLogs();
4210
+ }
4211
+ } catch (err) {
4212
+ console.error("Failed to dismiss error:", err);
4213
+ }
4214
+ }
4215
+ async function handleUndismiss(id) {
4216
+ try {
4217
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/undismiss`, {
4218
+ method: "PATCH",
4219
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4220
+ });
4221
+ if (response.ok) {
4222
+ fetchLogs();
4223
+ }
4224
+ } catch (err) {
4225
+ console.error("Failed to undismiss error:", err);
4226
+ }
4227
+ }
4228
+ function getErrorDefinition(code) {
4229
+ return ERROR_DEFINITIONS[code];
4230
+ }
4231
+ const totalErrors = logs.reduce((sum, log) => sum + log.count, 0);
4232
+ const recentCount = logs.filter((log) => isRecent(log.lastSeenAt)).length;
4233
+ return /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.container, "data-testid": "error-logs-panel", children: [
4234
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.header, children: [
4235
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("h2", { style: panelStyles3.title, children: "Error Tracking" }),
4236
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("p", { style: panelStyles3.subtitle, children: "Aggregated errors from QuizPlayer and AttemptViewer components, sorted by occurrence count" })
4237
+ ] }),
4238
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.tabs, children: [
4239
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4240
+ "button",
4241
+ {
4242
+ style: {
4243
+ ...panelStyles3.tab,
4244
+ ...showDismissed ? {} : panelStyles3.tabActive
4245
+ },
4246
+ onClick: () => {
4247
+ setShowDismissed(false);
4248
+ setPage(1);
4249
+ },
4250
+ "data-testid": "tab-active-errors",
4251
+ children: "Active Errors"
4252
+ }
4253
+ ),
4254
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4255
+ "button",
4256
+ {
4257
+ style: {
4258
+ ...panelStyles3.tab,
4259
+ ...showDismissed ? panelStyles3.tabActive : {}
4260
+ },
4261
+ onClick: () => {
4262
+ setShowDismissed(true);
4263
+ setPage(1);
4264
+ },
4265
+ "data-testid": "tab-dismissed-errors",
4266
+ children: "Dismissed"
4267
+ }
4268
+ )
4269
+ ] }),
4270
+ !showDismissed && /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.stats, children: [
4271
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.statCard, children: [
4272
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statValue, children: total }),
4273
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statLabel, children: "Unique Errors" })
4274
+ ] }),
4275
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.statCard, children: [
4276
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statValue, children: totalErrors }),
4277
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statLabel, children: "Total Occurrences" })
4278
+ ] }),
4279
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.statCard, children: [
4280
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: { ...panelStyles3.statValue, color: recentCount > 0 ? "#ef4444" : "#22c55e" }, children: recentCount }),
4281
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statLabel, children: "Last 24h" })
4282
+ ] })
4283
+ ] }),
4284
+ loading ? /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.loading, children: "Loading..." }) : logs.length === 0 ? /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.emptyState, children: showDismissed ? "No dismissed errors" : "No active errors" }) : /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.errorList, children: logs.map((log) => {
4285
+ const def = getErrorDefinition(log.errorCode);
4286
+ const recent = isRecent(log.lastSeenAt);
4287
+ return /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)(
4288
+ "div",
4289
+ {
4290
+ style: {
4291
+ ...panelStyles3.errorCard,
4292
+ ...recent ? panelStyles3.errorCardRecent : panelStyles3.errorCardOld
4293
+ },
4294
+ "data-testid": `error-log-${log.id}`,
4295
+ children: [
4296
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.errorHeader, children: [
4297
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.errorLeft, children: [
4298
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("span", { style: panelStyles3.errorCode, children: log.errorCode }),
4299
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("span", { style: panelStyles3.contextBadge, children: log.context }),
4300
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("span", { style: { ...panelStyles3.contextBadge, backgroundColor: "#f3e8ff", color: "#7c3aed" }, children: log.component })
4301
+ ] }),
4302
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)(
4303
+ "span",
4304
+ {
4305
+ style: {
4306
+ ...panelStyles3.countBadge,
4307
+ ...log.count < 10 ? panelStyles3.countBadgeLow : {}
4308
+ },
4309
+ children: [
4310
+ log.count,
4311
+ "x"
4312
+ ]
4313
+ }
4314
+ )
4315
+ ] }),
4316
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.userMessage, children: def?.userMessage || log.lastMessage || "Unknown error" }),
4317
+ log.resourceId && /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.resourceId, children: [
4318
+ "Resource:",
4319
+ " ",
4320
+ onResourceClick ? /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4321
+ "span",
4322
+ {
4323
+ style: panelStyles3.resourceLink,
4324
+ onClick: () => onResourceClick(log.resourceId, log.context),
4325
+ "data-testid": `link-resource-${log.id}`,
4326
+ children: log.resourceId
4327
+ }
4328
+ ) : log.resourceId
4329
+ ] }),
4330
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.timestamps, children: [
4331
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("span", { children: [
4332
+ "First: ",
4333
+ formatRelativeTime(log.firstSeenAt)
4334
+ ] }),
4335
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("span", { children: [
4336
+ "Last: ",
4337
+ formatRelativeTime(log.lastSeenAt)
4338
+ ] })
4339
+ ] }),
4340
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4341
+ "button",
4342
+ {
4343
+ style: panelStyles3.dismissButton,
4344
+ onClick: () => showDismissed ? handleUndismiss(log.id) : handleDismiss(log.id),
4345
+ "data-testid": `button-dismiss-${log.id}`,
4346
+ children: showDismissed ? "Restore" : "Dismiss"
4347
+ }
4348
+ )
4349
+ ]
4350
+ },
4351
+ log.id
4352
+ );
4353
+ }) }),
4354
+ totalPages > 1 && /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.pagination, children: [
4355
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4356
+ "button",
4357
+ {
4358
+ style: {
4359
+ ...panelStyles3.pageButton,
4360
+ ...page <= 1 ? panelStyles3.pageButtonDisabled : {}
4361
+ },
4362
+ onClick: () => setPage((p) => Math.max(1, p - 1)),
4363
+ disabled: page <= 1,
4364
+ "data-testid": "button-prev-page",
4365
+ children: "Previous"
4366
+ }
4367
+ ),
4368
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("span", { style: { padding: "8px 12px", color: "#6b7280" }, children: [
4369
+ "Page ",
4370
+ page,
4371
+ " of ",
4372
+ totalPages
4373
+ ] }),
4374
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4375
+ "button",
4376
+ {
4377
+ style: {
4378
+ ...panelStyles3.pageButton,
4379
+ ...page >= totalPages ? panelStyles3.pageButtonDisabled : {}
4380
+ },
4381
+ onClick: () => setPage((p) => Math.min(totalPages, p + 1)),
4382
+ disabled: page >= totalPages,
4383
+ "data-testid": "button-next-page",
4384
+ children: "Next"
4385
+ }
4386
+ )
4387
+ ] })
4388
+ ] });
4389
+ }
2987
4390
  // Annotate the CommonJS export names for ESM import in node:
2988
4391
  0 && (module.exports = {
2989
4392
  AttemptViewer,
4393
+ ERROR_DEFINITIONS,
4394
+ ErrorLogsPanel,
4395
+ ErrorTypesPanel,
4396
+ MaintenanceScreen,
2990
4397
  QuizApiClient,
2991
4398
  QuizPlayer,
2992
4399
  TextToSpeech,
2993
4400
  calculateScore,
2994
4401
  checkAnswer,
2995
4402
  createAnswerDetail,
2996
- formatTime
4403
+ formatTime,
4404
+ getErrorFromHttpStatus,
4405
+ getErrorFromMessage
2997
4406
  });
2998
4407
  //# sourceMappingURL=index.js.map