@schoolio/player 1.4.4 → 1.4.6

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
@@ -1056,8 +1069,191 @@ function QuestionChatPanel({
1056
1069
  ] });
1057
1070
  }
1058
1071
 
1059
- // 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
1060
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");
1061
1257
  var defaultStyles = {
1062
1258
  container: {
1063
1259
  fontFamily: "system-ui, -apple-system, sans-serif",
@@ -1126,7 +1322,9 @@ var defaultStyles = {
1126
1322
  option: {
1127
1323
  width: "100%",
1128
1324
  padding: "12px 16px",
1129
- border: "2px solid #e5e7eb",
1325
+ borderWidth: "2px",
1326
+ borderStyle: "solid",
1327
+ borderColor: "#e5e7eb",
1130
1328
  borderRadius: "8px",
1131
1329
  cursor: "pointer",
1132
1330
  transition: "all 0.2s ease",
@@ -1420,7 +1618,7 @@ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOr
1420
1618
  setDraggedIndex(null);
1421
1619
  setDragOverIndex(null);
1422
1620
  };
1423
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: currentOrder.map((itemIndex, position) => {
1621
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentOrder.map((itemIndex, position) => {
1424
1622
  const isCorrectPosition = correctOrder?.[position] === itemIndex;
1425
1623
  const isDragging = draggedIndex === position;
1426
1624
  const isDragOver = dragOverIndex === position;
@@ -1443,7 +1641,7 @@ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOr
1443
1641
  } else if (isDragOver) {
1444
1642
  itemStyle = { ...itemStyle, borderColor: "#6366f1", backgroundColor: "#eef2ff" };
1445
1643
  }
1446
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
1644
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1447
1645
  "div",
1448
1646
  {
1449
1647
  style: itemStyle,
@@ -1455,34 +1653,34 @@ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOr
1455
1653
  onDrop: (e) => handleDrop(e, position),
1456
1654
  onDragEnd: handleDragEnd,
1457
1655
  children: [
1458
- !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
1656
+ !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
1459
1657
  cursor: "grab",
1460
1658
  padding: "4px",
1461
1659
  display: "flex",
1462
1660
  alignItems: "center",
1463
1661
  color: "#9ca3af"
1464
- }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1465
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "9", cy: "5", r: "1" }),
1466
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "9", cy: "12", r: "1" }),
1467
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "9", cy: "19", r: "1" }),
1468
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "15", cy: "5", r: "1" }),
1469
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "15", cy: "12", r: "1" }),
1470
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "15", cy: "19", r: "1" })
1662
+ }, 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: [
1663
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "5", r: "1" }),
1664
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "12", r: "1" }),
1665
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "19", r: "1" }),
1666
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "5", r: "1" }),
1667
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "12", r: "1" }),
1668
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "19", r: "1" })
1471
1669
  ] }) }),
1472
- showFeedback && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
1670
+ showFeedback && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
1473
1671
  display: "flex",
1474
1672
  alignItems: "center",
1475
1673
  color: isCorrectPosition ? "#22c55e" : "#ef4444"
1476
- }, children: isCorrectPosition ? /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1477
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("path", { d: "M22 11.08V12a10 10 0 1 1-5.93-9.14" }),
1478
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polyline", { points: "22 4 12 14.01 9 11.01" })
1479
- ] }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1480
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "12", cy: "12", r: "10" }),
1481
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "15", y1: "9", x2: "9", y2: "15" }),
1482
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "9", y1: "9", x2: "15", y2: "15" })
1674
+ }, 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: [
1675
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("path", { d: "M22 11.08V12a10 10 0 1 1-5.93-9.14" }),
1676
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("polyline", { points: "22 4 12 14.01 9 11.01" })
1677
+ ] }) : /* @__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: [
1678
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "12", cy: "12", r: "10" }),
1679
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "15", y1: "9", x2: "9", y2: "15" }),
1680
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "9", y1: "9", x2: "15", y2: "15" })
1483
1681
  ] }) }),
1484
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: items[itemIndex] }),
1485
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(TextToSpeech, { text: items[itemIndex], size: "sm" }) })
1682
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: items[itemIndex] }),
1683
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(TextToSpeech, { text: items[itemIndex], size: "sm" }) })
1486
1684
  ]
1487
1685
  },
1488
1686
  itemIndex
@@ -1536,9 +1734,9 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1536
1734
  delete newMatches[leftItem];
1537
1735
  onMatchChange(newMatches);
1538
1736
  };
1539
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { display: "flex", gap: "24px", flexWrap: "wrap" }, children: [
1540
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1541
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Match these items:" }),
1737
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", gap: "24px", flexWrap: "wrap" }, children: [
1738
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1739
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Match these items:" }),
1542
1740
  leftItems.map((leftItem, idx) => {
1543
1741
  const matchedRight = currentMatches[leftItem];
1544
1742
  const correctMatch = correctMatches?.[leftItem];
@@ -1549,7 +1747,9 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1549
1747
  alignItems: "center",
1550
1748
  gap: "12px",
1551
1749
  padding: "12px 16px",
1552
- border: "2px dashed #e5e7eb",
1750
+ borderWidth: "2px",
1751
+ borderStyle: "dashed",
1752
+ borderColor: "#e5e7eb",
1553
1753
  borderRadius: "8px",
1554
1754
  backgroundColor: "#ffffff",
1555
1755
  minHeight: "56px",
@@ -1567,7 +1767,7 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1567
1767
  } else if (matchedRight) {
1568
1768
  rowStyle = { ...rowStyle, borderStyle: "solid", borderColor: "#22c55e" };
1569
1769
  }
1570
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
1770
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1571
1771
  "div",
1572
1772
  {
1573
1773
  style: rowStyle,
@@ -1576,9 +1776,9 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1576
1776
  onDragLeave: handleDragLeave,
1577
1777
  onDrop: (e) => handleDrop(e, leftItem),
1578
1778
  children: [
1579
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1, fontWeight: "500" }, children: leftItem }),
1580
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { color: "#6b7280" }, children: "\u2192" }),
1581
- matchedRight ? /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: {
1779
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1, fontWeight: "500" }, children: leftItem }),
1780
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { color: "#6b7280" }, children: "\u2192" }),
1781
+ matchedRight ? /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
1582
1782
  display: "flex",
1583
1783
  alignItems: "center",
1584
1784
  gap: "8px",
@@ -1587,8 +1787,8 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1587
1787
  borderRadius: "6px",
1588
1788
  fontSize: "14px"
1589
1789
  }, children: [
1590
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { children: matchedRight }),
1591
- !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
1790
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { children: matchedRight }),
1791
+ !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1592
1792
  "button",
1593
1793
  {
1594
1794
  onClick: () => handleClearMatch(leftItem),
@@ -1601,18 +1801,18 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1601
1801
  color: "#6b7280"
1602
1802
  },
1603
1803
  "aria-label": "Remove match",
1604
- children: /* @__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: [
1605
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1606
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1804
+ 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: [
1805
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1806
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1607
1807
  ] })
1608
1808
  }
1609
1809
  ),
1610
- showFeedback && (isCorrect ? /* @__PURE__ */ (0, import_jsx_runtime3.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_runtime3.jsx)("polyline", { points: "20 6 9 17 4 12" }) }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "#ef4444", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1611
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1612
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1810
+ 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: [
1811
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1812
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1613
1813
  ] }))
1614
- ] }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { color: "#9ca3af", fontSize: "14px", fontStyle: "italic" }, children: "Drop here" }),
1615
- showFeedback && !isCorrect && correctMatch && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("span", { style: { color: "#166534", fontSize: "13px", marginLeft: "8px" }, children: [
1814
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { color: "#9ca3af", fontSize: "14px", fontStyle: "italic" }, children: "Drop here" }),
1815
+ showFeedback && !isCorrect && correctMatch && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("span", { style: { color: "#166534", fontSize: "13px", marginLeft: "8px" }, children: [
1616
1816
  "(Correct: ",
1617
1817
  correctMatch,
1618
1818
  ")"
@@ -1623,16 +1823,18 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1623
1823
  );
1624
1824
  })
1625
1825
  ] }),
1626
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1627
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Drag to match:" }),
1826
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1827
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Drag to match:" }),
1628
1828
  unmatchedRightItems.length > 0 ? unmatchedRightItems.map((rightItem, idx) => {
1629
1829
  const isDragging = draggedItem === rightItem;
1630
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
1830
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1631
1831
  "div",
1632
1832
  {
1633
1833
  style: {
1634
1834
  padding: "12px 16px",
1635
- border: "2px solid #e5e7eb",
1835
+ borderWidth: "2px",
1836
+ borderStyle: "solid",
1837
+ borderColor: "#e5e7eb",
1636
1838
  borderRadius: "8px",
1637
1839
  backgroundColor: "#ffffff",
1638
1840
  cursor: showFeedback ? "default" : "grab",
@@ -1647,20 +1849,20 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1647
1849
  onDragEnd: handleDragEnd,
1648
1850
  "data-testid": `draggable-right-${idx}`,
1649
1851
  children: [
1650
- !showFeedback && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { color: "#9ca3af", display: "flex" }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1651
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "9", cy: "5", r: "1" }),
1652
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "9", cy: "12", r: "1" }),
1653
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "9", cy: "19", r: "1" }),
1654
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "15", cy: "5", r: "1" }),
1655
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "15", cy: "12", r: "1" }),
1656
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "15", cy: "19", r: "1" })
1852
+ !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: [
1853
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "5", r: "1" }),
1854
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "12", r: "1" }),
1855
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "9", cy: "19", r: "1" }),
1856
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "5", r: "1" }),
1857
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "12", r: "1" }),
1858
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("circle", { cx: "15", cy: "19", r: "1" })
1657
1859
  ] }) }),
1658
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: rightItem })
1860
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: rightItem })
1659
1861
  ]
1660
1862
  },
1661
1863
  idx
1662
1864
  );
1663
- }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
1865
+ }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
1664
1866
  padding: "16px",
1665
1867
  textAlign: "center",
1666
1868
  color: "#22c55e",
@@ -1670,7 +1872,7 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1670
1872
  ] });
1671
1873
  }
1672
1874
  function Spinner({ size = 16, color = "#ffffff" }) {
1673
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
1875
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1674
1876
  "span",
1675
1877
  {
1676
1878
  style: {
@@ -1682,7 +1884,7 @@ function Spinner({ size = 16, color = "#ffffff" }) {
1682
1884
  borderRadius: "50%",
1683
1885
  animation: "spin 0.8s linear infinite"
1684
1886
  },
1685
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1887
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1686
1888
  }
1687
1889
  );
1688
1890
  }
