@schoolio/player 1.4.4 → 1.4.5

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