@@ -1700,7 +1902,8 @@ function QuizPlayer({
1700
1902
  onProgress,
1701
1903
  onGenerateMoreQuestions,
1702
1904
  className,
1703
- forceNewAttempt = true
1905
+ forceNewAttempt = true,
1906
+ hideScore = false
1704
1907
  }) {
1705
1908
  const [quiz, setQuiz] = (0, import_react3.useState)(null);
1706
1909
  const [attempt, setAttempt] = (0, import_react3.useState)(null);
@@ -1711,10 +1914,14 @@ function QuizPlayer({
1711
1914
  const [isNavigating, setIsNavigating] = (0, import_react3.useState)(false);
1712
1915
  const [isCompleted, setIsCompleted] = (0, import_react3.useState)(false);
1713
1916
  const [result, setResult] = (0, import_react3.useState)(null);
1714
- const [error, setError] = (0, import_react3.useState)(null);
1917
+ const [errorCode, setErrorCode] = (0, import_react3.useState)(null);
1715
1918
  const [isLoading, setIsLoading] = (0, import_react3.useState)(true);
1716
1919
  const [elapsedSeconds, setElapsedSeconds] = (0, import_react3.useState)(0);
1717
1920
  const [showIntro, setShowIntro] = (0, import_react3.useState)(true);
1921
+ const [showResumeChoice, setShowResumeChoice] = (0, import_react3.useState)(false);
1922
+ const [hasExistingProgress, setHasExistingProgress] = (0, import_react3.useState)(false);
1923
+ const [existingProgressCount, setExistingProgressCount] = (0, import_react3.useState)(0);
1924
+ const [isStartingFresh, setIsStartingFresh] = (0, import_react3.useState)(false);
1718
1925
  const [timerStarted, setTimerStarted] = (0, import_react3.useState)(false);
1719
1926
  const [showFeedback, setShowFeedback] = (0, import_react3.useState)(false);
1720
1927
  const [currentAnswerDetail, setCurrentAnswerDetail] = (0, import_react3.useState)(null);
@@ -1728,6 +1935,7 @@ function QuizPlayer({
1728
1935
  const [showReportModal, setShowReportModal] = (0, import_react3.useState)(false);
1729
1936
  const [isReporting, setIsReporting] = (0, import_react3.useState)(false);
1730
1937
  const [reportComment, setReportComment] = (0, import_react3.useState)("");
1938
+ const [retryKey, setRetryKey] = (0, import_react3.useState)(0);
1731
1939
  const apiClient = (0, import_react3.useRef)(null);
1732
1940
  const timerRef = (0, import_react3.useRef)(null);
1733
1941
  const startTimeRef = (0, import_react3.useRef)(0);
@@ -1749,7 +1957,7 @@ function QuizPlayer({
1749
1957
  if (!apiClient.current) return;
1750
1958
  try {
1751
1959
  setIsLoading(true);
1752
- setError(null);
1960
+ setErrorCode(null);
1753
1961
  const quizData = await apiClient.current.getQuiz(quizId);
1754
1962
  setQuiz(quizData);
1755
1963
  const attemptData = await apiClient.current.createAttempt({
@@ -1762,7 +1970,10 @@ function QuizPlayer({
1762
1970
  forceNew: forceNewAttempt
1763
1971
  });
1764
1972
  setAttempt(attemptData);
1765
- if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0) {
1973
+ if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0 && attemptData.status === "in_progress") {
1974
+ setHasExistingProgress(true);
1975
+ setExistingProgressCount(attemptData.answers.length);
1976
+ setShowResumeChoice(true);
1766
1977
  setAnswersDetail(attemptData.answers);
1767
1978
  const answersMap = /* @__PURE__ */ new Map();
1768
1979
  attemptData.answers.forEach((a) => {
@@ -1785,15 +1996,23 @@ function QuizPlayer({
1785
1996
  setIsLoading(false);
1786
1997
  } catch (err) {
1787
1998
  const message = err instanceof Error ? err.message : "Failed to load quiz";
1788
- setError(message);
1999
+ const code = getErrorFromMessage(message, "quiz");
2000
+ setErrorCode(code);
2001
+ apiClient.current?.logError({
2002
+ errorCode: code,
2003
+ context: "quiz",
2004
+ resourceId: quizId,
2005
+ component: "QuizPlayer",
2006
+ message
2007
+ });
1789
2008
  setIsLoading(false);
1790
2009
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1791
2010
  }
1792
2011
  }
1793
2012
  initialize();
1794
- }, [quizId, lessonId, assignLessonId, courseId, childId, parentId, forceNewAttempt]);
2013
+ }, [quizId, lessonId, assignLessonId, courseId, childId, parentId, forceNewAttempt, retryKey]);
1795
2014
  (0, import_react3.useEffect)(() => {
1796
- if (timerStarted && !isCompleted && !error) {
2015
+ if (timerStarted && !isCompleted && !errorCode) {
1797
2016
  startTimeRef.current = Date.now();
1798
2017
  timerRef.current = setInterval(() => {
1799
2018
  setElapsedSeconds(Math.floor((Date.now() - startTimeRef.current) / 1e3));
@@ -1804,11 +2023,62 @@ function QuizPlayer({
1804
2023
  clearInterval(timerRef.current);
1805
2024
  }
1806
2025
  };
1807
- }, [timerStarted, isCompleted, error]);
2026
+ }, [timerStarted, isCompleted, errorCode]);
1808
2027
  const handleStart = (0, import_react3.useCallback)(() => {
1809
2028
  setShowIntro(false);
2029
+ setShowResumeChoice(false);
1810
2030
  setTimerStarted(true);
1811
2031
  }, []);
2032
+ const handleResumePrevious = (0, import_react3.useCallback)(() => {
2033
+ if (quiz && answers.size > 0) {
2034
+ const answeredIds = new Set(answers.keys());
2035
+ const allQs = [...quiz.questions, ...extraQuestions].filter((q) => !skippedQuestionIds.has(q.id));
2036
+ let resumeIndex = 0;
2037
+ for (let i = 0; i < allQs.length; i++) {
2038
+ if (!answeredIds.has(allQs[i].id)) {
2039
+ resumeIndex = i;
2040
+ break;
2041
+ }
2042
+ resumeIndex = allQs.length - 1;
2043
+ }
2044
+ setCurrentQuestionIndex(resumeIndex);
2045
+ }
2046
+ setShowIntro(false);
2047
+ setShowResumeChoice(false);
2048
+ setTimerStarted(true);
2049
+ }, [quiz, answers, extraQuestions, skippedQuestionIds]);
2050
+ const handleStartFresh = (0, import_react3.useCallback)(async () => {
2051
+ if (!apiClient.current || !attempt) return;
2052
+ setIsStartingFresh(true);
2053
+ try {
2054
+ await apiClient.current.updateAttempt(attempt.id, {
2055
+ status: "abandoned"
2056
+ });
2057
+ const newAttemptData = await apiClient.current.createAttempt({
2058
+ quizId,
2059
+ lessonId,
2060
+ assignLessonId,
2061
+ courseId,
2062
+ childId,
2063
+ parentId,
2064
+ forceNew: true
2065
+ });
2066
+ setAttempt(newAttemptData);
2067
+ setAnswers(/* @__PURE__ */ new Map());
2068
+ setAnswersDetail([]);
2069
+ setCurrentQuestionIndex(0);
2070
+ setHasExistingProgress(false);
2071
+ setExistingProgressCount(0);
2072
+ setShowResumeChoice(false);
2073
+ setShowIntro(false);
2074
+ setTimerStarted(true);
2075
+ } catch (err) {
2076
+ const message = err instanceof Error ? err.message : "Failed to start fresh";
2077
+ onErrorRef.current?.(new Error(message));
2078
+ } finally {
2079
+ setIsStartingFresh(false);
2080
+ }
2081
+ }, [attempt, quizId, lessonId, assignLessonId, courseId, childId, parentId]);
1812
2082
  (0, import_react3.useEffect)(() => {
1813
2083
  setShowFeedback(false);
1814
2084
  setCurrentAnswerDetail(null);
@@ -1928,7 +2198,15 @@ function QuizPlayer({
1928
2198
  onCompleteRef.current?.(quizResult);
1929
2199
  } catch (err) {
1930
2200
  const message = err instanceof Error ? err.message : "Failed to submit quiz";
1931
- setError(message);
2201
+ const code = getErrorFromMessage(message, "quiz");
2202
+ setErrorCode(code);
2203
+ apiClient.current?.logError({
2204
+ errorCode: code,
2205
+ context: "quiz",
2206
+ resourceId: quizId,
2207
+ component: "QuizPlayer",
2208
+ message
2209
+ });
1932
2210
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1933
2211
  } finally {
1934
2212
  setIsSubmitting(false);
@@ -2006,14 +2284,43 @@ function QuizPlayer({
2006
2284
  setIsReporting(false);
2007
2285
  }
2008
2286
  }, [currentQuestion, apiClient, attempt, quiz, childId, parentId, lessonId, courseId, assignLessonId]);
2287
+ const handleRetryQuiz = (0, import_react3.useCallback)(() => {
2288
+ setCurrentQuestionIndex(0);
2289
+ setAnswers(/* @__PURE__ */ new Map());
2290
+ setAnswersDetail([]);
2291
+ setIsCompleted(false);
2292
+ setResult(null);
2293
+ setErrorCode(null);
2294
+ setShowIntro(true);
2295
+ setShowFeedback(false);
2296
+ setCurrentAnswerDetail(null);
2297
+ setTimerStarted(false);
2298
+ setElapsedSeconds(0);
2299
+ startTimeRef.current = 0;
2300
+ setExtraQuestions([]);
2301
+ setSkippedQuestionIds(/* @__PURE__ */ new Set());
2302
+ setAttempt(null);
2303
+ setShowResumeChoice(false);
2304
+ setHasExistingProgress(false);
2305
+ setExistingProgressCount(0);
2306
+ setIsSubmitting(false);
2307
+ setIsNavigating(false);
2308
+ setIsGeneratingExtra(false);
2309
+ setIsStartingFresh(false);
2310
+ setShowSkipModal(false);
2311
+ setIsSkipping(false);
2312
+ setSkipComment("");
2313
+ setSelectedSkipReason(null);
2314
+ setShowReportModal(false);
2315
+ setIsReporting(false);
2316
+ setReportComment("");
2317
+ setRetryKey((prev) => prev + 1);
2318
+ }, []);
2009
2319
  if (isLoading) {
2010
- 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..." }) });
2320
+ 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..." }) });
2011
2321
  }
2012
- if (error) {
2013
- 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: [
2014
- "Error: ",
2015
- error
2016
- ] }) }) });
2322
+ if (errorCode) {
2323
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(MaintenanceScreen, { errorCode }) });
2017
2324
  }
2018
2325
  if (isCompleted && result) {
2019
2326
  const percentage = result.totalQuestions > 0 ? Math.round(result.correctAnswers / result.totalQuestions * 100) : 0;
@@ -2071,7 +2378,7 @@ function QuizPlayer({
2071
2378
  rotation: Math.random() * 360,
2072
2379
  size: 6 + Math.random() * 8
2073
2380
  }));
2074
- const StarIcon = ({ filled, delay }) => /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2381
+ const StarIcon = ({ filled, delay }) => /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2075
2382
  "svg",
2076
2383
  {
2077
2384
  width: "36",
@@ -2082,7 +2389,7 @@ function QuizPlayer({
2082
2389
  animationDelay: `${delay}s`,
2083
2390
  opacity: 0
2084
2391
  },
2085
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2392
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2086
2393
  "path",
2087
2394
  {
2088
2395
  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",
@@ -2093,62 +2400,23 @@ function QuizPlayer({
2093
2400
  )
2094
2401
  }
2095
2402
  );
2096
- const MascotOwl = ({ mood }) => {
2097
- const getEyeExpression = () => {
2098
- switch (mood) {
2099
- case "celebrating":
2100
- return { leftEye: ">", rightEye: "<", pupilY: 42 };
2101
- // Squinting happy
2102
- case "happy":
2103
- return { leftEye: null, rightEye: null, pupilY: 42 };
2104
- // Normal happy
2105
- case "encouraging":
2106
- return { leftEye: null, rightEye: null, pupilY: 44 };
2107
- // Looking down warmly
2108
- default:
2109
- return { leftEye: null, rightEye: null, pupilY: 42 };
2110
- }
2111
- };
2112
- const eyeExpr = getEyeExpression();
2113
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2114
- "svg",
2403
+ const Mascot = ({ mood }) => {
2404
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2405
+ "img",
2115
2406
  {
2116
- width: "120",
2117
- height: "120",
2118
- viewBox: "0 0 100 100",
2407
+ src: astronautImage,
2408
+ alt: "Astronaut mascot",
2409
+ width: "180",
2410
+ height: "180",
2119
2411
  style: {
2120
- animation: mood === "celebrating" ? "bounce 0.6s ease-in-out infinite" : "gentleBob 2s ease-in-out infinite"
2121
- },
2122
- children: [
2123
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "50", cy: "60", rx: "35", ry: "30", fill: "#8b5cf6" }),
2124
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "50", cy: "65", rx: "25", ry: "20", fill: "#c4b5fd" }),
2125
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("circle", { cx: "50", cy: "35", r: "28", fill: "#a78bfa" }),
2126
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "28,15 35,30 22,28", fill: "#7c3aed" }),
2127
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "72,15 65,30 78,28", fill: "#7c3aed" }),
2128
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "38", cy: "38", rx: "10", ry: "12", fill: "white" }),
2129
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "62", cy: "38", rx: "10", ry: "12", fill: "white" }),
2130
- 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" }),
2131
- 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" }),
2132
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "50,45 45,52 55,52", fill: "#fbbf24" }),
2133
- (mood === "celebrating" || mood === "happy") && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
2134
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "28", cy: "45", rx: "5", ry: "3", fill: "#fda4af", opacity: "0.6" }),
2135
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "72", cy: "45", rx: "5", ry: "3", fill: "#fda4af", opacity: "0.6" })
2136
- ] }),
2137
- mood === "celebrating" ? /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
2138
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "18", cy: "55", rx: "8", ry: "15", fill: "#7c3aed", transform: "rotate(-30 18 55)" }),
2139
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "82", cy: "55", rx: "8", ry: "15", fill: "#7c3aed", transform: "rotate(30 82 55)" })
2140
- ] }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
2141
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "20", cy: "60", rx: "8", ry: "15", fill: "#7c3aed" }),
2142
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "80", cy: "60", rx: "8", ry: "15", fill: "#7c3aed" })
2143
- ] }),
2144
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "40", cy: "88", rx: "8", ry: "4", fill: "#fbbf24" }),
2145
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("ellipse", { cx: "60", cy: "88", rx: "8", ry: "4", fill: "#fbbf24" })
2146
- ]
2412
+ animation: mood === "celebrating" ? "bounce 0.6s ease-in-out infinite" : "gentleBob 2s ease-in-out infinite",
2413
+ objectFit: "contain"
2414
+ }
2147
2415
  }
2148
2416
  );
2149
2417
  };
2150
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { className, style: defaultStyles.container, children: [
2151
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("style", { children: `
2418
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { className, style: defaultStyles.container, children: [
2419
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: `
2152
2420
  @keyframes confettiFall {
2153
2421
  0% {
2154
2422
  transform: translateY(-10px) rotate(0deg);
@@ -2211,9 +2479,19 @@ function QuizPlayer({
2211
2479
  opacity: 1;
2212
2480
  }
2213
2481
  }
2482
+ @keyframes buttonFadeIn {
2483
+ 0% {
2484
+ opacity: 0;
2485
+ transform: translateY(10px);
2486
+ }
2487
+ 100% {
2488
+ opacity: 1;
2489
+ transform: translateY(0);
2490
+ }
2491
+ }
2214
2492
  ` }),
2215
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.results, children: [
2216
- percentage >= 60 && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2493
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.results, children: [
2494
+ percentage >= 60 && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2217
2495
  "div",
2218
2496
  {
2219
2497
  style: {
@@ -2230,15 +2508,70 @@ function QuizPlayer({
2230
2508
  },
2231
2509
  piece.id
2232
2510
  )) }),
2233
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { ...defaultStyles.resultsBackground, background: theme.bgGradient } }),
2234
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.resultsContent, children: [
2235
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.resultStars, children: [
2236
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(StarIcon, { filled: theme.stars >= 1, delay: 0.3 }),
2237
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(StarIcon, { filled: theme.stars >= 2, delay: 0.5 }),
2238
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(StarIcon, { filled: theme.stars >= 3, delay: 0.7 })
2511
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { ...defaultStyles.resultsBackground, background: theme.bgGradient } }),
2512
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.resultsContent, children: hideScore ? /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
2513
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { marginBottom: "24px" }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Mascot, { mood: "happy" }) }),
2514
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2515
+ "div",
2516
+ {
2517
+ style: {
2518
+ background: "linear-gradient(135deg, #dcfce7 0%, #bbf7d0 100%)",
2519
+ padding: "12px 28px",
2520
+ borderRadius: "50px",
2521
+ boxShadow: "0 4px 12px rgba(0,0,0,0.15)",
2522
+ marginBottom: "20px",
2523
+ animation: "badgePop 0.6s ease-out 0.2s forwards",
2524
+ opacity: 0,
2525
+ border: "3px solid #22c55e"
2526
+ },
2527
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2528
+ "span",
2529
+ {
2530
+ style: {
2531
+ fontSize: "22px",
2532
+ fontWeight: "700",
2533
+ color: "#1f2937",
2534
+ textShadow: "0 1px 2px rgba(255,255,255,0.5)"
2535
+ },
2536
+ children: "All Done!"
2537
+ }
2538
+ )
2539
+ }
2540
+ ),
2541
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2542
+ "div",
2543
+ {
2544
+ style: {
2545
+ animation: "scoreSlideIn 0.5s ease-out 0.4s forwards",
2546
+ opacity: 0
2547
+ },
2548
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2549
+ "div",
2550
+ {
2551
+ style: {
2552
+ fontSize: "24px",
2553
+ fontWeight: "600",
2554
+ color: "#22c55e",
2555
+ lineHeight: "1.4",
2556
+ marginBottom: "12px"
2557
+ },
2558
+ children: "The quiz was submitted successfully!"
2559
+ }
2560
+ )
2561
+ }
2562
+ ),
2563
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2564
+ "Time: ",
2565
+ formatTime(result.timeSpentSeconds)
2566
+ ] })
2567
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
2568
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.resultStars, children: [
2569
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(StarIcon, { filled: theme.stars >= 1, delay: 0.3 }),
2570
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(StarIcon, { filled: theme.stars >= 2, delay: 0.5 }),
2571
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(StarIcon, { filled: theme.stars >= 3, delay: 0.7 })
2239
2572
  ] }),
2240
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(MascotOwl, { mood: theme.mascotMood }) }),
2241
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2573
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Mascot, { mood: theme.mascotMood }) }),
2574
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2242
2575
  "div",
2243
2576
  {
2244
2577
  style: {
@@ -2251,7 +2584,7 @@ function QuizPlayer({
2251
2584
  opacity: 0,
2252
2585
  border: `3px solid ${theme.badgeColor}`
2253
2586
  },
2254
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2587
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2255
2588
  "span",
2256
2589
  {
2257
2590
  style: {
@@ -2265,7 +2598,7 @@ function QuizPlayer({
2265
2598
  )
2266
2599
  }
2267
2600
  ),
2268
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2601
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2269
2602
  "div",
2270
2603
  {
2271
2604
  style: {
@@ -2273,7 +2606,7 @@ function QuizPlayer({
2273
2606
  opacity: 0
2274
2607
  },
2275
2608
  children: [
2276
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2609
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2277
2610
  "div",
2278
2611
  {
2279
2612
  style: {
@@ -2290,7 +2623,7 @@ function QuizPlayer({
2290
2623
  ]
2291
2624
  }
2292
2625
  ),
2293
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2626
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2294
2627
  "div",
2295
2628
  {
2296
2629
  style: {
@@ -2305,25 +2638,116 @@ function QuizPlayer({
2305
2638
  ]
2306
2639
  }
2307
2640
  ),
2308
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2641
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2309
2642
  "Time: ",
2310
2643
  formatTime(result.timeSpentSeconds)
2311
- ] })
2312
- ] })
2644
+ ] }),
2645
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2646
+ "button",
2647
+ {
2648
+ style: {
2649
+ marginTop: "24px",
2650
+ padding: "14px 32px",
2651
+ fontSize: "16px",
2652
+ fontWeight: "600",
2653
+ color: "#7c3aed",
2654
+ background: "white",
2655
+ border: "2px solid #7c3aed",
2656
+ borderRadius: "12px",
2657
+ cursor: "pointer",
2658
+ transition: "all 0.2s ease",
2659
+ animation: "buttonFadeIn 0.5s ease-out 0.8s forwards",
2660
+ opacity: 0
2661
+ },
2662
+ onClick: handleRetryQuiz,
2663
+ onMouseOver: (e) => {
2664
+ e.currentTarget.style.background = "#7c3aed";
2665
+ e.currentTarget.style.color = "white";
2666
+ e.currentTarget.style.transform = "translateY(-2px)";
2667
+ },
2668
+ onMouseOut: (e) => {
2669
+ e.currentTarget.style.background = "white";
2670
+ e.currentTarget.style.color = "#7c3aed";
2671
+ e.currentTarget.style.transform = "translateY(0)";
2672
+ },
2673
+ "data-testid": "button-retry-quiz",
2674
+ children: "Try Again"
2675
+ }
2676
+ )
2677
+ ] }) })
2313
2678
  ] })
2314
2679
  ] });
2315
2680
  }
2681
+ if (quiz && showIntro && showResumeChoice && hasExistingProgress) {
2682
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.intro, children: [
2683
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introTitle, children: "Welcome Back!" }),
2684
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introSubtitle, children: "You have an unfinished quiz. Would you like to continue or start over?" }),
2685
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.introQuestionCount, children: [
2686
+ existingProgressCount,
2687
+ " of ",
2688
+ quiz.questions.length,
2689
+ " question",
2690
+ quiz.questions.length !== 1 ? "s" : "",
2691
+ " answered"
2692
+ ] }),
2693
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", flexDirection: "column", gap: "12px", marginTop: "8px" }, children: [
2694
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2695
+ "button",
2696
+ {
2697
+ style: defaultStyles.startButton,
2698
+ onClick: handleResumePrevious,
2699
+ onMouseOver: (e) => {
2700
+ e.currentTarget.style.transform = "translateY(-2px)";
2701
+ e.currentTarget.style.boxShadow = "0 6px 20px rgba(124, 58, 237, 0.5)";
2702
+ },
2703
+ onMouseOut: (e) => {
2704
+ e.currentTarget.style.transform = "translateY(0)";
2705
+ e.currentTarget.style.boxShadow = "0 4px 14px rgba(124, 58, 237, 0.4)";
2706
+ },
2707
+ "data-testid": "button-continue-quiz",
2708
+ children: "Continue Where I Left Off"
2709
+ }
2710
+ ),
2711
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2712
+ "button",
2713
+ {
2714
+ style: {
2715
+ ...defaultStyles.startButton,
2716
+ background: "transparent",
2717
+ color: "#7c3aed",
2718
+ border: "2px solid #7c3aed",
2719
+ boxShadow: "none"
2720
+ },
2721
+ onClick: handleStartFresh,
2722
+ disabled: isStartingFresh,
2723
+ onMouseOver: (e) => {
2724
+ if (!isStartingFresh) {
2725
+ e.currentTarget.style.transform = "translateY(-2px)";
2726
+ e.currentTarget.style.background = "rgba(124, 58, 237, 0.1)";
2727
+ }
2728
+ },
2729
+ onMouseOut: (e) => {
2730
+ e.currentTarget.style.transform = "translateY(0)";
2731
+ e.currentTarget.style.background = "transparent";
2732
+ },
2733
+ "data-testid": "button-start-fresh",
2734
+ children: isStartingFresh ? "Starting..." : "Start Fresh"
2735
+ }
2736
+ )
2737
+ ] })
2738
+ ] }) });
2739
+ }
2316
2740
  if (quiz && showIntro) {
2317
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.intro, children: [
2318
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.introTitle, children: quiz.title }),
2319
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.introSubtitle, children: "Ready to test your knowledge?" }),
2320
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.introQuestionCount, children: [
2741
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.intro, children: [
2742
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introTitle, children: quiz.title }),
2743
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.introSubtitle, children: "Ready to test your knowledge?" }),
2744
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.introQuestionCount, children: [
2321
2745
  quiz.questions.length,
2322
2746
  " question",
2323
2747
  quiz.questions.length !== 1 ? "s" : "",
2324
2748
  " to answer"
2325
2749
  ] }),
2326
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2750
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2327
2751
  "button",
2328
2752
  {
2329
2753
  style: defaultStyles.startButton,
@@ -2343,7 +2767,7 @@ function QuizPlayer({
2343
2767
  ] }) });
2344
2768
  }
2345
2769
  if (!quiz || !currentQuestion) {
2346
- 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" }) });
2770
+ 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" }) });
2347
2771
  }
2348
2772
  const selectedAnswer = answers.get(currentQuestion.id);
2349
2773
  const isLastQuestion = currentQuestionIndex === totalQuestions - 1;
@@ -2351,294 +2775,166 @@ function QuizPlayer({
2351
2775
  const remainingSlots = maxQuestions - totalQuestions;
2352
2776
  const questionsToAdd = Math.min(5, remainingSlots);
2353
2777
  const canAddMore = onGenerateMoreQuestions && remainingSlots > 0;
2354
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.mainLayout, children: [
2355
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.quizContent, children: [
2356
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.header, children: [
2357
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.title, children: quiz.title }),
2358
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.progress, children: [
2359
- "Question ",
2360
- currentQuestionIndex + 1,
2361
- " of ",
2362
- totalQuestions
2363
- ] }),
2364
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2365
- ] }),
2366
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { ...defaultStyles.question, position: "relative", paddingBottom: "40px" }, children: [
2367
- /* @__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" }) }),
2368
- isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2369
- "button",
2370
- {
2371
- onClick: () => setShowSkipModal(true),
2372
- title: "Skip question",
2373
- style: {
2374
- position: "absolute",
2375
- bottom: "8px",
2376
- left: "0",
2377
- background: "transparent",
2378
- border: "none",
2379
- cursor: "pointer",
2380
- padding: "6px 10px",
2381
- borderRadius: "6px",
2382
- color: "#9ca3af",
2383
- display: "flex",
2384
- alignItems: "center",
2385
- justifyContent: "center",
2386
- gap: "4px",
2387
- fontSize: "12px",
2388
- opacity: 0.6,
2389
- transition: "opacity 0.2s, color 0.2s"
2390
- },
2391
- onMouseEnter: (e) => {
2392
- e.currentTarget.style.opacity = "1";
2393
- e.currentTarget.style.color = "#6b7280";
2394
- },
2395
- onMouseLeave: (e) => {
2396
- e.currentTarget.style.opacity = "0.6";
2397
- e.currentTarget.style.color = "#9ca3af";
2398
- },
2399
- "data-testid": "button-skip-question",
2400
- children: [
2401
- /* @__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: [
2402
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("polygon", { points: "5 4 15 12 5 20 5 4" }),
2403
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "19", y1: "5", x2: "19", y2: "19" })
2404
- ] }),
2405
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { children: "Skip" })
2406
- ]
2407
- }
2408
- ),
2409
- !isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2410
- "button",
2411
- {
2412
- onClick: () => setShowReportModal(true),
2413
- title: "Report an issue with this question",
2414
- style: {
2415
- position: "absolute",
2416
- bottom: "8px",
2417
- left: "0",
2418
- background: "transparent",
2419
- border: "none",
2420
- cursor: "pointer",
2421
- padding: "6px 10px",
2422
- borderRadius: "6px",
2423
- color: "#9ca3af",
2424
- display: "flex",
2425
- alignItems: "center",
2426
- justifyContent: "center",
2427
- gap: "4px",
2428
- fontSize: "12px",
2429
- opacity: 0.6,
2430
- transition: "opacity 0.2s, color 0.2s"
2431
- },
2432
- onMouseEnter: (e) => {
2433
- e.currentTarget.style.opacity = "1";
2434
- e.currentTarget.style.color = "#ef4444";
2435
- },
2436
- onMouseLeave: (e) => {
2437
- e.currentTarget.style.opacity = "0.6";
2438
- e.currentTarget.style.color = "#9ca3af";
2439
- },
2440
- "data-testid": "button-report-question",
2441
- children: [
2442
- /* @__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: [
2443
- /* @__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" }),
2444
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("line", { x1: "4", y1: "22", x2: "4", y2: "15" })
2445
- ] }),
2446
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { children: "Report" })
2447
- ]
2778
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { className, style: defaultStyles.container, children: [
2779
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: `
2780
+ .quiz-option:focus,
2781
+ .quiz-option:active,
2782
+ .quiz-option:focus-visible,
2783
+ .quiz-option:focus-within {
2784
+ outline: none !important;
2785
+ box-shadow: none !important;
2448
2786
  }
2449
- ),
2450
- (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2451
- const isSelected = selectedAnswer === option;
2452
- const isCorrectOption = currentQuestion.correctAnswer === option;
2453
- let optionStyle = { ...defaultStyles.option };
2454
- if (showFeedback) {
2455
- if (isCorrectOption) {
2456
- optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2457
- } else if (isSelected && !isCorrectOption) {
2458
- optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2459
- }
2460
- } else if (isSelected) {
2461
- optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2787
+ .quiz-option {
2788
+ -webkit-tap-highlight-color: transparent;
2462
2789
  }
2463
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2464
- "div",
2790
+ ` }),
2791
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.mainLayout, children: [
2792
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.quizContent, children: [
2793
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.header, children: [
2794
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.title, children: quiz.title }),
2795
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.progress, children: [
2796
+ "Question ",
2797
+ currentQuestionIndex + 1,
2798
+ " of ",
2799
+ totalQuestions
2800
+ ] }),
2801
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2802
+ ] }),
2803
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { ...defaultStyles.question, position: "relative", paddingBottom: "40px" }, children: [
2804
+ /* @__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" }) }),
2805
+ isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2806
+ "button",
2465
2807
  {
2808
+ onClick: () => setShowSkipModal(true),
2809
+ title: "Skip question",
2466
2810
  style: {
2467
- ...optionStyle,
2468
- cursor: showFeedback ? "default" : "pointer",
2811
+ position: "absolute",
2812
+ bottom: "8px",
2813
+ left: "0",
2814
+ background: "transparent",
2815
+ border: "none",
2816
+ cursor: "pointer",
2817
+ padding: "6px 10px",
2818
+ borderRadius: "6px",
2819
+ color: "#9ca3af",
2469
2820
  display: "flex",
2470
2821
  alignItems: "center",
2471
- gap: "8px"
2822
+ justifyContent: "center",
2823
+ gap: "4px",
2824
+ fontSize: "12px",
2825
+ opacity: 0.6,
2826
+ transition: "opacity 0.2s, color 0.2s"
2827
+ },
2828
+ onMouseEnter: (e) => {
2829
+ e.currentTarget.style.opacity = "1";
2830
+ e.currentTarget.style.color = "#6b7280";
2472
2831
  },
2473
- onClick: () => !showFeedback && handleAnswerChange(option),
2832
+ onMouseLeave: (e) => {
2833
+ e.currentTarget.style.opacity = "0.6";
2834
+ e.currentTarget.style.color = "#9ca3af";
2835
+ },
2836
+ "data-testid": "button-skip-question",
2474
2837
  children: [
2475
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2476
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: option })
2838
+ /* @__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: [
2839
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("polygon", { points: "5 4 15 12 5 20 5 4" }),
2840
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "19", y1: "5", x2: "19", y2: "19" })
2841
+ ] }),
2842
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { children: "Skip" })
2477
2843
  ]
2478
- },
2479
- idx
2480
- );
2481
- }) }),
2482
- currentQuestion.type === "multiple" && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2483
- const selected = Array.isArray(selectedAnswer) && selectedAnswer.includes(option);
2484
- const correctAnswers = Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer : currentQuestion.correctAnswer ? [currentQuestion.correctAnswer] : [];
2485
- const isCorrectOption = correctAnswers.includes(option);
2486
- let optionStyle = { ...defaultStyles.option };
2487
- if (showFeedback) {
2488
- if (isCorrectOption) {
2489
- optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2490
- } else if (selected && !isCorrectOption) {
2491
- optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2492
2844
  }
2493
- } else if (selected) {
2494
- optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2495
- }
2496
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(
2497
- "div",
2845
+ ),
2846
+ !isExtraQuestion && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2847
+ "button",
2498
2848
  {
2849
+ onClick: () => setShowReportModal(true),
2850
+ title: "Report an issue with this question",
2499
2851
  style: {
2500
- ...optionStyle,
2501
- cursor: showFeedback ? "default" : "pointer",
2852
+ position: "absolute",
2853
+ bottom: "8px",
2854
+ left: "0",
2855
+ background: "transparent",
2856
+ border: "none",
2857
+ cursor: "pointer",
2858
+ padding: "6px 10px",
2859
+ borderRadius: "6px",
2860
+ color: "#9ca3af",
2502
2861
  display: "flex",
2503
2862
  alignItems: "center",
2504
- gap: "8px"
2863
+ justifyContent: "center",
2864
+ gap: "4px",
2865
+ fontSize: "12px",
2866
+ opacity: 0.6,
2867
+ transition: "opacity 0.2s, color 0.2s"
2505
2868
  },
2506
- onClick: () => {
2507
- if (showFeedback) return;
2508
- const current = Array.isArray(selectedAnswer) ? selectedAnswer : [];
2509
- if (selected) {
2510
- handleAnswerChange(current.filter((o) => o !== option));
2511
- } else {
2512
- handleAnswerChange([...current, option]);
2513
- }
2869
+ onMouseEnter: (e) => {
2870
+ e.currentTarget.style.opacity = "1";
2871
+ e.currentTarget.style.color = "#ef4444";
2872
+ },
2873
+ onMouseLeave: (e) => {
2874
+ e.currentTarget.style.opacity = "0.6";
2875
+ e.currentTarget.style.color = "#9ca3af";
2514
2876
  },
2877
+ "data-testid": "button-report-question",
2515
2878
  children: [
2516
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2517
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: option })
2879
+ /* @__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: [
2880
+ /* @__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" }),
2881
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("line", { x1: "4", y1: "22", x2: "4", y2: "15" })
2882
+ ] }),
2883
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { children: "Report" })
2518
2884
  ]
2519
- },
2520
- idx
2521
- );
2522
- }) }),
2523
- (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2524
- "textarea",
2525
- {
2526
- style: { ...defaultStyles.input, minHeight: currentQuestion.type === "essay" ? "150px" : "60px" },
2527
- value: selectedAnswer || "",
2528
- onChange: (e) => handleAnswerChange(e.target.value),
2529
- placeholder: "Type your answer here...",
2530
- disabled: showFeedback
2531
- }
2532
- ),
2533
- currentQuestion.type === "fill" && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2534
- "input",
2535
- {
2536
- style: defaultStyles.input,
2537
- value: (Array.isArray(selectedAnswer) ? selectedAnswer[idx] : "") || "",
2538
- onChange: (e) => {
2539
- const current = Array.isArray(selectedAnswer) ? [...selectedAnswer] : [];
2540
- current[idx] = e.target.value;
2541
- handleAnswerChange(current);
2542
- },
2543
- placeholder: `Blank ${idx + 1}`,
2544
- disabled: showFeedback
2545
- },
2546
- idx
2547
- )) }),
2548
- currentQuestion.type === "sorting" && currentQuestion.items && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2549
- SortingDragDrop,
2550
- {
2551
- items: currentQuestion.items,
2552
- currentOrder: Array.isArray(selectedAnswer) ? selectedAnswer : currentQuestion.items.map((_, i) => i),
2553
- correctOrder: currentQuestion.correctOrder,
2554
- showFeedback,
2555
- onOrderChange: handleAnswerChange
2556
- }
2557
- ),
2558
- currentQuestion.type === "matrix" && currentQuestion.leftItems && currentQuestion.rightItems && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2559
- MatchingDragDrop,
2560
- {
2561
- leftItems: currentQuestion.leftItems,
2562
- rightItems: currentQuestion.rightItems,
2563
- currentMatches: typeof selectedAnswer === "object" && selectedAnswer !== null && !Array.isArray(selectedAnswer) ? selectedAnswer : {},
2564
- correctMatches: currentQuestion.correctMatches,
2565
- showFeedback,
2566
- onMatchChange: handleAnswerChange
2567
- }
2568
- ),
2569
- currentQuestion.type === "assessment" && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.options, children: (() => {
2570
- const scaleType = currentQuestion.scaleType || "likert";
2571
- if (scaleType === "yes-no") {
2572
- const options = ["Yes", "No"];
2573
- return options.map((option, idx) => {
2574
- const isSelected = selectedAnswer === option;
2575
- let optionStyle = { ...defaultStyles.option };
2576
- if (isSelected) {
2577
- optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2885
+ }
2886
+ ),
2887
+ (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2888
+ const isSelected = selectedAnswer === option;
2889
+ const isCorrectOption = currentQuestion.correctAnswer === option;
2890
+ let optionStyle = { ...defaultStyles.option };
2891
+ if (showFeedback) {
2892
+ if (isCorrectOption) {
2893
+ optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2894
+ } else if (isSelected && !isCorrectOption) {
2895
+ optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2578
2896
  }
2579
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2580
- "div",
2581
- {
2582
- style: {
2583
- ...optionStyle,
2584
- cursor: showFeedback ? "default" : "pointer",
2585
- display: "flex",
2586
- alignItems: "center",
2587
- gap: "8px"
2588
- },
2589
- onClick: () => !showFeedback && handleAnswerChange(option),
2590
- "data-testid": `assessment-option-${option.toLowerCase()}`,
2591
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: option })
2592
- },
2593
- idx
2594
- );
2595
- });
2596
- }
2597
- if (scaleType === "rating") {
2598
- const min = currentQuestion.scaleMin || 1;
2599
- const max = currentQuestion.scaleMax || 5;
2600
- const ratings = Array.from({ length: max - min + 1 }, (_, i) => min + i);
2601
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { display: "flex", gap: "8px", flexWrap: "wrap", justifyContent: "center" }, children: ratings.map((rating) => {
2602
- const isSelected = selectedAnswer === rating;
2603
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2604
- "button",
2605
- {
2606
- onClick: () => !showFeedback && handleAnswerChange(rating),
2607
- disabled: showFeedback,
2608
- style: {
2609
- width: "48px",
2610
- height: "48px",
2611
- borderRadius: "50%",
2612
- border: isSelected ? "2px solid #6721b0" : "2px solid #e5e7eb",
2613
- backgroundColor: isSelected ? "#f3e8ff" : "#ffffff",
2614
- cursor: showFeedback ? "not-allowed" : "pointer",
2615
- fontSize: "18px",
2616
- fontWeight: "600",
2617
- color: isSelected ? "#6721b0" : "#374151"
2618
- },
2619
- "data-testid": `assessment-rating-${rating}`,
2620
- children: rating
2897
+ } else if (isSelected) {
2898
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2899
+ }
2900
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2901
+ "div",
2902
+ {
2903
+ className: "quiz-option",
2904
+ style: {
2905
+ ...optionStyle,
2906
+ cursor: showFeedback ? "default" : "pointer",
2907
+ display: "flex",
2908
+ alignItems: "center",
2909
+ gap: "8px"
2621
2910
  },
2622
- rating
2623
- );
2624
- }) });
2625
- }
2626
- const likertOptions = [
2627
- "Strongly Disagree",
2628
- "Disagree",
2629
- "Neutral",
2630
- "Agree",
2631
- "Strongly Agree"
2632
- ];
2633
- return likertOptions.map((option, idx) => {
2634
- const isSelected = selectedAnswer === option;
2911
+ onClick: () => !showFeedback && handleAnswerChange(option),
2912
+ children: [
2913
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2914
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
2915
+ ]
2916
+ },
2917
+ idx
2918
+ );
2919
+ }) }),
2920
+ currentQuestion.type === "multiple" && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2921
+ const selected = Array.isArray(selectedAnswer) && selectedAnswer.includes(option);
2922
+ const correctAnswers = Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer : currentQuestion.correctAnswer ? [currentQuestion.correctAnswer] : [];
2923
+ const isCorrectOption = correctAnswers.includes(option);
2635
2924
  let optionStyle = { ...defaultStyles.option };
2636
- if (isSelected) {
2925
+ if (showFeedback) {
2926
+ if (isCorrectOption) {
2927
+ optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2928
+ } else if (selected && !isCorrectOption) {
2929
+ optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2930
+ }
2931
+ } else if (selected) {
2637
2932
  optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2638
2933
  }
2639
- return /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2934
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
2640
2935
  "div",
2641
2936
  {
2937
+ className: "quiz-option",
2642
2938
  style: {
2643
2939
  ...optionStyle,
2644
2940
  cursor: showFeedback ? "default" : "pointer",
@@ -2646,356 +2942,504 @@ function QuizPlayer({
2646
2942
  alignItems: "center",
2647
2943
  gap: "8px"
2648
2944
  },
2649
- onClick: () => !showFeedback && handleAnswerChange(option),
2650
- "data-testid": `assessment-likert-${idx}`,
2651
- children: /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("span", { style: { flex: 1 }, children: option })
2945
+ onClick: () => {
2946
+ if (showFeedback) return;
2947
+ const current = Array.isArray(selectedAnswer) ? selectedAnswer : [];
2948
+ if (selected) {
2949
+ handleAnswerChange(current.filter((o) => o !== option));
2950
+ } else {
2951
+ handleAnswerChange([...current, option]);
2952
+ }
2953
+ },
2954
+ children: [
2955
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(TextToSpeech, { text: option, size: "sm" }) }),
2956
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
2957
+ ]
2652
2958
  },
2653
2959
  idx
2654
2960
  );
2655
- });
2656
- })() }),
2657
- showFeedback && currentAnswerDetail && /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: {
2658
- ...defaultStyles.feedback,
2659
- ...currentQuestion.type === "assessment" ? defaultStyles.feedbackNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
2660
- }, children: [
2661
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
2662
- ...defaultStyles.feedbackTitle,
2663
- ...currentQuestion.type === "assessment" ? defaultStyles.feedbackTitleNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
2664
- }, children: currentQuestion.type === "assessment" ? "\u2713 Response recorded" : currentAnswerDetail.isCorrect ? "\u2713 Correct!" : "\u2717 Incorrect" }),
2665
- currentQuestion.explanation && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.feedbackExplanation, children: currentQuestion.explanation })
2666
- ] })
2667
- ] }),
2668
- showSkipModal && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
2669
- position: "fixed",
2670
- top: 0,
2671
- left: 0,
2672
- right: 0,
2673
- bottom: 0,
2674
- backgroundColor: "rgba(0, 0, 0, 0.5)",
2675
- display: "flex",
2676
- alignItems: "center",
2677
- justifyContent: "center",
2678
- zIndex: 1e3
2679
- }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: {
2680
- backgroundColor: "#ffffff",
2681
- borderRadius: "12px",
2682
- padding: "24px",
2683
- maxWidth: "400px",
2684
- width: "90%",
2685
- boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2686
- }, children: [
2687
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Skip Question" }),
2688
- /* @__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." }),
2689
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { display: "flex", flexDirection: "column", gap: "8px", marginBottom: "16px" }, children: [
2690
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2691
- "button",
2961
+ }) }),
2962
+ (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2963
+ "textarea",
2692
2964
  {
2693
- onClick: () => setSelectedSkipReason("question_issue"),
2694
- disabled: isSkipping,
2695
- style: {
2696
- padding: "12px 16px",
2697
- borderRadius: "8px",
2698
- border: selectedSkipReason === "question_issue" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
2699
- backgroundColor: selectedSkipReason === "question_issue" ? "#f5f3ff" : "#f9fafb",
2700
- cursor: isSkipping ? "not-allowed" : "pointer",
2701
- fontSize: "14px",
2702
- fontWeight: "500",
2703
- color: "#374151",
2704
- textAlign: "left",
2705
- opacity: isSkipping ? 0.6 : 1
2706
- },
2707
- "data-testid": "button-skip-reason-issue",
2708
- children: "Question has an issue"
2965
+ style: { ...defaultStyles.input, minHeight: currentQuestion.type === "essay" ? "150px" : "60px" },
2966
+ value: selectedAnswer || "",
2967
+ onChange: (e) => handleAnswerChange(e.target.value),
2968
+ placeholder: "Type your answer here...",
2969
+ disabled: showFeedback
2709
2970
  }
2710
2971
  ),
2711
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2712
- "button",
2972
+ currentQuestion.type === "fill" && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2973
+ "input",
2713
2974
  {
2714
- onClick: () => setSelectedSkipReason("dont_know"),
2715
- disabled: isSkipping,
2716
- style: {
2717
- padding: "12px 16px",
2718
- borderRadius: "8px",
2719
- border: selectedSkipReason === "dont_know" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
2720
- backgroundColor: selectedSkipReason === "dont_know" ? "#f5f3ff" : "#f9fafb",
2721
- cursor: isSkipping ? "not-allowed" : "pointer",
2722
- fontSize: "14px",
2723
- fontWeight: "500",
2724
- color: "#374151",
2725
- textAlign: "left",
2726
- opacity: isSkipping ? 0.6 : 1
2975
+ style: defaultStyles.input,
2976
+ value: (Array.isArray(selectedAnswer) ? selectedAnswer[idx] : "") || "",
2977
+ onChange: (e) => {
2978
+ const current = Array.isArray(selectedAnswer) ? [...selectedAnswer] : [];
2979
+ current[idx] = e.target.value;
2980
+ handleAnswerChange(current);
2727
2981
  },
2728
- "data-testid": "button-skip-reason-dont-know",
2729
- children: "I don't know the answer"
2730
- }
2731
- )
2732
- ] }),
2733
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { marginBottom: "16px" }, children: [
2734
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("label", { style: { display: "block", fontSize: "13px", fontWeight: "500", color: "#374151", marginBottom: "6px" }, children: "Additional details (optional)" }),
2735
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2736
- "textarea",
2982
+ placeholder: `Blank ${idx + 1}`,
2983
+ disabled: showFeedback
2984
+ },
2985
+ idx
2986
+ )) }),
2987
+ currentQuestion.type === "sorting" && currentQuestion.items && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2988
+ SortingDragDrop,
2737
2989
  {
2738
- value: skipComment,
2739
- onChange: (e) => setSkipComment(e.target.value.slice(0, 200)),
2740
- placeholder: "Tell us more about the issue...",
2741
- disabled: isSkipping,
2742
- style: {
2743
- width: "100%",
2744
- minHeight: "80px",
2745
- padding: "10px 12px",
2746
- borderRadius: "8px",
2747
- border: "1px solid #e5e7eb",
2748
- fontSize: "14px",
2749
- resize: "vertical",
2750
- fontFamily: "inherit",
2751
- boxSizing: "border-box"
2752
- },
2753
- "data-testid": "input-skip-comment"
2990
+ items: currentQuestion.items,
2991
+ currentOrder: Array.isArray(selectedAnswer) ? selectedAnswer : currentQuestion.items.map((_, i) => i),
2992
+ correctOrder: currentQuestion.correctOrder,
2993
+ showFeedback,
2994
+ onOrderChange: handleAnswerChange
2754
2995
  }
2755
2996
  ),
2756
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2757
- skipComment.length,
2758
- "/200"
2759
- ] })
2760
- ] }),
2761
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
2762
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2763
- "button",
2997
+ currentQuestion.type === "matrix" && currentQuestion.leftItems && currentQuestion.rightItems && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2998
+ MatchingDragDrop,
2764
2999
  {
2765
- onClick: () => {
2766
- setShowSkipModal(false);
2767
- setSkipComment("");
2768
- setSelectedSkipReason(null);
2769
- },
2770
- style: {
2771
- flex: 1,
2772
- padding: "10px 16px",
2773
- borderRadius: "8px",
2774
- border: "1px solid #e5e7eb",
2775
- backgroundColor: "#ffffff",
2776
- cursor: "pointer",
2777
- fontSize: "14px",
2778
- fontWeight: "500",
2779
- color: "#6b7280"
2780
- },
2781
- "data-testid": "button-skip-cancel",
2782
- children: "Cancel"
3000
+ leftItems: currentQuestion.leftItems,
3001
+ rightItems: currentQuestion.rightItems,
3002
+ currentMatches: typeof selectedAnswer === "object" && selectedAnswer !== null && !Array.isArray(selectedAnswer) ? selectedAnswer : {},
3003
+ correctMatches: currentQuestion.correctMatches,
3004
+ showFeedback,
3005
+ onMatchChange: handleAnswerChange
2783
3006
  }
2784
3007
  ),
2785
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2786
- "button",
2787
- {
2788
- onClick: () => selectedSkipReason && handleSkipQuestion(selectedSkipReason, skipComment),
2789
- disabled: isSkipping || !selectedSkipReason,
2790
- style: {
2791
- flex: 1,
2792
- padding: "10px 16px",
2793
- borderRadius: "8px",
2794
- border: "none",
2795
- backgroundColor: selectedSkipReason ? "#8b5cf6" : "#d1d5db",
2796
- cursor: isSkipping || !selectedSkipReason ? "not-allowed" : "pointer",
2797
- fontSize: "14px",
2798
- fontWeight: "500",
2799
- color: "#ffffff",
2800
- opacity: isSkipping ? 0.6 : 1
2801
- },
2802
- "data-testid": "button-skip-submit",
2803
- children: isSkipping ? "Skipping..." : "Skip Question"
3008
+ currentQuestion.type === "assessment" && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.options, children: (() => {
3009
+ const scaleType = currentQuestion.scaleType || "likert";
3010
+ if (scaleType === "yes-no") {
3011
+ const options = ["Yes", "No"];
3012
+ return options.map((option, idx) => {
3013
+ const isSelected = selectedAnswer === option;
3014
+ let optionStyle = { ...defaultStyles.option };
3015
+ if (isSelected) {
3016
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
3017
+ }
3018
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3019
+ "div",
3020
+ {
3021
+ className: "quiz-option",
3022
+ style: {
3023
+ ...optionStyle,
3024
+ cursor: showFeedback ? "default" : "pointer",
3025
+ display: "flex",
3026
+ alignItems: "center",
3027
+ gap: "8px"
3028
+ },
3029
+ onClick: () => !showFeedback && handleAnswerChange(option),
3030
+ "data-testid": `assessment-option-${option.toLowerCase()}`,
3031
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
3032
+ },
3033
+ idx
3034
+ );
3035
+ });
2804
3036
  }
2805
- )
2806
- ] })
2807
- ] }) }),
2808
- showReportModal && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: {
2809
- position: "fixed",
2810
- top: 0,
2811
- left: 0,
2812
- right: 0,
2813
- bottom: 0,
2814
- backgroundColor: "rgba(0, 0, 0, 0.5)",
2815
- display: "flex",
2816
- alignItems: "center",
2817
- justifyContent: "center",
2818
- zIndex: 1e3
2819
- }, children: /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: {
2820
- backgroundColor: "#ffffff",
2821
- borderRadius: "12px",
2822
- padding: "24px",
2823
- maxWidth: "400px",
2824
- width: "90%",
2825
- boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2826
- }, children: [
2827
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Report Question" }),
2828
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "What's wrong with this question?" }),
2829
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { marginBottom: "16px" }, children: [
2830
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2831
- "textarea",
2832
- {
2833
- value: reportComment,
2834
- onChange: (e) => setReportComment(e.target.value.slice(0, 300)),
2835
- placeholder: "Describe the issue with this question...",
2836
- disabled: isReporting,
2837
- style: {
2838
- width: "100%",
2839
- minHeight: "120px",
2840
- padding: "10px 12px",
2841
- borderRadius: "8px",
2842
- border: "1px solid #e5e7eb",
2843
- fontSize: "14px",
2844
- resize: "vertical",
2845
- fontFamily: "inherit",
2846
- boxSizing: "border-box"
2847
- },
2848
- "data-testid": "input-report-comment"
3037
+ if (scaleType === "rating") {
3038
+ const min = currentQuestion.scaleMin || 1;
3039
+ const max = currentQuestion.scaleMax || 5;
3040
+ const ratings = Array.from({ length: max - min + 1 }, (_, i) => min + i);
3041
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { display: "flex", gap: "8px", flexWrap: "wrap", justifyContent: "center" }, children: ratings.map((rating) => {
3042
+ const isSelected = selectedAnswer === rating;
3043
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3044
+ "button",
3045
+ {
3046
+ className: "quiz-option",
3047
+ onClick: () => !showFeedback && handleAnswerChange(rating),
3048
+ disabled: showFeedback,
3049
+ style: {
3050
+ width: "48px",
3051
+ height: "48px",
3052
+ borderRadius: "50%",
3053
+ border: isSelected ? "2px solid #6721b0" : "2px solid #e5e7eb",
3054
+ backgroundColor: isSelected ? "#f3e8ff" : "#ffffff",
3055
+ cursor: showFeedback ? "not-allowed" : "pointer",
3056
+ fontSize: "18px",
3057
+ fontWeight: "600",
3058
+ color: isSelected ? "#6721b0" : "#374151",
3059
+ outline: "none"
3060
+ },
3061
+ "data-testid": `assessment-rating-${rating}`,
3062
+ children: rating
3063
+ },
3064
+ rating
3065
+ );
3066
+ }) });
2849
3067
  }
2850
- ),
2851
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2852
- reportComment.length,
2853
- "/300"
3068
+ const likertOptions = [
3069
+ "Strongly Disagree",
3070
+ "Disagree",
3071
+ "Neutral",
3072
+ "Agree",
3073
+ "Strongly Agree"
3074
+ ];
3075
+ return likertOptions.map((option, idx) => {
3076
+ const isSelected = selectedAnswer === option;
3077
+ let optionStyle = { ...defaultStyles.option };
3078
+ if (isSelected) {
3079
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
3080
+ }
3081
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3082
+ "div",
3083
+ {
3084
+ className: "quiz-option",
3085
+ style: {
3086
+ ...optionStyle,
3087
+ cursor: showFeedback ? "default" : "pointer",
3088
+ display: "flex",
3089
+ alignItems: "center",
3090
+ gap: "8px"
3091
+ },
3092
+ onClick: () => !showFeedback && handleAnswerChange(option),
3093
+ "data-testid": `assessment-likert-${idx}`,
3094
+ children: /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { flex: 1 }, children: option })
3095
+ },
3096
+ idx
3097
+ );
3098
+ });
3099
+ })() }),
3100
+ showFeedback && currentAnswerDetail && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
3101
+ ...defaultStyles.feedback,
3102
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
3103
+ }, children: [
3104
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
3105
+ ...defaultStyles.feedbackTitle,
3106
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackTitleNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
3107
+ }, children: currentQuestion.type === "assessment" ? "\u2713 Response recorded" : currentAnswerDetail.isCorrect ? "\u2713 Correct!" : "\u2717 Incorrect" }),
3108
+ currentQuestion.explanation && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.feedbackExplanation, children: currentQuestion.explanation })
2854
3109
  ] })
2855
3110
  ] }),
2856
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
2857
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
3111
+ showSkipModal && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
3112
+ position: "fixed",
3113
+ top: 0,
3114
+ left: 0,
3115
+ right: 0,
3116
+ bottom: 0,
3117
+ backgroundColor: "rgba(0, 0, 0, 0.5)",
3118
+ display: "flex",
3119
+ alignItems: "center",
3120
+ justifyContent: "center",
3121
+ zIndex: 1e3
3122
+ }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
3123
+ backgroundColor: "#ffffff",
3124
+ borderRadius: "12px",
3125
+ padding: "24px",
3126
+ maxWidth: "400px",
3127
+ width: "90%",
3128
+ boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
3129
+ }, children: [
3130
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Skip Question" }),
3131
+ /* @__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." }),
3132
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", flexDirection: "column", gap: "8px", marginBottom: "16px" }, children: [
3133
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3134
+ "button",
3135
+ {
3136
+ onClick: () => setSelectedSkipReason("question_issue"),
3137
+ disabled: isSkipping,
3138
+ style: {
3139
+ padding: "12px 16px",
3140
+ borderRadius: "8px",
3141
+ border: selectedSkipReason === "question_issue" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
3142
+ backgroundColor: selectedSkipReason === "question_issue" ? "#f5f3ff" : "#f9fafb",
3143
+ cursor: isSkipping ? "not-allowed" : "pointer",
3144
+ fontSize: "14px",
3145
+ fontWeight: "500",
3146
+ color: "#374151",
3147
+ textAlign: "left",
3148
+ opacity: isSkipping ? 0.6 : 1
3149
+ },
3150
+ "data-testid": "button-skip-reason-issue",
3151
+ children: "Question has an issue"
3152
+ }
3153
+ ),
3154
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3155
+ "button",
3156
+ {
3157
+ onClick: () => setSelectedSkipReason("dont_know"),
3158
+ disabled: isSkipping,
3159
+ style: {
3160
+ padding: "12px 16px",
3161
+ borderRadius: "8px",
3162
+ border: selectedSkipReason === "dont_know" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
3163
+ backgroundColor: selectedSkipReason === "dont_know" ? "#f5f3ff" : "#f9fafb",
3164
+ cursor: isSkipping ? "not-allowed" : "pointer",
3165
+ fontSize: "14px",
3166
+ fontWeight: "500",
3167
+ color: "#374151",
3168
+ textAlign: "left",
3169
+ opacity: isSkipping ? 0.6 : 1
3170
+ },
3171
+ "data-testid": "button-skip-reason-dont-know",
3172
+ children: "I don't know the answer"
3173
+ }
3174
+ )
3175
+ ] }),
3176
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { marginBottom: "16px" }, children: [
3177
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("label", { style: { display: "block", fontSize: "13px", fontWeight: "500", color: "#374151", marginBottom: "6px" }, children: "Additional details (optional)" }),
3178
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3179
+ "textarea",
3180
+ {
3181
+ value: skipComment,
3182
+ onChange: (e) => setSkipComment(e.target.value.slice(0, 200)),
3183
+ placeholder: "Tell us more about the issue...",
3184
+ disabled: isSkipping,
3185
+ style: {
3186
+ width: "100%",
3187
+ minHeight: "80px",
3188
+ padding: "10px 12px",
3189
+ borderRadius: "8px",
3190
+ border: "1px solid #e5e7eb",
3191
+ fontSize: "14px",
3192
+ resize: "vertical",
3193
+ fontFamily: "inherit",
3194
+ boxSizing: "border-box"
3195
+ },
3196
+ "data-testid": "input-skip-comment"
3197
+ }
3198
+ ),
3199
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3200
+ skipComment.length,
3201
+ "/200"
3202
+ ] })
3203
+ ] }),
3204
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
3205
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3206
+ "button",
3207
+ {
3208
+ onClick: () => {
3209
+ setShowSkipModal(false);
3210
+ setSkipComment("");
3211
+ setSelectedSkipReason(null);
3212
+ },
3213
+ style: {
3214
+ flex: 1,
3215
+ padding: "10px 16px",
3216
+ borderRadius: "8px",
3217
+ border: "1px solid #e5e7eb",
3218
+ backgroundColor: "#ffffff",
3219
+ cursor: "pointer",
3220
+ fontSize: "14px",
3221
+ fontWeight: "500",
3222
+ color: "#6b7280"
3223
+ },
3224
+ "data-testid": "button-skip-cancel",
3225
+ children: "Cancel"
3226
+ }
3227
+ ),
3228
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3229
+ "button",
3230
+ {
3231
+ onClick: () => selectedSkipReason && handleSkipQuestion(selectedSkipReason, skipComment),
3232
+ disabled: isSkipping || !selectedSkipReason,
3233
+ style: {
3234
+ flex: 1,
3235
+ padding: "10px 16px",
3236
+ borderRadius: "8px",
3237
+ border: "none",
3238
+ backgroundColor: selectedSkipReason ? "#8b5cf6" : "#d1d5db",
3239
+ cursor: isSkipping || !selectedSkipReason ? "not-allowed" : "pointer",
3240
+ fontSize: "14px",
3241
+ fontWeight: "500",
3242
+ color: "#ffffff",
3243
+ opacity: isSkipping ? 0.6 : 1
3244
+ },
3245
+ "data-testid": "button-skip-submit",
3246
+ children: isSkipping ? "Skipping..." : "Skip Question"
3247
+ }
3248
+ )
3249
+ ] })
3250
+ ] }) }),
3251
+ showReportModal && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: {
3252
+ position: "fixed",
3253
+ top: 0,
3254
+ left: 0,
3255
+ right: 0,
3256
+ bottom: 0,
3257
+ backgroundColor: "rgba(0, 0, 0, 0.5)",
3258
+ display: "flex",
3259
+ alignItems: "center",
3260
+ justifyContent: "center",
3261
+ zIndex: 1e3
3262
+ }, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
3263
+ backgroundColor: "#ffffff",
3264
+ borderRadius: "12px",
3265
+ padding: "24px",
3266
+ maxWidth: "400px",
3267
+ width: "90%",
3268
+ boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
3269
+ }, children: [
3270
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Report Question" }),
3271
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "What's wrong with this question?" }),
3272
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { marginBottom: "16px" }, children: [
3273
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3274
+ "textarea",
3275
+ {
3276
+ value: reportComment,
3277
+ onChange: (e) => setReportComment(e.target.value.slice(0, 300)),
3278
+ placeholder: "Describe the issue with this question...",
3279
+ disabled: isReporting,
3280
+ style: {
3281
+ width: "100%",
3282
+ minHeight: "120px",
3283
+ padding: "10px 12px",
3284
+ borderRadius: "8px",
3285
+ border: "1px solid #e5e7eb",
3286
+ fontSize: "14px",
3287
+ resize: "vertical",
3288
+ fontFamily: "inherit",
3289
+ boxSizing: "border-box"
3290
+ },
3291
+ "data-testid": "input-report-comment"
3292
+ }
3293
+ ),
3294
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3295
+ reportComment.length,
3296
+ "/300"
3297
+ ] })
3298
+ ] }),
3299
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", gap: "10px" }, children: [
3300
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3301
+ "button",
3302
+ {
3303
+ onClick: () => {
3304
+ setShowReportModal(false);
3305
+ setReportComment("");
3306
+ },
3307
+ style: {
3308
+ flex: 1,
3309
+ padding: "10px 16px",
3310
+ borderRadius: "8px",
3311
+ border: "1px solid #e5e7eb",
3312
+ backgroundColor: "#ffffff",
3313
+ cursor: "pointer",
3314
+ fontSize: "14px",
3315
+ fontWeight: "500",
3316
+ color: "#6b7280"
3317
+ },
3318
+ "data-testid": "button-report-cancel",
3319
+ children: "Cancel"
3320
+ }
3321
+ ),
3322
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3323
+ "button",
3324
+ {
3325
+ onClick: () => handleReportQuestion(reportComment),
3326
+ disabled: isReporting || !reportComment.trim(),
3327
+ style: {
3328
+ flex: 1,
3329
+ padding: "10px 16px",
3330
+ borderRadius: "8px",
3331
+ border: "none",
3332
+ backgroundColor: reportComment.trim() ? "#ef4444" : "#d1d5db",
3333
+ cursor: isReporting || !reportComment.trim() ? "not-allowed" : "pointer",
3334
+ fontSize: "14px",
3335
+ fontWeight: "500",
3336
+ color: "#ffffff",
3337
+ opacity: isReporting ? 0.6 : 1
3338
+ },
3339
+ "data-testid": "button-report-submit",
3340
+ children: isReporting ? "Reporting..." : "Report"
3341
+ }
3342
+ )
3343
+ ] })
3344
+ ] }) }),
3345
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles.buttonsColumn, children: [
3346
+ showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
2858
3347
  "button",
2859
3348
  {
2860
- onClick: () => {
2861
- setShowReportModal(false);
2862
- setReportComment("");
2863
- },
2864
3349
  style: {
2865
- flex: 1,
2866
- padding: "10px 16px",
2867
- borderRadius: "8px",
2868
- border: "1px solid #e5e7eb",
2869
- backgroundColor: "#ffffff",
2870
- cursor: "pointer",
2871
- fontSize: "14px",
2872
- fontWeight: "500",
2873
- color: "#6b7280"
3350
+ ...defaultStyles.buttonAddMore,
3351
+ ...isGeneratingExtra ? defaultStyles.buttonAddMoreDisabled : {}
2874
3352
  },
2875
- "data-testid": "button-report-cancel",
2876
- children: "Cancel"
3353
+ onClick: handleAddMoreQuestions,
3354
+ disabled: isGeneratingExtra,
3355
+ "data-testid": "button-add-more-questions",
3356
+ children: isGeneratingExtra ? /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
3357
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Spinner, { size: 16, color: "#9ca3af" }),
3358
+ "Generating Questions..."
3359
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
3360
+ "+ Add ",
3361
+ questionsToAdd,
3362
+ " More Question",
3363
+ questionsToAdd !== 1 ? "s" : ""
3364
+ ] })
2877
3365
  }
2878
3366
  ),
2879
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2880
- "button",
2881
- {
2882
- onClick: () => handleReportQuestion(reportComment),
2883
- disabled: isReporting || !reportComment.trim(),
2884
- style: {
2885
- flex: 1,
2886
- padding: "10px 16px",
2887
- borderRadius: "8px",
2888
- border: "none",
2889
- backgroundColor: reportComment.trim() ? "#ef4444" : "#d1d5db",
2890
- cursor: isReporting || !reportComment.trim() ? "not-allowed" : "pointer",
2891
- fontSize: "14px",
2892
- fontWeight: "500",
2893
- color: "#ffffff",
2894
- opacity: isReporting ? 0.6 : 1
2895
- },
2896
- "data-testid": "button-report-submit",
2897
- children: isReporting ? "Reporting..." : "Report"
2898
- }
2899
- )
3367
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
3368
+ // After viewing feedback
3369
+ isLastQuestion ? /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3370
+ "button",
3371
+ {
3372
+ style: {
3373
+ ...defaultStyles.button,
3374
+ ...isSubmitting || isGeneratingExtra ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
3375
+ },
3376
+ onClick: handleSubmit,
3377
+ disabled: isSubmitting || isGeneratingExtra,
3378
+ "data-testid": "button-submit-quiz",
3379
+ children: isSubmitting ? /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
3380
+ }
3381
+ ) : /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3382
+ "button",
3383
+ {
3384
+ style: {
3385
+ ...defaultStyles.button,
3386
+ ...defaultStyles.buttonPrimary
3387
+ },
3388
+ onClick: handleContinue,
3389
+ "data-testid": "button-continue",
3390
+ children: "Continue"
3391
+ }
3392
+ )
3393
+ ) : (
3394
+ // Before checking answer
3395
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3396
+ "button",
3397
+ {
3398
+ style: {
3399
+ ...defaultStyles.button,
3400
+ ...isNavigating || selectedAnswer === void 0 ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
3401
+ },
3402
+ onClick: handleCheckAnswer,
3403
+ disabled: isNavigating || selectedAnswer === void 0,
3404
+ "data-testid": "button-check-answer",
3405
+ children: isNavigating ? /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
3406
+ }
3407
+ )
3408
+ ) })
2900
3409
  ] })
2901
- ] }) }),
2902
- /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)("div", { style: defaultStyles.buttonsColumn, children: [
2903
- showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2904
- "button",
2905
- {
2906
- style: {
2907
- ...defaultStyles.buttonAddMore,
2908
- ...isGeneratingExtra ? defaultStyles.buttonAddMoreDisabled : {}
2909
- },
2910
- onClick: handleAddMoreQuestions,
2911
- disabled: isGeneratingExtra,
2912
- "data-testid": "button-add-more-questions",
2913
- children: isGeneratingExtra ? /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
2914
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(Spinner, { size: 16, color: "#9ca3af" }),
2915
- "Generating Questions..."
2916
- ] }) : /* @__PURE__ */ (0, import_jsx_runtime3.jsxs)(import_jsx_runtime3.Fragment, { children: [
2917
- "+ Add ",
2918
- questionsToAdd,
2919
- " More Question",
2920
- questionsToAdd !== 1 ? "s" : ""
2921
- ] })
2922
- }
2923
- ),
2924
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
2925
- // After viewing feedback
2926
- isLastQuestion ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2927
- "button",
2928
- {
2929
- style: {
2930
- ...defaultStyles.button,
2931
- ...isSubmitting || isGeneratingExtra ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
2932
- },
2933
- onClick: handleSubmit,
2934
- disabled: isSubmitting || isGeneratingExtra,
2935
- "data-testid": "button-submit-quiz",
2936
- children: isSubmitting ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
2937
- }
2938
- ) : /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2939
- "button",
2940
- {
2941
- style: {
2942
- ...defaultStyles.button,
2943
- ...defaultStyles.buttonPrimary
2944
- },
2945
- onClick: handleContinue,
2946
- "data-testid": "button-continue",
2947
- children: "Continue"
2948
- }
2949
- )
2950
- ) : (
2951
- // Before checking answer
2952
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2953
- "button",
2954
- {
2955
- style: {
2956
- ...defaultStyles.button,
2957
- ...isNavigating || selectedAnswer === void 0 ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
2958
- },
2959
- onClick: handleCheckAnswer,
2960
- disabled: isNavigating || selectedAnswer === void 0,
2961
- "data-testid": "button-check-answer",
2962
- children: isNavigating ? /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
2963
- }
2964
- )
2965
- ) })
2966
- ] })
2967
- ] }),
2968
- /* @__PURE__ */ (0, import_jsx_runtime3.jsx)("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ (0, import_jsx_runtime3.jsx)(
2969
- QuestionChatPanel,
2970
- {
2971
- apiClient: apiClient.current,
2972
- question: {
2973
- id: currentQuestion.id,
2974
- question: currentQuestion.question,
2975
- type: currentQuestion.type,
2976
- options: currentQuestion.options,
2977
- correctAnswer: currentQuestion.correctAnswer,
2978
- explanation: currentQuestion.explanation
2979
- },
2980
- quizId: quiz.id,
2981
- childId,
2982
- parentId,
2983
- lessonId,
2984
- courseId,
2985
- answerResult: showFeedback && currentAnswerDetail ? {
2986
- wasIncorrect: currentQuestion.type !== "assessment" && !currentAnswerDetail.isCorrect,
2987
- selectedAnswer: typeof selectedAnswer === "string" ? selectedAnswer : Array.isArray(selectedAnswer) ? selectedAnswer.join(", ") : void 0,
2988
- correctAnswer: typeof currentQuestion.correctAnswer === "string" ? currentQuestion.correctAnswer : Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer.join(", ") : void 0,
2989
- explanation: currentQuestion.explanation
2990
- } : void 0
2991
- }
2992
- ) })
2993
- ] }) });
3410
+ ] }),
3411
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3412
+ QuestionChatPanel,
3413
+ {
3414
+ apiClient: apiClient.current,
3415
+ question: {
3416
+ id: currentQuestion.id,
3417
+ question: currentQuestion.question,
3418
+ type: currentQuestion.type,
3419
+ options: currentQuestion.options,
3420
+ correctAnswer: currentQuestion.correctAnswer,
3421
+ explanation: currentQuestion.explanation
3422
+ },
3423
+ quizId: quiz.id,
3424
+ childId,
3425
+ parentId,
3426
+ lessonId,
3427
+ courseId,
3428
+ answerResult: showFeedback && currentAnswerDetail ? {
3429
+ wasIncorrect: currentQuestion.type !== "assessment" && !currentAnswerDetail.isCorrect,
3430
+ selectedAnswer: typeof selectedAnswer === "string" ? selectedAnswer : Array.isArray(selectedAnswer) ? selectedAnswer.join(", ") : void 0,
3431
+ correctAnswer: typeof currentQuestion.correctAnswer === "string" ? currentQuestion.correctAnswer : Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer.join(", ") : void 0,
3432
+ explanation: currentQuestion.explanation
3433
+ } : void 0
3434
+ }
3435
+ ) })
3436
+ ] })
3437
+ ] });
2994
3438
  }
2995
3439
 
2996
3440
  // src/AttemptViewer.tsx
2997
3441
  var import_react4 = require("react");
2998
- var import_jsx_runtime4 = require("react/jsx-runtime");
3442
+ var import_jsx_runtime5 = require("react/jsx-runtime");
2999
3443
  var defaultStyles2 = {
3000
3444
  container: {
3001
3445
  fontFamily: "system-ui, -apple-system, sans-serif",
@@ -3138,7 +3582,12 @@ var defaultStyles2 = {
3138
3582
  marginTop: "12px",
3139
3583
  display: "flex",
3140
3584
  flexDirection: "column",
3141
- gap: "8px"
3585
+ gap: "8px",
3586
+ backgroundColor: "#ffffff",
3587
+ padding: "16px",
3588
+ borderRadius: "8px",
3589
+ border: "1px solid #e5e7eb",
3590
+ boxShadow: "0 1px 3px rgba(0, 0, 0, 0.05)"
3142
3591
  },
3143
3592
  chatMessage: {
3144
3593
  padding: "8px 12px",
@@ -3248,23 +3697,39 @@ function AttemptViewer({
3248
3697
  }) {
3249
3698
  const [attempt, setAttempt] = (0, import_react4.useState)(null);
3250
3699
  const [loading, setLoading] = (0, import_react4.useState)(true);
3251
- const [error, setError] = (0, import_react4.useState)(null);
3700
+ const [errorCode, setErrorCode] = (0, import_react4.useState)(null);
3252
3701
  const [chatHistories, setChatHistories] = (0, import_react4.useState)({});
3253
3702
  const [expandedChats, setExpandedChats] = (0, import_react4.useState)(/* @__PURE__ */ new Set());
3703
+ const [fetchedAttemptId, setFetchedAttemptId] = (0, import_react4.useState)(null);
3704
+ const onErrorRef = (0, import_react4.useRef)(onError);
3705
+ onErrorRef.current = onError;
3254
3706
  (0, import_react4.useEffect)(() => {
3707
+ if (fetchedAttemptId === attemptId) return;
3255
3708
  const apiClient = new QuizApiClient({
3256
3709
  baseUrl: apiBaseUrl,
3257
3710
  authToken
3258
3711
  });
3259
3712
  async function fetchAttempt() {
3260
3713
  setLoading(true);
3261
- setError(null);
3714
+ setErrorCode(null);
3262
3715
  try {
3263
3716
  const response = await fetch(`${apiBaseUrl}/api/external/quiz-attempts/${attemptId}`, {
3264
3717
  headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
3265
3718
  });
3266
3719
  if (!response.ok) {
3267
- throw new Error(`Failed to fetch attempt: ${response.statusText}`);
3720
+ const code = getErrorFromHttpStatus(response.status, "attempt");
3721
+ setErrorCode(code);
3722
+ apiClient.logError({
3723
+ errorCode: code,
3724
+ context: "attempt",
3725
+ resourceId: attemptId,
3726
+ component: "AttemptViewer",
3727
+ message: `HTTP ${response.status}: ${response.statusText}`
3728
+ });
3729
+ onErrorRef.current?.(new Error(`Failed to fetch attempt: ${response.statusText}`));
3730
+ setLoading(false);
3731
+ setFetchedAttemptId(attemptId);
3732
+ return;
3268
3733
  }
3269
3734
  const data = await response.json();
3270
3735
  setAttempt(data);
@@ -3278,14 +3743,23 @@ function AttemptViewer({
3278
3743
  }
3279
3744
  } catch (err) {
3280
3745
  const errorMessage = err instanceof Error ? err.message : "Failed to load attempt";
3281
- setError(errorMessage);
3282
- onError?.(err instanceof Error ? err : new Error(errorMessage));
3746
+ const code = getErrorFromMessage(errorMessage, "attempt");
3747
+ setErrorCode(code);
3748
+ apiClient.logError({
3749
+ errorCode: code,
3750
+ context: "attempt",
3751
+ resourceId: attemptId,
3752
+ component: "AttemptViewer",
3753
+ message: errorMessage
3754
+ });
3755
+ onErrorRef.current?.(err instanceof Error ? err : new Error(errorMessage));
3283
3756
  } finally {
3284
3757
  setLoading(false);
3758
+ setFetchedAttemptId(attemptId);
3285
3759
  }
3286
3760
  }
3287
3761
  fetchAttempt();
3288
- }, [attemptId, apiBaseUrl, authToken, onError, showConversation]);
3762
+ }, [attemptId, apiBaseUrl, authToken, showConversation, fetchedAttemptId]);
3289
3763
  const toggleChatExpanded = (questionId) => {
3290
3764
  setExpandedChats((prev) => {
3291
3765
  const newSet = new Set(prev);
@@ -3299,53 +3773,49 @@ function AttemptViewer({
3299
3773
  };
3300
3774
  const handleRetry = () => {
3301
3775
  setLoading(true);
3302
- setError(null);
3776
+ setErrorCode(null);
3303
3777
  window.location.reload();
3304
3778
  };
3305
3779
  if (loading) {
3306
- return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.container, className, children: [
3307
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: spinnerKeyframes }),
3308
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.loading, children: [
3309
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.spinner }),
3310
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { marginTop: "16px", color: "#6b7280" }, children: "Loading attempt..." })
3780
+ return /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.container, className, children: [
3781
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("style", { children: spinnerKeyframes }),
3782
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.loading, children: [
3783
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.spinner }),
3784
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("p", { style: { marginTop: "16px", color: "#6b7280" }, children: "Loading attempt..." })
3311
3785
  ] })
3312
3786
  ] });
3313
3787
  }
3314
- if (error || !attempt) {
3315
- return /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.container, className, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.error, children: [
3316
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { fontSize: "18px", fontWeight: "500" }, children: "Failed to load attempt" }),
3317
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("p", { style: { marginTop: "8px", color: "#6b7280" }, children: error }),
3318
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("button", { style: defaultStyles2.retryButton, onClick: handleRetry, children: "Try Again" })
3319
- ] }) });
3788
+ if (errorCode || !attempt) {
3789
+ 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" }) });
3320
3790
  }
3321
3791
  const scorePercentage = attempt.score ?? 0;
3322
3792
  const correctCount = attempt.correctAnswers ?? 0;
3323
3793
  const totalQuestions = attempt.totalQuestions;
3324
3794
  const timeSpent = attempt.timeSpentSeconds ?? 0;
3325
- return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.container, className, children: [
3326
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("style", { children: spinnerKeyframes }),
3327
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.header, children: /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryGrid, children: [
3328
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3329
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
3795
+ return /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.container, className, children: [
3796
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("style", { children: spinnerKeyframes }),
3797
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.header, children: /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryGrid, children: [
3798
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3799
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
3330
3800
  scorePercentage,
3331
3801
  "%"
3332
3802
  ] }),
3333
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Score" })
3803
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Score" })
3334
3804
  ] }),
3335
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3336
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
3805
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3806
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryValue, children: [
3337
3807
  correctCount,
3338
3808
  "/",
3339
3809
  totalQuestions
3340
3810
  ] }),
3341
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
3811
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
3342
3812
  ] }),
3343
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3344
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryValue, children: formatTime(timeSpent) }),
3345
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Time" })
3813
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.summaryCard, children: [
3814
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryValue, children: formatTime(timeSpent) }),
3815
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.summaryLabel, children: "Time" })
3346
3816
  ] })
3347
3817
  ] }) }),
3348
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
3818
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)(
3349
3819
  "div",
3350
3820
  {
3351
3821
  style: {
@@ -3353,12 +3823,12 @@ function AttemptViewer({
3353
3823
  ...answer.isCorrect ? defaultStyles2.questionCardCorrect : defaultStyles2.questionCardIncorrect
3354
3824
  },
3355
3825
  children: [
3356
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.questionHeader, children: [
3357
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("span", { style: defaultStyles2.questionNumber, children: [
3826
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.questionHeader, children: [
3827
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("span", { style: defaultStyles2.questionNumber, children: [
3358
3828
  "Question ",
3359
3829
  index + 1
3360
3830
  ] }),
3361
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
3831
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)(
3362
3832
  "span",
3363
3833
  {
3364
3834
  style: {
@@ -3369,35 +3839,35 @@ function AttemptViewer({
3369
3839
  }
3370
3840
  )
3371
3841
  ] }),
3372
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: defaultStyles2.questionText, children: answer.questionText }),
3373
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.answerSection, children: [
3374
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: defaultStyles2.answerLabel, children: "Your answer:" }),
3375
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: defaultStyles2.studentAnswer, children: formatAnswer(answer.selectedAnswer, answer.questionType, answer.items, answer.leftItems) })
3842
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("div", { style: defaultStyles2.questionText, children: answer.questionText }),
3843
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.answerSection, children: [
3844
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("span", { style: defaultStyles2.answerLabel, children: "Your answer:" }),
3845
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("span", { style: defaultStyles2.studentAnswer, children: formatAnswer(answer.selectedAnswer, answer.questionType, answer.items, answer.leftItems) })
3376
3846
  ] }),
3377
- !answer.isCorrect && answer.correctAnswer && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.answerSection, children: [
3378
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: defaultStyles2.answerLabel, children: "Correct answer:" }),
3379
- /* @__PURE__ */ (0, import_jsx_runtime4.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) })
3847
+ !answer.isCorrect && answer.correctAnswer && /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.answerSection, children: [
3848
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("span", { style: defaultStyles2.answerLabel, children: "Correct answer:" }),
3849
+ /* @__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) })
3380
3850
  ] }),
3381
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.points, children: [
3851
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.points, children: [
3382
3852
  answer.pointsEarned,
3383
3853
  " / ",
3384
3854
  answer.points,
3385
3855
  " points"
3386
3856
  ] }),
3387
- showExplanations && answer.explanation && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.explanation, children: [
3388
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("strong", { children: "Explanation:" }),
3857
+ showExplanations && answer.explanation && /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.explanation, children: [
3858
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsx)("strong", { children: "Explanation:" }),
3389
3859
  " ",
3390
3860
  answer.explanation
3391
3861
  ] }),
3392
- showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: defaultStyles2.chatHistorySection, children: [
3393
- /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
3862
+ showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)("div", { style: defaultStyles2.chatHistorySection, children: [
3863
+ /* @__PURE__ */ (0, import_jsx_runtime5.jsxs)(
3394
3864
  "button",
3395
3865
  {
3396
3866
  style: defaultStyles2.chatToggleButton,
3397
3867
  onClick: () => toggleChatExpanded(answer.questionId),
3398
3868
  "data-testid": `button-toggle-chat-${answer.questionId}`,
3399
3869
  children: [
3400
- /* @__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" }) }),
3870
+ /* @__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" }) }),
3401
3871
  expandedChats.has(answer.questionId) ? "Hide" : "View",
3402
3872
  " Chat History (",
3403
3873
  chatHistories[answer.questionId].messages.length,
@@ -3405,7 +3875,7 @@ function AttemptViewer({
3405
3875
  ]
3406
3876
  }
3407
3877
  ),
3408
- 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)(
3878
+ 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)(
3409
3879
  "div",
3410
3880
  {
3411
3881
  style: {
@@ -3423,15 +3893,621 @@ function AttemptViewer({
3423
3893
  )) })
3424
3894
  ] });
3425
3895
  }
3896
+
3897
+ // src/ErrorTypesPanel.tsx
3898
+ var import_jsx_runtime6 = require("react/jsx-runtime");
3899
+ var panelStyles2 = {
3900
+ container: {
3901
+ fontFamily: "system-ui, -apple-system, sans-serif",
3902
+ padding: "24px",
3903
+ backgroundColor: "#ffffff",
3904
+ borderRadius: "12px",
3905
+ maxWidth: "800px"
3906
+ },
3907
+ header: {
3908
+ marginBottom: "24px"
3909
+ },
3910
+ title: {
3911
+ fontSize: "20px",
3912
+ fontWeight: "600",
3913
+ color: "#111827",
3914
+ marginBottom: "8px"
3915
+ },
3916
+ subtitle: {
3917
+ fontSize: "14px",
3918
+ color: "#6b7280"
3919
+ },
3920
+ section: {
3921
+ marginBottom: "24px"
3922
+ },
3923
+ sectionTitle: {
3924
+ fontSize: "14px",
3925
+ fontWeight: "600",
3926
+ color: "#374151",
3927
+ marginBottom: "12px",
3928
+ textTransform: "uppercase",
3929
+ letterSpacing: "0.05em"
3930
+ },
3931
+ errorList: {
3932
+ display: "flex",
3933
+ flexDirection: "column",
3934
+ gap: "12px"
3935
+ },
3936
+ errorCard: {
3937
+ padding: "16px",
3938
+ backgroundColor: "#f9fafb",
3939
+ borderRadius: "8px",
3940
+ border: "1px solid #e5e7eb"
3941
+ },
3942
+ errorCardBlocking: {
3943
+ borderLeft: "4px solid #ef4444"
3944
+ },
3945
+ errorCardNonBlocking: {
3946
+ borderLeft: "4px solid #f59e0b"
3947
+ },
3948
+ errorHeader: {
3949
+ display: "flex",
3950
+ justifyContent: "space-between",
3951
+ alignItems: "flex-start",
3952
+ marginBottom: "8px"
3953
+ },
3954
+ errorCode: {
3955
+ fontSize: "13px",
3956
+ fontWeight: "600",
3957
+ color: "#1f2937",
3958
+ fontFamily: "monospace",
3959
+ backgroundColor: "#e5e7eb",
3960
+ padding: "2px 8px",
3961
+ borderRadius: "4px"
3962
+ },
3963
+ errorBadge: {
3964
+ fontSize: "11px",
3965
+ fontWeight: "500",
3966
+ padding: "2px 8px",
3967
+ borderRadius: "12px"
3968
+ },
3969
+ blockingBadge: {
3970
+ backgroundColor: "#fee2e2",
3971
+ color: "#dc2626"
3972
+ },
3973
+ nonBlockingBadge: {
3974
+ backgroundColor: "#fef3c7",
3975
+ color: "#d97706"
3976
+ },
3977
+ userMessage: {
3978
+ fontSize: "15px",
3979
+ fontWeight: "500",
3980
+ color: "#111827",
3981
+ marginBottom: "4px"
3982
+ },
3983
+ subMessage: {
3984
+ fontSize: "13px",
3985
+ color: "#6b7280",
3986
+ marginBottom: "8px"
3987
+ },
3988
+ causeLabel: {
3989
+ fontSize: "11px",
3990
+ fontWeight: "600",
3991
+ color: "#9ca3af",
3992
+ textTransform: "uppercase",
3993
+ marginBottom: "4px"
3994
+ },
3995
+ causeText: {
3996
+ fontSize: "13px",
3997
+ color: "#4b5563",
3998
+ fontStyle: "italic"
3999
+ }
4000
+ };
4001
+ function ErrorTypesPanel({ className }) {
4002
+ const blockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => e.isBlocking);
4003
+ const nonBlockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => !e.isBlocking);
4004
+ const renderErrorCard = (error) => /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)(
4005
+ "div",
4006
+ {
4007
+ style: {
4008
+ ...panelStyles2.errorCard,
4009
+ ...error.isBlocking ? panelStyles2.errorCardBlocking : panelStyles2.errorCardNonBlocking
4010
+ },
4011
+ "data-testid": `error-card-${error.code}`,
4012
+ children: [
4013
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.errorHeader, children: [
4014
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("code", { style: panelStyles2.errorCode, children: error.code }),
4015
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)(
4016
+ "span",
4017
+ {
4018
+ style: {
4019
+ ...panelStyles2.errorBadge,
4020
+ ...error.isBlocking ? panelStyles2.blockingBadge : panelStyles2.nonBlockingBadge
4021
+ },
4022
+ children: error.isBlocking ? "Blocking" : "Non-Blocking"
4023
+ }
4024
+ )
4025
+ ] }),
4026
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.userMessage, children: error.userMessage }),
4027
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.subMessage, children: error.subMessage }),
4028
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.causeLabel, children: "Why this happens:" }),
4029
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.causeText, children: error.cause })
4030
+ ]
4031
+ },
4032
+ error.code
4033
+ );
4034
+ return /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.container, className, "data-testid": "error-types-panel", children: [
4035
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.header, children: [
4036
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("h2", { style: panelStyles2.title, children: "Error Types Reference" }),
4037
+ /* @__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." })
4038
+ ] }),
4039
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.section, children: [
4040
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("h3", { style: panelStyles2.sectionTitle, children: [
4041
+ "Blocking Errors (",
4042
+ blockingErrors.length,
4043
+ ")"
4044
+ ] }),
4045
+ /* @__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." }),
4046
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.errorList, children: blockingErrors.map(renderErrorCard) })
4047
+ ] }),
4048
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("div", { style: panelStyles2.section, children: [
4049
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsxs)("h3", { style: panelStyles2.sectionTitle, children: [
4050
+ "Non-Blocking Errors (",
4051
+ nonBlockingErrors.length,
4052
+ ")"
4053
+ ] }),
4054
+ /* @__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." }),
4055
+ /* @__PURE__ */ (0, import_jsx_runtime6.jsx)("div", { style: panelStyles2.errorList, children: nonBlockingErrors.map(renderErrorCard) })
4056
+ ] })
4057
+ ] });
4058
+ }
4059
+
4060
+ // src/ErrorLogsPanel.tsx
4061
+ var import_react5 = require("react");
4062
+ var import_jsx_runtime7 = require("react/jsx-runtime");
4063
+ var panelStyles3 = {
4064
+ container: {
4065
+ fontFamily: "system-ui, -apple-system, sans-serif",
4066
+ padding: "24px",
4067
+ backgroundColor: "#ffffff",
4068
+ borderRadius: "12px",
4069
+ maxWidth: "1000px"
4070
+ },
4071
+ header: {
4072
+ marginBottom: "24px"
4073
+ },
4074
+ title: {
4075
+ fontSize: "20px",
4076
+ fontWeight: "600",
4077
+ color: "#111827",
4078
+ marginBottom: "8px"
4079
+ },
4080
+ subtitle: {
4081
+ fontSize: "14px",
4082
+ color: "#6b7280"
4083
+ },
4084
+ tabs: {
4085
+ display: "flex",
4086
+ gap: "8px",
4087
+ marginBottom: "16px",
4088
+ borderBottom: "1px solid #e5e7eb",
4089
+ paddingBottom: "12px"
4090
+ },
4091
+ tab: {
4092
+ padding: "8px 16px",
4093
+ fontSize: "14px",
4094
+ fontWeight: "500",
4095
+ borderRadius: "6px",
4096
+ cursor: "pointer",
4097
+ border: "none",
4098
+ backgroundColor: "transparent",
4099
+ color: "#6b7280"
4100
+ },
4101
+ tabActive: {
4102
+ backgroundColor: "#6721b0",
4103
+ color: "#ffffff"
4104
+ },
4105
+ stats: {
4106
+ display: "flex",
4107
+ gap: "16px",
4108
+ marginBottom: "20px"
4109
+ },
4110
+ statCard: {
4111
+ padding: "12px 16px",
4112
+ backgroundColor: "#f9fafb",
4113
+ borderRadius: "8px",
4114
+ flex: 1
4115
+ },
4116
+ statValue: {
4117
+ fontSize: "24px",
4118
+ fontWeight: "700",
4119
+ color: "#111827"
4120
+ },
4121
+ statLabel: {
4122
+ fontSize: "12px",
4123
+ color: "#6b7280",
4124
+ textTransform: "uppercase",
4125
+ letterSpacing: "0.05em"
4126
+ },
4127
+ errorList: {
4128
+ display: "flex",
4129
+ flexDirection: "column",
4130
+ gap: "12px"
4131
+ },
4132
+ errorCard: {
4133
+ padding: "16px",
4134
+ backgroundColor: "#f9fafb",
4135
+ borderRadius: "8px",
4136
+ border: "1px solid #e5e7eb"
4137
+ },
4138
+ errorCardRecent: {
4139
+ borderLeft: "4px solid #ef4444",
4140
+ backgroundColor: "#fef2f2"
4141
+ },
4142
+ errorCardOld: {
4143
+ borderLeft: "4px solid #9ca3af",
4144
+ opacity: 0.7
4145
+ },
4146
+ errorHeader: {
4147
+ display: "flex",
4148
+ justifyContent: "space-between",
4149
+ alignItems: "flex-start",
4150
+ marginBottom: "8px"
4151
+ },
4152
+ errorLeft: {
4153
+ flex: 1
4154
+ },
4155
+ errorCode: {
4156
+ fontSize: "13px",
4157
+ fontWeight: "600",
4158
+ color: "#1f2937",
4159
+ fontFamily: "monospace",
4160
+ backgroundColor: "#e5e7eb",
4161
+ padding: "2px 8px",
4162
+ borderRadius: "4px",
4163
+ display: "inline-block"
4164
+ },
4165
+ countBadge: {
4166
+ fontSize: "14px",
4167
+ fontWeight: "700",
4168
+ padding: "4px 12px",
4169
+ borderRadius: "16px",
4170
+ backgroundColor: "#ef4444",
4171
+ color: "#ffffff"
4172
+ },
4173
+ countBadgeLow: {
4174
+ backgroundColor: "#f59e0b"
4175
+ },
4176
+ contextBadge: {
4177
+ fontSize: "11px",
4178
+ fontWeight: "500",
4179
+ padding: "2px 8px",
4180
+ borderRadius: "4px",
4181
+ marginLeft: "8px",
4182
+ backgroundColor: "#dbeafe",
4183
+ color: "#1d4ed8"
4184
+ },
4185
+ userMessage: {
4186
+ fontSize: "15px",
4187
+ fontWeight: "500",
4188
+ color: "#111827",
4189
+ marginTop: "8px",
4190
+ marginBottom: "4px"
4191
+ },
4192
+ resourceId: {
4193
+ fontSize: "12px",
4194
+ color: "#6b7280",
4195
+ fontFamily: "monospace",
4196
+ marginTop: "4px"
4197
+ },
4198
+ resourceLink: {
4199
+ color: "#6721b0",
4200
+ textDecoration: "underline",
4201
+ cursor: "pointer"
4202
+ },
4203
+ timestamps: {
4204
+ display: "flex",
4205
+ gap: "16px",
4206
+ marginTop: "8px",
4207
+ fontSize: "12px",
4208
+ color: "#9ca3af"
4209
+ },
4210
+ dismissButton: {
4211
+ padding: "6px 12px",
4212
+ fontSize: "12px",
4213
+ fontWeight: "500",
4214
+ borderRadius: "4px",
4215
+ cursor: "pointer",
4216
+ border: "1px solid #e5e7eb",
4217
+ backgroundColor: "#ffffff",
4218
+ color: "#6b7280",
4219
+ marginTop: "8px"
4220
+ },
4221
+ dismissButtonHover: {
4222
+ backgroundColor: "#f3f4f6"
4223
+ },
4224
+ emptyState: {
4225
+ textAlign: "center",
4226
+ padding: "40px",
4227
+ color: "#9ca3af",
4228
+ fontSize: "14px"
4229
+ },
4230
+ loading: {
4231
+ textAlign: "center",
4232
+ padding: "40px",
4233
+ color: "#6b7280",
4234
+ fontSize: "14px"
4235
+ },
4236
+ pagination: {
4237
+ display: "flex",
4238
+ justifyContent: "center",
4239
+ gap: "8px",
4240
+ marginTop: "20px"
4241
+ },
4242
+ pageButton: {
4243
+ padding: "8px 12px",
4244
+ fontSize: "14px",
4245
+ borderRadius: "4px",
4246
+ cursor: "pointer",
4247
+ border: "1px solid #e5e7eb",
4248
+ backgroundColor: "#ffffff",
4249
+ color: "#374151"
4250
+ },
4251
+ pageButtonDisabled: {
4252
+ opacity: 0.5,
4253
+ cursor: "not-allowed"
4254
+ }
4255
+ };
4256
+ function formatRelativeTime(dateStr) {
4257
+ const date = new Date(dateStr);
4258
+ const now = /* @__PURE__ */ new Date();
4259
+ const diffMs = now.getTime() - date.getTime();
4260
+ const diffMins = Math.floor(diffMs / 6e4);
4261
+ const diffHours = Math.floor(diffMs / 36e5);
4262
+ const diffDays = Math.floor(diffMs / 864e5);
4263
+ if (diffMins < 1) return "Just now";
4264
+ if (diffMins < 60) return `${diffMins}m ago`;
4265
+ if (diffHours < 24) return `${diffHours}h ago`;
4266
+ if (diffDays < 7) return `${diffDays}d ago`;
4267
+ return date.toLocaleDateString();
4268
+ }
4269
+ function isRecent(dateStr) {
4270
+ const date = new Date(dateStr);
4271
+ const now = /* @__PURE__ */ new Date();
4272
+ const diffMs = now.getTime() - date.getTime();
4273
+ const diffHours = diffMs / 36e5;
4274
+ return diffHours < 24;
4275
+ }
4276
+ function ErrorLogsPanel({ apiBaseUrl, authToken, onResourceClick }) {
4277
+ const [logs, setLogs] = (0, import_react5.useState)([]);
4278
+ const [loading, setLoading] = (0, import_react5.useState)(true);
4279
+ const [showDismissed, setShowDismissed] = (0, import_react5.useState)(false);
4280
+ const [page, setPage] = (0, import_react5.useState)(1);
4281
+ const [totalPages, setTotalPages] = (0, import_react5.useState)(1);
4282
+ const [total, setTotal] = (0, import_react5.useState)(0);
4283
+ (0, import_react5.useEffect)(() => {
4284
+ fetchLogs();
4285
+ }, [showDismissed, page]);
4286
+ async function fetchLogs() {
4287
+ setLoading(true);
4288
+ try {
4289
+ const params = new URLSearchParams();
4290
+ params.set("dismissed", showDismissed ? "true" : "false");
4291
+ params.set("page", page.toString());
4292
+ params.set("limit", "20");
4293
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs?${params}`, {
4294
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4295
+ });
4296
+ if (response.ok) {
4297
+ const data = await response.json();
4298
+ setLogs(data.items || []);
4299
+ setTotalPages(data.totalPages || 1);
4300
+ setTotal(data.total || 0);
4301
+ }
4302
+ } catch (err) {
4303
+ console.error("Failed to fetch error logs:", err);
4304
+ } finally {
4305
+ setLoading(false);
4306
+ }
4307
+ }
4308
+ async function handleDismiss(id) {
4309
+ try {
4310
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/dismiss`, {
4311
+ method: "PATCH",
4312
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4313
+ });
4314
+ if (response.ok) {
4315
+ fetchLogs();
4316
+ }
4317
+ } catch (err) {
4318
+ console.error("Failed to dismiss error:", err);
4319
+ }
4320
+ }
4321
+ async function handleUndismiss(id) {
4322
+ try {
4323
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/undismiss`, {
4324
+ method: "PATCH",
4325
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4326
+ });
4327
+ if (response.ok) {
4328
+ fetchLogs();
4329
+ }
4330
+ } catch (err) {
4331
+ console.error("Failed to undismiss error:", err);
4332
+ }
4333
+ }
4334
+ function getErrorDefinition(code) {
4335
+ return ERROR_DEFINITIONS[code];
4336
+ }
4337
+ const totalErrors = logs.reduce((sum, log) => sum + log.count, 0);
4338
+ const recentCount = logs.filter((log) => isRecent(log.lastSeenAt)).length;
4339
+ return /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.container, "data-testid": "error-logs-panel", children: [
4340
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.header, children: [
4341
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("h2", { style: panelStyles3.title, children: "Error Tracking" }),
4342
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("p", { style: panelStyles3.subtitle, children: "Aggregated errors from QuizPlayer and AttemptViewer components, sorted by occurrence count" })
4343
+ ] }),
4344
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.tabs, children: [
4345
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4346
+ "button",
4347
+ {
4348
+ style: {
4349
+ ...panelStyles3.tab,
4350
+ ...showDismissed ? {} : panelStyles3.tabActive
4351
+ },
4352
+ onClick: () => {
4353
+ setShowDismissed(false);
4354
+ setPage(1);
4355
+ },
4356
+ "data-testid": "tab-active-errors",
4357
+ children: "Active Errors"
4358
+ }
4359
+ ),
4360
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4361
+ "button",
4362
+ {
4363
+ style: {
4364
+ ...panelStyles3.tab,
4365
+ ...showDismissed ? panelStyles3.tabActive : {}
4366
+ },
4367
+ onClick: () => {
4368
+ setShowDismissed(true);
4369
+ setPage(1);
4370
+ },
4371
+ "data-testid": "tab-dismissed-errors",
4372
+ children: "Dismissed"
4373
+ }
4374
+ )
4375
+ ] }),
4376
+ !showDismissed && /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.stats, children: [
4377
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.statCard, children: [
4378
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statValue, children: total }),
4379
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statLabel, children: "Unique Errors" })
4380
+ ] }),
4381
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.statCard, children: [
4382
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statValue, children: totalErrors }),
4383
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statLabel, children: "Total Occurrences" })
4384
+ ] }),
4385
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.statCard, children: [
4386
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: { ...panelStyles3.statValue, color: recentCount > 0 ? "#ef4444" : "#22c55e" }, children: recentCount }),
4387
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.statLabel, children: "Last 24h" })
4388
+ ] })
4389
+ ] }),
4390
+ 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) => {
4391
+ const def = getErrorDefinition(log.errorCode);
4392
+ const recent = isRecent(log.lastSeenAt);
4393
+ return /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)(
4394
+ "div",
4395
+ {
4396
+ style: {
4397
+ ...panelStyles3.errorCard,
4398
+ ...recent ? panelStyles3.errorCardRecent : panelStyles3.errorCardOld
4399
+ },
4400
+ "data-testid": `error-log-${log.id}`,
4401
+ children: [
4402
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.errorHeader, children: [
4403
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.errorLeft, children: [
4404
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("span", { style: panelStyles3.errorCode, children: log.errorCode }),
4405
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("span", { style: panelStyles3.contextBadge, children: log.context }),
4406
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("span", { style: { ...panelStyles3.contextBadge, backgroundColor: "#f3e8ff", color: "#7c3aed" }, children: log.component })
4407
+ ] }),
4408
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)(
4409
+ "span",
4410
+ {
4411
+ style: {
4412
+ ...panelStyles3.countBadge,
4413
+ ...log.count < 10 ? panelStyles3.countBadgeLow : {}
4414
+ },
4415
+ children: [
4416
+ log.count,
4417
+ "x"
4418
+ ]
4419
+ }
4420
+ )
4421
+ ] }),
4422
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("div", { style: panelStyles3.userMessage, children: def?.userMessage || log.lastMessage || "Unknown error" }),
4423
+ log.resourceId && /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.resourceId, children: [
4424
+ "Resource:",
4425
+ " ",
4426
+ onResourceClick ? /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4427
+ "span",
4428
+ {
4429
+ style: panelStyles3.resourceLink,
4430
+ onClick: () => onResourceClick(log.resourceId, log.context),
4431
+ "data-testid": `link-resource-${log.id}`,
4432
+ children: log.resourceId
4433
+ }
4434
+ ) : log.resourceId
4435
+ ] }),
4436
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.timestamps, children: [
4437
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("span", { children: [
4438
+ "First: ",
4439
+ formatRelativeTime(log.firstSeenAt)
4440
+ ] }),
4441
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("span", { children: [
4442
+ "Last: ",
4443
+ formatRelativeTime(log.lastSeenAt)
4444
+ ] })
4445
+ ] }),
4446
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4447
+ "button",
4448
+ {
4449
+ style: panelStyles3.dismissButton,
4450
+ onClick: () => showDismissed ? handleUndismiss(log.id) : handleDismiss(log.id),
4451
+ "data-testid": `button-dismiss-${log.id}`,
4452
+ children: showDismissed ? "Restore" : "Dismiss"
4453
+ }
4454
+ )
4455
+ ]
4456
+ },
4457
+ log.id
4458
+ );
4459
+ }) }),
4460
+ totalPages > 1 && /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("div", { style: panelStyles3.pagination, children: [
4461
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4462
+ "button",
4463
+ {
4464
+ style: {
4465
+ ...panelStyles3.pageButton,
4466
+ ...page <= 1 ? panelStyles3.pageButtonDisabled : {}
4467
+ },
4468
+ onClick: () => setPage((p) => Math.max(1, p - 1)),
4469
+ disabled: page <= 1,
4470
+ "data-testid": "button-prev-page",
4471
+ children: "Previous"
4472
+ }
4473
+ ),
4474
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsxs)("span", { style: { padding: "8px 12px", color: "#6b7280" }, children: [
4475
+ "Page ",
4476
+ page,
4477
+ " of ",
4478
+ totalPages
4479
+ ] }),
4480
+ /* @__PURE__ */ (0, import_jsx_runtime7.jsx)(
4481
+ "button",
4482
+ {
4483
+ style: {
4484
+ ...panelStyles3.pageButton,
4485
+ ...page >= totalPages ? panelStyles3.pageButtonDisabled : {}
4486
+ },
4487
+ onClick: () => setPage((p) => Math.min(totalPages, p + 1)),
4488
+ disabled: page >= totalPages,
4489
+ "data-testid": "button-next-page",
4490
+ children: "Next"
4491
+ }
4492
+ )
4493
+ ] })
4494
+ ] });
4495
+ }
3426
4496
  // Annotate the CommonJS export names for ESM import in node:
3427
4497
  0 && (module.exports = {
3428
4498
  AttemptViewer,
4499
+ ERROR_DEFINITIONS,
4500
+ ErrorLogsPanel,
4501
+ ErrorTypesPanel,
4502
+ MaintenanceScreen,
3429
4503
  QuizApiClient,
3430
4504
  QuizPlayer,
3431
4505
  TextToSpeech,
3432
4506
  calculateScore,
3433
4507
  checkAnswer,
3434
4508
  createAnswerDetail,
3435
- formatTime
4509
+ formatTime,
4510
+ getErrorFromHttpStatus,
4511
+ getErrorFromMessage
3436
4512
  });
3437
4513
  //# sourceMappingURL=index.js.map