@schoolio/player 1.4.4 → 1.4.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.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",
@@ -1093,7 +1283,9 @@ var defaultStyles = {
1093
1283
  option: {
1094
1284
  width: "100%",
1095
1285
  padding: "12px 16px",
1096
- border: "2px solid #e5e7eb",
1286
+ borderWidth: "2px",
1287
+ borderStyle: "solid",
1288
+ borderColor: "#e5e7eb",
1097
1289
  borderRadius: "8px",
1098
1290
  cursor: "pointer",
1099
1291
  transition: "all 0.2s ease",
@@ -1387,7 +1579,7 @@ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOr
1387
1579
  setDraggedIndex(null);
1388
1580
  setDragOverIndex(null);
1389
1581
  };
1390
- return /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: currentOrder.map((itemIndex, position) => {
1582
+ return /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentOrder.map((itemIndex, position) => {
1391
1583
  const isCorrectPosition = correctOrder?.[position] === itemIndex;
1392
1584
  const isDragging = draggedIndex === position;
1393
1585
  const isDragOver = dragOverIndex === position;
@@ -1410,7 +1602,7 @@ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOr
1410
1602
  } else if (isDragOver) {
1411
1603
  itemStyle = { ...itemStyle, borderColor: "#6366f1", backgroundColor: "#eef2ff" };
1412
1604
  }
1413
- return /* @__PURE__ */ jsxs3(
1605
+ return /* @__PURE__ */ jsxs4(
1414
1606
  "div",
1415
1607
  {
1416
1608
  style: itemStyle,
@@ -1422,34 +1614,34 @@ function SortingDragDrop({ items, currentOrder, correctOrder, showFeedback, onOr
1422
1614
  onDrop: (e) => handleDrop(e, position),
1423
1615
  onDragEnd: handleDragEnd,
1424
1616
  children: [
1425
- !showFeedback && /* @__PURE__ */ jsx3("div", { style: {
1617
+ !showFeedback && /* @__PURE__ */ jsx4("div", { style: {
1426
1618
  cursor: "grab",
1427
1619
  padding: "4px",
1428
1620
  display: "flex",
1429
1621
  alignItems: "center",
1430
1622
  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" })
1623
+ }, children: /* @__PURE__ */ jsxs4("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1624
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "5", r: "1" }),
1625
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "12", r: "1" }),
1626
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "19", r: "1" }),
1627
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "5", r: "1" }),
1628
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "12", r: "1" }),
1629
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "19", r: "1" })
1438
1630
  ] }) }),
1439
- showFeedback && /* @__PURE__ */ jsx3("div", { style: {
1631
+ showFeedback && /* @__PURE__ */ jsx4("div", { style: {
1440
1632
  display: "flex",
1441
1633
  alignItems: "center",
1442
1634
  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" })
1635
+ }, 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: [
1636
+ /* @__PURE__ */ jsx4("path", { d: "M22 11.08V12a10 10 0 1 1-5.93-9.14" }),
1637
+ /* @__PURE__ */ jsx4("polyline", { points: "22 4 12 14.01 9 11.01" })
1638
+ ] }) : /* @__PURE__ */ jsxs4("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1639
+ /* @__PURE__ */ jsx4("circle", { cx: "12", cy: "12", r: "10" }),
1640
+ /* @__PURE__ */ jsx4("line", { x1: "15", y1: "9", x2: "9", y2: "15" }),
1641
+ /* @__PURE__ */ jsx4("line", { x1: "9", y1: "9", x2: "15", y2: "15" })
1450
1642
  ] }) }),
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" }) })
1643
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: items[itemIndex] }),
1644
+ /* @__PURE__ */ jsx4("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx4(TextToSpeech, { text: items[itemIndex], size: "sm" }) })
1453
1645
  ]
1454
1646
  },
1455
1647
  itemIndex
@@ -1503,9 +1695,9 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1503
1695
  delete newMatches[leftItem];
1504
1696
  onMatchChange(newMatches);
1505
1697
  };
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:" }),
1698
+ return /* @__PURE__ */ jsxs4("div", { style: { display: "flex", gap: "24px", flexWrap: "wrap" }, children: [
1699
+ /* @__PURE__ */ jsxs4("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1700
+ /* @__PURE__ */ jsx4("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Match these items:" }),
1509
1701
  leftItems.map((leftItem, idx) => {
1510
1702
  const matchedRight = currentMatches[leftItem];
1511
1703
  const correctMatch = correctMatches?.[leftItem];
@@ -1516,7 +1708,9 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1516
1708
  alignItems: "center",
1517
1709
  gap: "12px",
1518
1710
  padding: "12px 16px",
1519
- border: "2px dashed #e5e7eb",
1711
+ borderWidth: "2px",
1712
+ borderStyle: "dashed",
1713
+ borderColor: "#e5e7eb",
1520
1714
  borderRadius: "8px",
1521
1715
  backgroundColor: "#ffffff",
1522
1716
  minHeight: "56px",
@@ -1534,7 +1728,7 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1534
1728
  } else if (matchedRight) {
1535
1729
  rowStyle = { ...rowStyle, borderStyle: "solid", borderColor: "#22c55e" };
1536
1730
  }
1537
- return /* @__PURE__ */ jsxs3(
1731
+ return /* @__PURE__ */ jsxs4(
1538
1732
  "div",
1539
1733
  {
1540
1734
  style: rowStyle,
@@ -1543,9 +1737,9 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1543
1737
  onDragLeave: handleDragLeave,
1544
1738
  onDrop: (e) => handleDrop(e, leftItem),
1545
1739
  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: {
1740
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1, fontWeight: "500" }, children: leftItem }),
1741
+ /* @__PURE__ */ jsx4("span", { style: { color: "#6b7280" }, children: "\u2192" }),
1742
+ matchedRight ? /* @__PURE__ */ jsxs4("div", { style: {
1549
1743
  display: "flex",
1550
1744
  alignItems: "center",
1551
1745
  gap: "8px",
@@ -1554,8 +1748,8 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1554
1748
  borderRadius: "6px",
1555
1749
  fontSize: "14px"
1556
1750
  }, children: [
1557
- /* @__PURE__ */ jsx3("span", { children: matchedRight }),
1558
- !showFeedback && /* @__PURE__ */ jsx3(
1751
+ /* @__PURE__ */ jsx4("span", { children: matchedRight }),
1752
+ !showFeedback && /* @__PURE__ */ jsx4(
1559
1753
  "button",
1560
1754
  {
1561
1755
  onClick: () => handleClearMatch(leftItem),
@@ -1568,18 +1762,18 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1568
1762
  color: "#6b7280"
1569
1763
  },
1570
1764
  "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" })
1765
+ children: /* @__PURE__ */ jsxs4("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
1766
+ /* @__PURE__ */ jsx4("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1767
+ /* @__PURE__ */ jsx4("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1574
1768
  ] })
1575
1769
  }
1576
1770
  ),
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" })
1771
+ 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: [
1772
+ /* @__PURE__ */ jsx4("line", { x1: "18", y1: "6", x2: "6", y2: "18" }),
1773
+ /* @__PURE__ */ jsx4("line", { x1: "6", y1: "6", x2: "18", y2: "18" })
1580
1774
  ] }))
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: [
1775
+ ] }) : /* @__PURE__ */ jsx4("span", { style: { color: "#9ca3af", fontSize: "14px", fontStyle: "italic" }, children: "Drop here" }),
1776
+ showFeedback && !isCorrect && correctMatch && /* @__PURE__ */ jsxs4("span", { style: { color: "#166534", fontSize: "13px", marginLeft: "8px" }, children: [
1583
1777
  "(Correct: ",
1584
1778
  correctMatch,
1585
1779
  ")"
@@ -1590,16 +1784,18 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1590
1784
  );
1591
1785
  })
1592
1786
  ] }),
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:" }),
1787
+ /* @__PURE__ */ jsxs4("div", { style: { flex: 1, minWidth: "200px", display: "flex", flexDirection: "column", gap: "8px" }, children: [
1788
+ /* @__PURE__ */ jsx4("div", { style: { fontSize: "14px", fontWeight: "600", color: "#6b7280", marginBottom: "4px" }, children: "Drag to match:" }),
1595
1789
  unmatchedRightItems.length > 0 ? unmatchedRightItems.map((rightItem, idx) => {
1596
1790
  const isDragging = draggedItem === rightItem;
1597
- return /* @__PURE__ */ jsxs3(
1791
+ return /* @__PURE__ */ jsxs4(
1598
1792
  "div",
1599
1793
  {
1600
1794
  style: {
1601
1795
  padding: "12px 16px",
1602
- border: "2px solid #e5e7eb",
1796
+ borderWidth: "2px",
1797
+ borderStyle: "solid",
1798
+ borderColor: "#e5e7eb",
1603
1799
  borderRadius: "8px",
1604
1800
  backgroundColor: "#ffffff",
1605
1801
  cursor: showFeedback ? "default" : "grab",
@@ -1614,20 +1810,20 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1614
1810
  onDragEnd: handleDragEnd,
1615
1811
  "data-testid": `draggable-right-${idx}`,
1616
1812
  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" })
1813
+ !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: [
1814
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "5", r: "1" }),
1815
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "12", r: "1" }),
1816
+ /* @__PURE__ */ jsx4("circle", { cx: "9", cy: "19", r: "1" }),
1817
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "5", r: "1" }),
1818
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "12", r: "1" }),
1819
+ /* @__PURE__ */ jsx4("circle", { cx: "15", cy: "19", r: "1" })
1624
1820
  ] }) }),
1625
- /* @__PURE__ */ jsx3("span", { style: { flex: 1 }, children: rightItem })
1821
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: rightItem })
1626
1822
  ]
1627
1823
  },
1628
1824
  idx
1629
1825
  );
1630
- }) : /* @__PURE__ */ jsx3("div", { style: {
1826
+ }) : /* @__PURE__ */ jsx4("div", { style: {
1631
1827
  padding: "16px",
1632
1828
  textAlign: "center",
1633
1829
  color: "#22c55e",
@@ -1637,7 +1833,7 @@ function MatchingDragDrop({ leftItems, rightItems, currentMatches, correctMatche
1637
1833
  ] });
1638
1834
  }
1639
1835
  function Spinner({ size = 16, color = "#ffffff" }) {
1640
- return /* @__PURE__ */ jsx3(
1836
+ return /* @__PURE__ */ jsx4(
1641
1837
  "span",
1642
1838
  {
1643
1839
  style: {
@@ -1649,7 +1845,7 @@ function Spinner({ size = 16, color = "#ffffff" }) {
1649
1845
  borderRadius: "50%",
1650
1846
  animation: "spin 0.8s linear infinite"
1651
1847
  },
1652
- children: /* @__PURE__ */ jsx3("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1848
+ children: /* @__PURE__ */ jsx4("style", { children: `@keyframes spin { to { transform: rotate(360deg); } }` })
1653
1849
  }
1654
1850
  );
1655
1851
  }
@@ -1667,7 +1863,8 @@ function QuizPlayer({
1667
1863
  onProgress,
1668
1864
  onGenerateMoreQuestions,
1669
1865
  className,
1670
- forceNewAttempt = true
1866
+ forceNewAttempt = true,
1867
+ hideScore = false
1671
1868
  }) {
1672
1869
  const [quiz, setQuiz] = useState3(null);
1673
1870
  const [attempt, setAttempt] = useState3(null);
@@ -1678,10 +1875,14 @@ function QuizPlayer({
1678
1875
  const [isNavigating, setIsNavigating] = useState3(false);
1679
1876
  const [isCompleted, setIsCompleted] = useState3(false);
1680
1877
  const [result, setResult] = useState3(null);
1681
- const [error, setError] = useState3(null);
1878
+ const [errorCode, setErrorCode] = useState3(null);
1682
1879
  const [isLoading, setIsLoading] = useState3(true);
1683
1880
  const [elapsedSeconds, setElapsedSeconds] = useState3(0);
1684
1881
  const [showIntro, setShowIntro] = useState3(true);
1882
+ const [showResumeChoice, setShowResumeChoice] = useState3(false);
1883
+ const [hasExistingProgress, setHasExistingProgress] = useState3(false);
1884
+ const [existingProgressCount, setExistingProgressCount] = useState3(0);
1885
+ const [isStartingFresh, setIsStartingFresh] = useState3(false);
1685
1886
  const [timerStarted, setTimerStarted] = useState3(false);
1686
1887
  const [showFeedback, setShowFeedback] = useState3(false);
1687
1888
  const [currentAnswerDetail, setCurrentAnswerDetail] = useState3(null);
@@ -1695,6 +1896,7 @@ function QuizPlayer({
1695
1896
  const [showReportModal, setShowReportModal] = useState3(false);
1696
1897
  const [isReporting, setIsReporting] = useState3(false);
1697
1898
  const [reportComment, setReportComment] = useState3("");
1899
+ const [retryKey, setRetryKey] = useState3(0);
1698
1900
  const apiClient = useRef3(null);
1699
1901
  const timerRef = useRef3(null);
1700
1902
  const startTimeRef = useRef3(0);
@@ -1716,7 +1918,7 @@ function QuizPlayer({
1716
1918
  if (!apiClient.current) return;
1717
1919
  try {
1718
1920
  setIsLoading(true);
1719
- setError(null);
1921
+ setErrorCode(null);
1720
1922
  const quizData = await apiClient.current.getQuiz(quizId);
1721
1923
  setQuiz(quizData);
1722
1924
  const attemptData = await apiClient.current.createAttempt({
@@ -1729,7 +1931,10 @@ function QuizPlayer({
1729
1931
  forceNew: forceNewAttempt
1730
1932
  });
1731
1933
  setAttempt(attemptData);
1732
- if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0) {
1934
+ if (!forceNewAttempt && attemptData.answers && attemptData.answers.length > 0 && attemptData.status === "in_progress") {
1935
+ setHasExistingProgress(true);
1936
+ setExistingProgressCount(attemptData.answers.length);
1937
+ setShowResumeChoice(true);
1733
1938
  setAnswersDetail(attemptData.answers);
1734
1939
  const answersMap = /* @__PURE__ */ new Map();
1735
1940
  attemptData.answers.forEach((a) => {
@@ -1752,15 +1957,23 @@ function QuizPlayer({
1752
1957
  setIsLoading(false);
1753
1958
  } catch (err) {
1754
1959
  const message = err instanceof Error ? err.message : "Failed to load quiz";
1755
- setError(message);
1960
+ const code = getErrorFromMessage(message, "quiz");
1961
+ setErrorCode(code);
1962
+ apiClient.current?.logError({
1963
+ errorCode: code,
1964
+ context: "quiz",
1965
+ resourceId: quizId,
1966
+ component: "QuizPlayer",
1967
+ message
1968
+ });
1756
1969
  setIsLoading(false);
1757
1970
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1758
1971
  }
1759
1972
  }
1760
1973
  initialize();
1761
- }, [quizId, lessonId, assignLessonId, courseId, childId, parentId, forceNewAttempt]);
1974
+ }, [quizId, lessonId, assignLessonId, courseId, childId, parentId, forceNewAttempt, retryKey]);
1762
1975
  useEffect3(() => {
1763
- if (timerStarted && !isCompleted && !error) {
1976
+ if (timerStarted && !isCompleted && !errorCode) {
1764
1977
  startTimeRef.current = Date.now();
1765
1978
  timerRef.current = setInterval(() => {
1766
1979
  setElapsedSeconds(Math.floor((Date.now() - startTimeRef.current) / 1e3));
@@ -1771,11 +1984,62 @@ function QuizPlayer({
1771
1984
  clearInterval(timerRef.current);
1772
1985
  }
1773
1986
  };
1774
- }, [timerStarted, isCompleted, error]);
1987
+ }, [timerStarted, isCompleted, errorCode]);
1775
1988
  const handleStart = useCallback2(() => {
1776
1989
  setShowIntro(false);
1990
+ setShowResumeChoice(false);
1777
1991
  setTimerStarted(true);
1778
1992
  }, []);
1993
+ const handleResumePrevious = useCallback2(() => {
1994
+ if (quiz && answers.size > 0) {
1995
+ const answeredIds = new Set(answers.keys());
1996
+ const allQs = [...quiz.questions, ...extraQuestions].filter((q) => !skippedQuestionIds.has(q.id));
1997
+ let resumeIndex = 0;
1998
+ for (let i = 0; i < allQs.length; i++) {
1999
+ if (!answeredIds.has(allQs[i].id)) {
2000
+ resumeIndex = i;
2001
+ break;
2002
+ }
2003
+ resumeIndex = allQs.length - 1;
2004
+ }
2005
+ setCurrentQuestionIndex(resumeIndex);
2006
+ }
2007
+ setShowIntro(false);
2008
+ setShowResumeChoice(false);
2009
+ setTimerStarted(true);
2010
+ }, [quiz, answers, extraQuestions, skippedQuestionIds]);
2011
+ const handleStartFresh = useCallback2(async () => {
2012
+ if (!apiClient.current || !attempt) return;
2013
+ setIsStartingFresh(true);
2014
+ try {
2015
+ await apiClient.current.updateAttempt(attempt.id, {
2016
+ status: "abandoned"
2017
+ });
2018
+ const newAttemptData = await apiClient.current.createAttempt({
2019
+ quizId,
2020
+ lessonId,
2021
+ assignLessonId,
2022
+ courseId,
2023
+ childId,
2024
+ parentId,
2025
+ forceNew: true
2026
+ });
2027
+ setAttempt(newAttemptData);
2028
+ setAnswers(/* @__PURE__ */ new Map());
2029
+ setAnswersDetail([]);
2030
+ setCurrentQuestionIndex(0);
2031
+ setHasExistingProgress(false);
2032
+ setExistingProgressCount(0);
2033
+ setShowResumeChoice(false);
2034
+ setShowIntro(false);
2035
+ setTimerStarted(true);
2036
+ } catch (err) {
2037
+ const message = err instanceof Error ? err.message : "Failed to start fresh";
2038
+ onErrorRef.current?.(new Error(message));
2039
+ } finally {
2040
+ setIsStartingFresh(false);
2041
+ }
2042
+ }, [attempt, quizId, lessonId, assignLessonId, courseId, childId, parentId]);
1779
2043
  useEffect3(() => {
1780
2044
  setShowFeedback(false);
1781
2045
  setCurrentAnswerDetail(null);
@@ -1895,7 +2159,15 @@ function QuizPlayer({
1895
2159
  onCompleteRef.current?.(quizResult);
1896
2160
  } catch (err) {
1897
2161
  const message = err instanceof Error ? err.message : "Failed to submit quiz";
1898
- setError(message);
2162
+ const code = getErrorFromMessage(message, "quiz");
2163
+ setErrorCode(code);
2164
+ apiClient.current?.logError({
2165
+ errorCode: code,
2166
+ context: "quiz",
2167
+ resourceId: quizId,
2168
+ component: "QuizPlayer",
2169
+ message
2170
+ });
1899
2171
  onErrorRef.current?.(err instanceof Error ? err : new Error(message));
1900
2172
  } finally {
1901
2173
  setIsSubmitting(false);
@@ -1973,14 +2245,43 @@ function QuizPlayer({
1973
2245
  setIsReporting(false);
1974
2246
  }
1975
2247
  }, [currentQuestion, apiClient, attempt, quiz, childId, parentId, lessonId, courseId, assignLessonId]);
2248
+ const handleRetryQuiz = useCallback2(() => {
2249
+ setCurrentQuestionIndex(0);
2250
+ setAnswers(/* @__PURE__ */ new Map());
2251
+ setAnswersDetail([]);
2252
+ setIsCompleted(false);
2253
+ setResult(null);
2254
+ setErrorCode(null);
2255
+ setShowIntro(true);
2256
+ setShowFeedback(false);
2257
+ setCurrentAnswerDetail(null);
2258
+ setTimerStarted(false);
2259
+ setElapsedSeconds(0);
2260
+ startTimeRef.current = 0;
2261
+ setExtraQuestions([]);
2262
+ setSkippedQuestionIds(/* @__PURE__ */ new Set());
2263
+ setAttempt(null);
2264
+ setShowResumeChoice(false);
2265
+ setHasExistingProgress(false);
2266
+ setExistingProgressCount(0);
2267
+ setIsSubmitting(false);
2268
+ setIsNavigating(false);
2269
+ setIsGeneratingExtra(false);
2270
+ setIsStartingFresh(false);
2271
+ setShowSkipModal(false);
2272
+ setIsSkipping(false);
2273
+ setSkipComment("");
2274
+ setSelectedSkipReason(null);
2275
+ setShowReportModal(false);
2276
+ setIsReporting(false);
2277
+ setReportComment("");
2278
+ setRetryKey((prev) => prev + 1);
2279
+ }, []);
1976
2280
  if (isLoading) {
1977
- return /* @__PURE__ */ jsx3("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx3("div", { style: defaultStyles.loading, children: "Loading quiz..." }) });
2281
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx4("div", { style: defaultStyles.loading, children: "Loading quiz..." }) });
1978
2282
  }
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
- ] }) }) });
2283
+ if (errorCode) {
2284
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx4(MaintenanceScreen, { errorCode }) });
1984
2285
  }
1985
2286
  if (isCompleted && result) {
1986
2287
  const percentage = result.totalQuestions > 0 ? Math.round(result.correctAnswers / result.totalQuestions * 100) : 0;
@@ -2038,7 +2339,7 @@ function QuizPlayer({
2038
2339
  rotation: Math.random() * 360,
2039
2340
  size: 6 + Math.random() * 8
2040
2341
  }));
2041
- const StarIcon = ({ filled, delay }) => /* @__PURE__ */ jsx3(
2342
+ const StarIcon = ({ filled, delay }) => /* @__PURE__ */ jsx4(
2042
2343
  "svg",
2043
2344
  {
2044
2345
  width: "36",
@@ -2049,7 +2350,7 @@ function QuizPlayer({
2049
2350
  animationDelay: `${delay}s`,
2050
2351
  opacity: 0
2051
2352
  },
2052
- children: /* @__PURE__ */ jsx3(
2353
+ children: /* @__PURE__ */ jsx4(
2053
2354
  "path",
2054
2355
  {
2055
2356
  d: "M12 2l3.09 6.26L22 9.27l-5 4.87 1.18 6.88L12 17.77l-6.18 3.25L7 14.14 2 9.27l6.91-1.01L12 2z",
@@ -2060,62 +2361,23 @@ function QuizPlayer({
2060
2361
  )
2061
2362
  }
2062
2363
  );
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",
2364
+ const Mascot = ({ mood }) => {
2365
+ return /* @__PURE__ */ jsx4(
2366
+ "img",
2082
2367
  {
2083
- width: "120",
2084
- height: "120",
2085
- viewBox: "0 0 100 100",
2368
+ src: astronautImage,
2369
+ alt: "Astronaut mascot",
2370
+ width: "180",
2371
+ height: "180",
2086
2372
  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
- ]
2373
+ animation: mood === "celebrating" ? "bounce 0.6s ease-in-out infinite" : "gentleBob 2s ease-in-out infinite",
2374
+ objectFit: "contain"
2375
+ }
2114
2376
  }
2115
2377
  );
2116
2378
  };
2117
- return /* @__PURE__ */ jsxs3("div", { className, style: defaultStyles.container, children: [
2118
- /* @__PURE__ */ jsx3("style", { children: `
2379
+ return /* @__PURE__ */ jsxs4("div", { className, style: defaultStyles.container, children: [
2380
+ /* @__PURE__ */ jsx4("style", { children: `
2119
2381
  @keyframes confettiFall {
2120
2382
  0% {
2121
2383
  transform: translateY(-10px) rotate(0deg);
@@ -2178,9 +2440,19 @@ function QuizPlayer({
2178
2440
  opacity: 1;
2179
2441
  }
2180
2442
  }
2443
+ @keyframes buttonFadeIn {
2444
+ 0% {
2445
+ opacity: 0;
2446
+ transform: translateY(10px);
2447
+ }
2448
+ 100% {
2449
+ opacity: 1;
2450
+ transform: translateY(0);
2451
+ }
2452
+ }
2181
2453
  ` }),
2182
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.results, children: [
2183
- percentage >= 60 && /* @__PURE__ */ jsx3("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ jsx3(
2454
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.results, children: [
2455
+ percentage >= 60 && /* @__PURE__ */ jsx4("div", { style: defaultStyles.confettiContainer, children: confettiPieces.map((piece) => /* @__PURE__ */ jsx4(
2184
2456
  "div",
2185
2457
  {
2186
2458
  style: {
@@ -2197,15 +2469,70 @@ function QuizPlayer({
2197
2469
  },
2198
2470
  piece.id
2199
2471
  )) }),
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 })
2472
+ /* @__PURE__ */ jsx4("div", { style: { ...defaultStyles.resultsBackground, background: theme.bgGradient } }),
2473
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.resultsContent, children: hideScore ? /* @__PURE__ */ jsxs4(Fragment2, { children: [
2474
+ /* @__PURE__ */ jsx4("div", { style: { marginBottom: "24px" }, children: /* @__PURE__ */ jsx4(Mascot, { mood: "happy" }) }),
2475
+ /* @__PURE__ */ jsx4(
2476
+ "div",
2477
+ {
2478
+ style: {
2479
+ background: "linear-gradient(135deg, #dcfce7 0%, #bbf7d0 100%)",
2480
+ padding: "12px 28px",
2481
+ borderRadius: "50px",
2482
+ boxShadow: "0 4px 12px rgba(0,0,0,0.15)",
2483
+ marginBottom: "20px",
2484
+ animation: "badgePop 0.6s ease-out 0.2s forwards",
2485
+ opacity: 0,
2486
+ border: "3px solid #22c55e"
2487
+ },
2488
+ children: /* @__PURE__ */ jsx4(
2489
+ "span",
2490
+ {
2491
+ style: {
2492
+ fontSize: "22px",
2493
+ fontWeight: "700",
2494
+ color: "#1f2937",
2495
+ textShadow: "0 1px 2px rgba(255,255,255,0.5)"
2496
+ },
2497
+ children: "All Done!"
2498
+ }
2499
+ )
2500
+ }
2501
+ ),
2502
+ /* @__PURE__ */ jsx4(
2503
+ "div",
2504
+ {
2505
+ style: {
2506
+ animation: "scoreSlideIn 0.5s ease-out 0.4s forwards",
2507
+ opacity: 0
2508
+ },
2509
+ children: /* @__PURE__ */ jsx4(
2510
+ "div",
2511
+ {
2512
+ style: {
2513
+ fontSize: "24px",
2514
+ fontWeight: "600",
2515
+ color: "#22c55e",
2516
+ lineHeight: "1.4",
2517
+ marginBottom: "12px"
2518
+ },
2519
+ children: "The quiz was submitted successfully!"
2520
+ }
2521
+ )
2522
+ }
2523
+ ),
2524
+ /* @__PURE__ */ jsxs4("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2525
+ "Time: ",
2526
+ formatTime(result.timeSpentSeconds)
2527
+ ] })
2528
+ ] }) : /* @__PURE__ */ jsxs4(Fragment2, { children: [
2529
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.resultStars, children: [
2530
+ /* @__PURE__ */ jsx4(StarIcon, { filled: theme.stars >= 1, delay: 0.3 }),
2531
+ /* @__PURE__ */ jsx4(StarIcon, { filled: theme.stars >= 2, delay: 0.5 }),
2532
+ /* @__PURE__ */ jsx4(StarIcon, { filled: theme.stars >= 3, delay: 0.7 })
2206
2533
  ] }),
2207
- /* @__PURE__ */ jsx3("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ jsx3(MascotOwl, { mood: theme.mascotMood }) }),
2208
- /* @__PURE__ */ jsx3(
2534
+ /* @__PURE__ */ jsx4("div", { style: { marginBottom: "16px" }, children: /* @__PURE__ */ jsx4(Mascot, { mood: theme.mascotMood }) }),
2535
+ /* @__PURE__ */ jsx4(
2209
2536
  "div",
2210
2537
  {
2211
2538
  style: {
@@ -2218,7 +2545,7 @@ function QuizPlayer({
2218
2545
  opacity: 0,
2219
2546
  border: `3px solid ${theme.badgeColor}`
2220
2547
  },
2221
- children: /* @__PURE__ */ jsx3(
2548
+ children: /* @__PURE__ */ jsx4(
2222
2549
  "span",
2223
2550
  {
2224
2551
  style: {
@@ -2232,7 +2559,7 @@ function QuizPlayer({
2232
2559
  )
2233
2560
  }
2234
2561
  ),
2235
- /* @__PURE__ */ jsxs3(
2562
+ /* @__PURE__ */ jsxs4(
2236
2563
  "div",
2237
2564
  {
2238
2565
  style: {
@@ -2240,7 +2567,7 @@ function QuizPlayer({
2240
2567
  opacity: 0
2241
2568
  },
2242
2569
  children: [
2243
- /* @__PURE__ */ jsxs3(
2570
+ /* @__PURE__ */ jsxs4(
2244
2571
  "div",
2245
2572
  {
2246
2573
  style: {
@@ -2257,7 +2584,7 @@ function QuizPlayer({
2257
2584
  ]
2258
2585
  }
2259
2586
  ),
2260
- /* @__PURE__ */ jsx3(
2587
+ /* @__PURE__ */ jsx4(
2261
2588
  "div",
2262
2589
  {
2263
2590
  style: {
@@ -2272,25 +2599,116 @@ function QuizPlayer({
2272
2599
  ]
2273
2600
  }
2274
2601
  ),
2275
- /* @__PURE__ */ jsxs3("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2602
+ /* @__PURE__ */ jsxs4("div", { style: { ...defaultStyles.resultDetails, marginTop: "8px" }, children: [
2276
2603
  "Time: ",
2277
2604
  formatTime(result.timeSpentSeconds)
2278
- ] })
2279
- ] })
2605
+ ] }),
2606
+ /* @__PURE__ */ jsx4(
2607
+ "button",
2608
+ {
2609
+ style: {
2610
+ marginTop: "24px",
2611
+ padding: "14px 32px",
2612
+ fontSize: "16px",
2613
+ fontWeight: "600",
2614
+ color: "#7c3aed",
2615
+ background: "white",
2616
+ border: "2px solid #7c3aed",
2617
+ borderRadius: "12px",
2618
+ cursor: "pointer",
2619
+ transition: "all 0.2s ease",
2620
+ animation: "buttonFadeIn 0.5s ease-out 0.8s forwards",
2621
+ opacity: 0
2622
+ },
2623
+ onClick: handleRetryQuiz,
2624
+ onMouseOver: (e) => {
2625
+ e.currentTarget.style.background = "#7c3aed";
2626
+ e.currentTarget.style.color = "white";
2627
+ e.currentTarget.style.transform = "translateY(-2px)";
2628
+ },
2629
+ onMouseOut: (e) => {
2630
+ e.currentTarget.style.background = "white";
2631
+ e.currentTarget.style.color = "#7c3aed";
2632
+ e.currentTarget.style.transform = "translateY(0)";
2633
+ },
2634
+ "data-testid": "button-retry-quiz",
2635
+ children: "Try Again"
2636
+ }
2637
+ )
2638
+ ] }) })
2280
2639
  ] })
2281
2640
  ] });
2282
2641
  }
2642
+ if (quiz && showIntro && showResumeChoice && hasExistingProgress) {
2643
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsxs4("div", { style: defaultStyles.intro, children: [
2644
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introTitle, children: "Welcome Back!" }),
2645
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introSubtitle, children: "You have an unfinished quiz. Would you like to continue or start over?" }),
2646
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.introQuestionCount, children: [
2647
+ existingProgressCount,
2648
+ " of ",
2649
+ quiz.questions.length,
2650
+ " question",
2651
+ quiz.questions.length !== 1 ? "s" : "",
2652
+ " answered"
2653
+ ] }),
2654
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", flexDirection: "column", gap: "12px", marginTop: "8px" }, children: [
2655
+ /* @__PURE__ */ jsx4(
2656
+ "button",
2657
+ {
2658
+ style: defaultStyles.startButton,
2659
+ onClick: handleResumePrevious,
2660
+ onMouseOver: (e) => {
2661
+ e.currentTarget.style.transform = "translateY(-2px)";
2662
+ e.currentTarget.style.boxShadow = "0 6px 20px rgba(124, 58, 237, 0.5)";
2663
+ },
2664
+ onMouseOut: (e) => {
2665
+ e.currentTarget.style.transform = "translateY(0)";
2666
+ e.currentTarget.style.boxShadow = "0 4px 14px rgba(124, 58, 237, 0.4)";
2667
+ },
2668
+ "data-testid": "button-continue-quiz",
2669
+ children: "Continue Where I Left Off"
2670
+ }
2671
+ ),
2672
+ /* @__PURE__ */ jsx4(
2673
+ "button",
2674
+ {
2675
+ style: {
2676
+ ...defaultStyles.startButton,
2677
+ background: "transparent",
2678
+ color: "#7c3aed",
2679
+ border: "2px solid #7c3aed",
2680
+ boxShadow: "none"
2681
+ },
2682
+ onClick: handleStartFresh,
2683
+ disabled: isStartingFresh,
2684
+ onMouseOver: (e) => {
2685
+ if (!isStartingFresh) {
2686
+ e.currentTarget.style.transform = "translateY(-2px)";
2687
+ e.currentTarget.style.background = "rgba(124, 58, 237, 0.1)";
2688
+ }
2689
+ },
2690
+ onMouseOut: (e) => {
2691
+ e.currentTarget.style.transform = "translateY(0)";
2692
+ e.currentTarget.style.background = "transparent";
2693
+ },
2694
+ "data-testid": "button-start-fresh",
2695
+ children: isStartingFresh ? "Starting..." : "Start Fresh"
2696
+ }
2697
+ )
2698
+ ] })
2699
+ ] }) });
2700
+ }
2283
2701
  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: [
2702
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsxs4("div", { style: defaultStyles.intro, children: [
2703
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introTitle, children: quiz.title }),
2704
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.introSubtitle, children: "Ready to test your knowledge?" }),
2705
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.introQuestionCount, children: [
2288
2706
  quiz.questions.length,
2289
2707
  " question",
2290
2708
  quiz.questions.length !== 1 ? "s" : "",
2291
2709
  " to answer"
2292
2710
  ] }),
2293
- /* @__PURE__ */ jsx3(
2711
+ /* @__PURE__ */ jsx4(
2294
2712
  "button",
2295
2713
  {
2296
2714
  style: defaultStyles.startButton,
@@ -2310,7 +2728,7 @@ function QuizPlayer({
2310
2728
  ] }) });
2311
2729
  }
2312
2730
  if (!quiz || !currentQuestion) {
2313
- return /* @__PURE__ */ jsx3("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx3("div", { style: defaultStyles.error, children: "No quiz data available" }) });
2731
+ return /* @__PURE__ */ jsx4("div", { className, style: defaultStyles.container, children: /* @__PURE__ */ jsx4("div", { style: defaultStyles.error, children: "No quiz data available" }) });
2314
2732
  }
2315
2733
  const selectedAnswer = answers.get(currentQuestion.id);
2316
2734
  const isLastQuestion = currentQuestionIndex === totalQuestions - 1;
@@ -2318,294 +2736,166 @@ function QuizPlayer({
2318
2736
  const remainingSlots = maxQuestions - totalQuestions;
2319
2737
  const questionsToAdd = Math.min(5, remainingSlots);
2320
2738
  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: [
2326
- "Question ",
2327
- currentQuestionIndex + 1,
2328
- " of ",
2329
- totalQuestions
2330
- ] }),
2331
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ jsx3("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2332
- ] }),
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(
2336
- "button",
2337
- {
2338
- onClick: () => setShowSkipModal(true),
2339
- title: "Skip question",
2340
- style: {
2341
- position: "absolute",
2342
- bottom: "8px",
2343
- left: "0",
2344
- background: "transparent",
2345
- border: "none",
2346
- cursor: "pointer",
2347
- padding: "6px 10px",
2348
- borderRadius: "6px",
2349
- color: "#9ca3af",
2350
- display: "flex",
2351
- alignItems: "center",
2352
- justifyContent: "center",
2353
- gap: "4px",
2354
- fontSize: "12px",
2355
- opacity: 0.6,
2356
- transition: "opacity 0.2s, color 0.2s"
2357
- },
2358
- onMouseEnter: (e) => {
2359
- e.currentTarget.style.opacity = "1";
2360
- e.currentTarget.style.color = "#6b7280";
2361
- },
2362
- onMouseLeave: (e) => {
2363
- e.currentTarget.style.opacity = "0.6";
2364
- e.currentTarget.style.color = "#9ca3af";
2365
- },
2366
- "data-testid": "button-skip-question",
2367
- 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" })
2371
- ] }),
2372
- /* @__PURE__ */ jsx3("span", { children: "Skip" })
2373
- ]
2374
- }
2375
- ),
2376
- !isExtraQuestion && /* @__PURE__ */ jsxs3(
2377
- "button",
2378
- {
2379
- onClick: () => setShowReportModal(true),
2380
- title: "Report an issue with this question",
2381
- style: {
2382
- position: "absolute",
2383
- bottom: "8px",
2384
- left: "0",
2385
- background: "transparent",
2386
- border: "none",
2387
- cursor: "pointer",
2388
- padding: "6px 10px",
2389
- borderRadius: "6px",
2390
- color: "#9ca3af",
2391
- display: "flex",
2392
- alignItems: "center",
2393
- justifyContent: "center",
2394
- gap: "4px",
2395
- fontSize: "12px",
2396
- opacity: 0.6,
2397
- transition: "opacity 0.2s, color 0.2s"
2398
- },
2399
- onMouseEnter: (e) => {
2400
- e.currentTarget.style.opacity = "1";
2401
- e.currentTarget.style.color = "#ef4444";
2402
- },
2403
- onMouseLeave: (e) => {
2404
- e.currentTarget.style.opacity = "0.6";
2405
- e.currentTarget.style.color = "#9ca3af";
2406
- },
2407
- "data-testid": "button-report-question",
2408
- 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" })
2412
- ] }),
2413
- /* @__PURE__ */ jsx3("span", { children: "Report" })
2414
- ]
2739
+ return /* @__PURE__ */ jsxs4("div", { className, style: defaultStyles.container, children: [
2740
+ /* @__PURE__ */ jsx4("style", { children: `
2741
+ .quiz-option:focus,
2742
+ .quiz-option:active,
2743
+ .quiz-option:focus-visible,
2744
+ .quiz-option:focus-within {
2745
+ outline: none !important;
2746
+ box-shadow: none !important;
2415
2747
  }
2416
- ),
2417
- (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2418
- const isSelected = selectedAnswer === option;
2419
- const isCorrectOption = currentQuestion.correctAnswer === option;
2420
- let optionStyle = { ...defaultStyles.option };
2421
- if (showFeedback) {
2422
- if (isCorrectOption) {
2423
- optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2424
- } else if (isSelected && !isCorrectOption) {
2425
- optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2426
- }
2427
- } else if (isSelected) {
2428
- optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2748
+ .quiz-option {
2749
+ -webkit-tap-highlight-color: transparent;
2429
2750
  }
2430
- return /* @__PURE__ */ jsxs3(
2431
- "div",
2751
+ ` }),
2752
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.mainLayout, children: [
2753
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.quizContent, children: [
2754
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.header, children: [
2755
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.title, children: quiz.title }),
2756
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.progress, children: [
2757
+ "Question ",
2758
+ currentQuestionIndex + 1,
2759
+ " of ",
2760
+ totalQuestions
2761
+ ] }),
2762
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.progressBar, children: /* @__PURE__ */ jsx4("div", { style: { ...defaultStyles.progressFill, width: `${progressPercent}%` } }) })
2763
+ ] }),
2764
+ /* @__PURE__ */ jsxs4("div", { style: { ...defaultStyles.question, position: "relative", paddingBottom: "40px" }, children: [
2765
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.questionText, children: /* @__PURE__ */ jsx4(TextToSpeech, { text: currentQuestion.question, inline: true, size: "md" }) }),
2766
+ isExtraQuestion && /* @__PURE__ */ jsxs4(
2767
+ "button",
2432
2768
  {
2769
+ onClick: () => setShowSkipModal(true),
2770
+ title: "Skip question",
2433
2771
  style: {
2434
- ...optionStyle,
2435
- cursor: showFeedback ? "default" : "pointer",
2772
+ position: "absolute",
2773
+ bottom: "8px",
2774
+ left: "0",
2775
+ background: "transparent",
2776
+ border: "none",
2777
+ cursor: "pointer",
2778
+ padding: "6px 10px",
2779
+ borderRadius: "6px",
2780
+ color: "#9ca3af",
2436
2781
  display: "flex",
2437
2782
  alignItems: "center",
2438
- gap: "8px"
2783
+ justifyContent: "center",
2784
+ gap: "4px",
2785
+ fontSize: "12px",
2786
+ opacity: 0.6,
2787
+ transition: "opacity 0.2s, color 0.2s"
2788
+ },
2789
+ onMouseEnter: (e) => {
2790
+ e.currentTarget.style.opacity = "1";
2791
+ e.currentTarget.style.color = "#6b7280";
2792
+ },
2793
+ onMouseLeave: (e) => {
2794
+ e.currentTarget.style.opacity = "0.6";
2795
+ e.currentTarget.style.color = "#9ca3af";
2439
2796
  },
2440
- onClick: () => !showFeedback && handleAnswerChange(option),
2797
+ "data-testid": "button-skip-question",
2441
2798
  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 })
2799
+ /* @__PURE__ */ jsxs4("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2800
+ /* @__PURE__ */ jsx4("polygon", { points: "5 4 15 12 5 20 5 4" }),
2801
+ /* @__PURE__ */ jsx4("line", { x1: "19", y1: "5", x2: "19", y2: "19" })
2802
+ ] }),
2803
+ /* @__PURE__ */ jsx4("span", { children: "Skip" })
2444
2804
  ]
2445
- },
2446
- idx
2447
- );
2448
- }) }),
2449
- currentQuestion.type === "multiple" && /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2450
- const selected = Array.isArray(selectedAnswer) && selectedAnswer.includes(option);
2451
- const correctAnswers = Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer : currentQuestion.correctAnswer ? [currentQuestion.correctAnswer] : [];
2452
- const isCorrectOption = correctAnswers.includes(option);
2453
- let optionStyle = { ...defaultStyles.option };
2454
- if (showFeedback) {
2455
- if (isCorrectOption) {
2456
- optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2457
- } else if (selected && !isCorrectOption) {
2458
- optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2459
2805
  }
2460
- } else if (selected) {
2461
- optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2462
- }
2463
- return /* @__PURE__ */ jsxs3(
2464
- "div",
2806
+ ),
2807
+ !isExtraQuestion && /* @__PURE__ */ jsxs4(
2808
+ "button",
2465
2809
  {
2810
+ onClick: () => setShowReportModal(true),
2811
+ title: "Report an issue with this question",
2466
2812
  style: {
2467
- ...optionStyle,
2468
- cursor: showFeedback ? "default" : "pointer",
2813
+ position: "absolute",
2814
+ bottom: "8px",
2815
+ left: "0",
2816
+ background: "transparent",
2817
+ border: "none",
2818
+ cursor: "pointer",
2819
+ padding: "6px 10px",
2820
+ borderRadius: "6px",
2821
+ color: "#9ca3af",
2469
2822
  display: "flex",
2470
2823
  alignItems: "center",
2471
- gap: "8px"
2824
+ justifyContent: "center",
2825
+ gap: "4px",
2826
+ fontSize: "12px",
2827
+ opacity: 0.6,
2828
+ transition: "opacity 0.2s, color 0.2s"
2472
2829
  },
2473
- onClick: () => {
2474
- if (showFeedback) return;
2475
- const current = Array.isArray(selectedAnswer) ? selectedAnswer : [];
2476
- if (selected) {
2477
- handleAnswerChange(current.filter((o) => o !== option));
2478
- } else {
2479
- handleAnswerChange([...current, option]);
2480
- }
2830
+ onMouseEnter: (e) => {
2831
+ e.currentTarget.style.opacity = "1";
2832
+ e.currentTarget.style.color = "#ef4444";
2481
2833
  },
2834
+ onMouseLeave: (e) => {
2835
+ e.currentTarget.style.opacity = "0.6";
2836
+ e.currentTarget.style.color = "#9ca3af";
2837
+ },
2838
+ "data-testid": "button-report-question",
2482
2839
  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 })
2840
+ /* @__PURE__ */ jsxs4("svg", { width: "14", height: "14", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", strokeLinecap: "round", strokeLinejoin: "round", children: [
2841
+ /* @__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" }),
2842
+ /* @__PURE__ */ jsx4("line", { x1: "4", y1: "22", x2: "4", y2: "15" })
2843
+ ] }),
2844
+ /* @__PURE__ */ jsx4("span", { children: "Report" })
2485
2845
  ]
2486
- },
2487
- idx
2488
- );
2489
- }) }),
2490
- (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ jsx3(
2491
- "textarea",
2492
- {
2493
- style: { ...defaultStyles.input, minHeight: currentQuestion.type === "essay" ? "150px" : "60px" },
2494
- value: selectedAnswer || "",
2495
- onChange: (e) => handleAnswerChange(e.target.value),
2496
- placeholder: "Type your answer here...",
2497
- disabled: showFeedback
2498
- }
2499
- ),
2500
- currentQuestion.type === "fill" && /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ jsx3(
2501
- "input",
2502
- {
2503
- style: defaultStyles.input,
2504
- value: (Array.isArray(selectedAnswer) ? selectedAnswer[idx] : "") || "",
2505
- onChange: (e) => {
2506
- const current = Array.isArray(selectedAnswer) ? [...selectedAnswer] : [];
2507
- current[idx] = e.target.value;
2508
- handleAnswerChange(current);
2509
- },
2510
- placeholder: `Blank ${idx + 1}`,
2511
- disabled: showFeedback
2512
- },
2513
- idx
2514
- )) }),
2515
- currentQuestion.type === "sorting" && currentQuestion.items && /* @__PURE__ */ jsx3(
2516
- SortingDragDrop,
2517
- {
2518
- items: currentQuestion.items,
2519
- currentOrder: Array.isArray(selectedAnswer) ? selectedAnswer : currentQuestion.items.map((_, i) => i),
2520
- correctOrder: currentQuestion.correctOrder,
2521
- showFeedback,
2522
- onOrderChange: handleAnswerChange
2523
- }
2524
- ),
2525
- currentQuestion.type === "matrix" && currentQuestion.leftItems && currentQuestion.rightItems && /* @__PURE__ */ jsx3(
2526
- MatchingDragDrop,
2527
- {
2528
- leftItems: currentQuestion.leftItems,
2529
- rightItems: currentQuestion.rightItems,
2530
- currentMatches: typeof selectedAnswer === "object" && selectedAnswer !== null && !Array.isArray(selectedAnswer) ? selectedAnswer : {},
2531
- correctMatches: currentQuestion.correctMatches,
2532
- showFeedback,
2533
- onMatchChange: handleAnswerChange
2534
- }
2535
- ),
2536
- currentQuestion.type === "assessment" && /* @__PURE__ */ jsx3("div", { style: defaultStyles.options, children: (() => {
2537
- const scaleType = currentQuestion.scaleType || "likert";
2538
- if (scaleType === "yes-no") {
2539
- const options = ["Yes", "No"];
2540
- return options.map((option, idx) => {
2541
- const isSelected = selectedAnswer === option;
2542
- let optionStyle = { ...defaultStyles.option };
2543
- if (isSelected) {
2544
- optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2846
+ }
2847
+ ),
2848
+ (currentQuestion.type === "single" || currentQuestion.type === "true-false") && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2849
+ const isSelected = selectedAnswer === option;
2850
+ const isCorrectOption = currentQuestion.correctAnswer === option;
2851
+ let optionStyle = { ...defaultStyles.option };
2852
+ if (showFeedback) {
2853
+ if (isCorrectOption) {
2854
+ optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2855
+ } else if (isSelected && !isCorrectOption) {
2856
+ optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2545
2857
  }
2546
- return /* @__PURE__ */ jsx3(
2547
- "div",
2548
- {
2549
- style: {
2550
- ...optionStyle,
2551
- cursor: showFeedback ? "default" : "pointer",
2552
- display: "flex",
2553
- alignItems: "center",
2554
- gap: "8px"
2555
- },
2556
- onClick: () => !showFeedback && handleAnswerChange(option),
2557
- "data-testid": `assessment-option-${option.toLowerCase()}`,
2558
- children: /* @__PURE__ */ jsx3("span", { style: { flex: 1 }, children: option })
2559
- },
2560
- idx
2561
- );
2562
- });
2563
- }
2564
- if (scaleType === "rating") {
2565
- const min = currentQuestion.scaleMin || 1;
2566
- const max = currentQuestion.scaleMax || 5;
2567
- 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) => {
2569
- const isSelected = selectedAnswer === rating;
2570
- return /* @__PURE__ */ jsx3(
2571
- "button",
2572
- {
2573
- onClick: () => !showFeedback && handleAnswerChange(rating),
2574
- disabled: showFeedback,
2575
- style: {
2576
- width: "48px",
2577
- height: "48px",
2578
- borderRadius: "50%",
2579
- border: isSelected ? "2px solid #6721b0" : "2px solid #e5e7eb",
2580
- backgroundColor: isSelected ? "#f3e8ff" : "#ffffff",
2581
- cursor: showFeedback ? "not-allowed" : "pointer",
2582
- fontSize: "18px",
2583
- fontWeight: "600",
2584
- color: isSelected ? "#6721b0" : "#374151"
2585
- },
2586
- "data-testid": `assessment-rating-${rating}`,
2587
- children: rating
2858
+ } else if (isSelected) {
2859
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2860
+ }
2861
+ return /* @__PURE__ */ jsxs4(
2862
+ "div",
2863
+ {
2864
+ className: "quiz-option",
2865
+ style: {
2866
+ ...optionStyle,
2867
+ cursor: showFeedback ? "default" : "pointer",
2868
+ display: "flex",
2869
+ alignItems: "center",
2870
+ gap: "8px"
2588
2871
  },
2589
- rating
2590
- );
2591
- }) });
2592
- }
2593
- const likertOptions = [
2594
- "Strongly Disagree",
2595
- "Disagree",
2596
- "Neutral",
2597
- "Agree",
2598
- "Strongly Agree"
2599
- ];
2600
- return likertOptions.map((option, idx) => {
2601
- const isSelected = selectedAnswer === option;
2872
+ onClick: () => !showFeedback && handleAnswerChange(option),
2873
+ children: [
2874
+ /* @__PURE__ */ jsx4("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx4(TextToSpeech, { text: option, size: "sm" }) }),
2875
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
2876
+ ]
2877
+ },
2878
+ idx
2879
+ );
2880
+ }) }),
2881
+ currentQuestion.type === "multiple" && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentQuestion.options?.map((option, idx) => {
2882
+ const selected = Array.isArray(selectedAnswer) && selectedAnswer.includes(option);
2883
+ const correctAnswers = Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer : currentQuestion.correctAnswer ? [currentQuestion.correctAnswer] : [];
2884
+ const isCorrectOption = correctAnswers.includes(option);
2602
2885
  let optionStyle = { ...defaultStyles.option };
2603
- if (isSelected) {
2886
+ if (showFeedback) {
2887
+ if (isCorrectOption) {
2888
+ optionStyle = { ...optionStyle, ...defaultStyles.optionCorrect };
2889
+ } else if (selected && !isCorrectOption) {
2890
+ optionStyle = { ...optionStyle, ...defaultStyles.optionIncorrect };
2891
+ }
2892
+ } else if (selected) {
2604
2893
  optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2605
2894
  }
2606
- return /* @__PURE__ */ jsx3(
2895
+ return /* @__PURE__ */ jsxs4(
2607
2896
  "div",
2608
2897
  {
2898
+ className: "quiz-option",
2609
2899
  style: {
2610
2900
  ...optionStyle,
2611
2901
  cursor: showFeedback ? "default" : "pointer",
@@ -2613,356 +2903,504 @@ function QuizPlayer({
2613
2903
  alignItems: "center",
2614
2904
  gap: "8px"
2615
2905
  },
2616
- onClick: () => !showFeedback && handleAnswerChange(option),
2617
- "data-testid": `assessment-likert-${idx}`,
2618
- children: /* @__PURE__ */ jsx3("span", { style: { flex: 1 }, children: option })
2906
+ onClick: () => {
2907
+ if (showFeedback) return;
2908
+ const current = Array.isArray(selectedAnswer) ? selectedAnswer : [];
2909
+ if (selected) {
2910
+ handleAnswerChange(current.filter((o) => o !== option));
2911
+ } else {
2912
+ handleAnswerChange([...current, option]);
2913
+ }
2914
+ },
2915
+ children: [
2916
+ /* @__PURE__ */ jsx4("span", { onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsx4(TextToSpeech, { text: option, size: "sm" }) }),
2917
+ /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
2918
+ ]
2619
2919
  },
2620
2920
  idx
2621
2921
  );
2622
- });
2623
- })() }),
2624
- showFeedback && currentAnswerDetail && /* @__PURE__ */ jsxs3("div", { style: {
2625
- ...defaultStyles.feedback,
2626
- ...currentQuestion.type === "assessment" ? defaultStyles.feedbackNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
2627
- }, children: [
2628
- /* @__PURE__ */ jsx3("div", { style: {
2629
- ...defaultStyles.feedbackTitle,
2630
- ...currentQuestion.type === "assessment" ? defaultStyles.feedbackTitleNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
2631
- }, 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 })
2633
- ] })
2634
- ] }),
2635
- showSkipModal && /* @__PURE__ */ jsx3("div", { style: {
2636
- position: "fixed",
2637
- top: 0,
2638
- left: 0,
2639
- right: 0,
2640
- bottom: 0,
2641
- backgroundColor: "rgba(0, 0, 0, 0.5)",
2642
- display: "flex",
2643
- alignItems: "center",
2644
- justifyContent: "center",
2645
- zIndex: 1e3
2646
- }, children: /* @__PURE__ */ jsxs3("div", { style: {
2647
- backgroundColor: "#ffffff",
2648
- borderRadius: "12px",
2649
- padding: "24px",
2650
- maxWidth: "400px",
2651
- width: "90%",
2652
- boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2653
- }, 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(
2658
- "button",
2922
+ }) }),
2923
+ (currentQuestion.type === "free" || currentQuestion.type === "essay") && /* @__PURE__ */ jsx4(
2924
+ "textarea",
2659
2925
  {
2660
- onClick: () => setSelectedSkipReason("question_issue"),
2661
- disabled: isSkipping,
2662
- style: {
2663
- padding: "12px 16px",
2664
- borderRadius: "8px",
2665
- border: selectedSkipReason === "question_issue" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
2666
- backgroundColor: selectedSkipReason === "question_issue" ? "#f5f3ff" : "#f9fafb",
2667
- cursor: isSkipping ? "not-allowed" : "pointer",
2668
- fontSize: "14px",
2669
- fontWeight: "500",
2670
- color: "#374151",
2671
- textAlign: "left",
2672
- opacity: isSkipping ? 0.6 : 1
2673
- },
2674
- "data-testid": "button-skip-reason-issue",
2675
- children: "Question has an issue"
2926
+ style: { ...defaultStyles.input, minHeight: currentQuestion.type === "essay" ? "150px" : "60px" },
2927
+ value: selectedAnswer || "",
2928
+ onChange: (e) => handleAnswerChange(e.target.value),
2929
+ placeholder: "Type your answer here...",
2930
+ disabled: showFeedback
2676
2931
  }
2677
2932
  ),
2678
- /* @__PURE__ */ jsx3(
2679
- "button",
2933
+ currentQuestion.type === "fill" && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: currentQuestion.blanks?.map((_, idx) => /* @__PURE__ */ jsx4(
2934
+ "input",
2680
2935
  {
2681
- onClick: () => setSelectedSkipReason("dont_know"),
2682
- disabled: isSkipping,
2683
- style: {
2684
- padding: "12px 16px",
2685
- borderRadius: "8px",
2686
- border: selectedSkipReason === "dont_know" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
2687
- backgroundColor: selectedSkipReason === "dont_know" ? "#f5f3ff" : "#f9fafb",
2688
- cursor: isSkipping ? "not-allowed" : "pointer",
2689
- fontSize: "14px",
2690
- fontWeight: "500",
2691
- color: "#374151",
2692
- textAlign: "left",
2693
- opacity: isSkipping ? 0.6 : 1
2936
+ style: defaultStyles.input,
2937
+ value: (Array.isArray(selectedAnswer) ? selectedAnswer[idx] : "") || "",
2938
+ onChange: (e) => {
2939
+ const current = Array.isArray(selectedAnswer) ? [...selectedAnswer] : [];
2940
+ current[idx] = e.target.value;
2941
+ handleAnswerChange(current);
2694
2942
  },
2695
- "data-testid": "button-skip-reason-dont-know",
2696
- children: "I don't know the answer"
2697
- }
2698
- )
2699
- ] }),
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(
2703
- "textarea",
2943
+ placeholder: `Blank ${idx + 1}`,
2944
+ disabled: showFeedback
2945
+ },
2946
+ idx
2947
+ )) }),
2948
+ currentQuestion.type === "sorting" && currentQuestion.items && /* @__PURE__ */ jsx4(
2949
+ SortingDragDrop,
2704
2950
  {
2705
- value: skipComment,
2706
- onChange: (e) => setSkipComment(e.target.value.slice(0, 200)),
2707
- placeholder: "Tell us more about the issue...",
2708
- disabled: isSkipping,
2709
- style: {
2710
- width: "100%",
2711
- minHeight: "80px",
2712
- padding: "10px 12px",
2713
- borderRadius: "8px",
2714
- border: "1px solid #e5e7eb",
2715
- fontSize: "14px",
2716
- resize: "vertical",
2717
- fontFamily: "inherit",
2718
- boxSizing: "border-box"
2719
- },
2720
- "data-testid": "input-skip-comment"
2951
+ items: currentQuestion.items,
2952
+ currentOrder: Array.isArray(selectedAnswer) ? selectedAnswer : currentQuestion.items.map((_, i) => i),
2953
+ correctOrder: currentQuestion.correctOrder,
2954
+ showFeedback,
2955
+ onOrderChange: handleAnswerChange
2721
2956
  }
2722
2957
  ),
2723
- /* @__PURE__ */ jsxs3("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2724
- skipComment.length,
2725
- "/200"
2726
- ] })
2727
- ] }),
2728
- /* @__PURE__ */ jsxs3("div", { style: { display: "flex", gap: "10px" }, children: [
2729
- /* @__PURE__ */ jsx3(
2730
- "button",
2958
+ currentQuestion.type === "matrix" && currentQuestion.leftItems && currentQuestion.rightItems && /* @__PURE__ */ jsx4(
2959
+ MatchingDragDrop,
2731
2960
  {
2732
- onClick: () => {
2733
- setShowSkipModal(false);
2734
- setSkipComment("");
2735
- setSelectedSkipReason(null);
2736
- },
2737
- style: {
2738
- flex: 1,
2739
- padding: "10px 16px",
2740
- borderRadius: "8px",
2741
- border: "1px solid #e5e7eb",
2742
- backgroundColor: "#ffffff",
2743
- cursor: "pointer",
2744
- fontSize: "14px",
2745
- fontWeight: "500",
2746
- color: "#6b7280"
2747
- },
2748
- "data-testid": "button-skip-cancel",
2749
- children: "Cancel"
2961
+ leftItems: currentQuestion.leftItems,
2962
+ rightItems: currentQuestion.rightItems,
2963
+ currentMatches: typeof selectedAnswer === "object" && selectedAnswer !== null && !Array.isArray(selectedAnswer) ? selectedAnswer : {},
2964
+ correctMatches: currentQuestion.correctMatches,
2965
+ showFeedback,
2966
+ onMatchChange: handleAnswerChange
2750
2967
  }
2751
2968
  ),
2752
- /* @__PURE__ */ jsx3(
2753
- "button",
2754
- {
2755
- onClick: () => selectedSkipReason && handleSkipQuestion(selectedSkipReason, skipComment),
2756
- disabled: isSkipping || !selectedSkipReason,
2757
- style: {
2758
- flex: 1,
2759
- padding: "10px 16px",
2760
- borderRadius: "8px",
2761
- border: "none",
2762
- backgroundColor: selectedSkipReason ? "#8b5cf6" : "#d1d5db",
2763
- cursor: isSkipping || !selectedSkipReason ? "not-allowed" : "pointer",
2764
- fontSize: "14px",
2765
- fontWeight: "500",
2766
- color: "#ffffff",
2767
- opacity: isSkipping ? 0.6 : 1
2768
- },
2769
- "data-testid": "button-skip-submit",
2770
- children: isSkipping ? "Skipping..." : "Skip Question"
2969
+ currentQuestion.type === "assessment" && /* @__PURE__ */ jsx4("div", { style: defaultStyles.options, children: (() => {
2970
+ const scaleType = currentQuestion.scaleType || "likert";
2971
+ if (scaleType === "yes-no") {
2972
+ const options = ["Yes", "No"];
2973
+ return options.map((option, idx) => {
2974
+ const isSelected = selectedAnswer === option;
2975
+ let optionStyle = { ...defaultStyles.option };
2976
+ if (isSelected) {
2977
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
2978
+ }
2979
+ return /* @__PURE__ */ jsx4(
2980
+ "div",
2981
+ {
2982
+ className: "quiz-option",
2983
+ style: {
2984
+ ...optionStyle,
2985
+ cursor: showFeedback ? "default" : "pointer",
2986
+ display: "flex",
2987
+ alignItems: "center",
2988
+ gap: "8px"
2989
+ },
2990
+ onClick: () => !showFeedback && handleAnswerChange(option),
2991
+ "data-testid": `assessment-option-${option.toLowerCase()}`,
2992
+ children: /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
2993
+ },
2994
+ idx
2995
+ );
2996
+ });
2771
2997
  }
2772
- )
2773
- ] })
2774
- ] }) }),
2775
- showReportModal && /* @__PURE__ */ jsx3("div", { style: {
2776
- position: "fixed",
2777
- top: 0,
2778
- left: 0,
2779
- right: 0,
2780
- bottom: 0,
2781
- backgroundColor: "rgba(0, 0, 0, 0.5)",
2782
- display: "flex",
2783
- alignItems: "center",
2784
- justifyContent: "center",
2785
- zIndex: 1e3
2786
- }, children: /* @__PURE__ */ jsxs3("div", { style: {
2787
- backgroundColor: "#ffffff",
2788
- borderRadius: "12px",
2789
- padding: "24px",
2790
- maxWidth: "400px",
2791
- width: "90%",
2792
- boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
2793
- }, 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(
2798
- "textarea",
2799
- {
2800
- value: reportComment,
2801
- onChange: (e) => setReportComment(e.target.value.slice(0, 300)),
2802
- placeholder: "Describe the issue with this question...",
2803
- disabled: isReporting,
2804
- style: {
2805
- width: "100%",
2806
- minHeight: "120px",
2807
- padding: "10px 12px",
2808
- borderRadius: "8px",
2809
- border: "1px solid #e5e7eb",
2810
- fontSize: "14px",
2811
- resize: "vertical",
2812
- fontFamily: "inherit",
2813
- boxSizing: "border-box"
2814
- },
2815
- "data-testid": "input-report-comment"
2998
+ if (scaleType === "rating") {
2999
+ const min = currentQuestion.scaleMin || 1;
3000
+ const max = currentQuestion.scaleMax || 5;
3001
+ const ratings = Array.from({ length: max - min + 1 }, (_, i) => min + i);
3002
+ return /* @__PURE__ */ jsx4("div", { style: { display: "flex", gap: "8px", flexWrap: "wrap", justifyContent: "center" }, children: ratings.map((rating) => {
3003
+ const isSelected = selectedAnswer === rating;
3004
+ return /* @__PURE__ */ jsx4(
3005
+ "button",
3006
+ {
3007
+ className: "quiz-option",
3008
+ onClick: () => !showFeedback && handleAnswerChange(rating),
3009
+ disabled: showFeedback,
3010
+ style: {
3011
+ width: "48px",
3012
+ height: "48px",
3013
+ borderRadius: "50%",
3014
+ border: isSelected ? "2px solid #6721b0" : "2px solid #e5e7eb",
3015
+ backgroundColor: isSelected ? "#f3e8ff" : "#ffffff",
3016
+ cursor: showFeedback ? "not-allowed" : "pointer",
3017
+ fontSize: "18px",
3018
+ fontWeight: "600",
3019
+ color: isSelected ? "#6721b0" : "#374151",
3020
+ outline: "none"
3021
+ },
3022
+ "data-testid": `assessment-rating-${rating}`,
3023
+ children: rating
3024
+ },
3025
+ rating
3026
+ );
3027
+ }) });
2816
3028
  }
2817
- ),
2818
- /* @__PURE__ */ jsxs3("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
2819
- reportComment.length,
2820
- "/300"
3029
+ const likertOptions = [
3030
+ "Strongly Disagree",
3031
+ "Disagree",
3032
+ "Neutral",
3033
+ "Agree",
3034
+ "Strongly Agree"
3035
+ ];
3036
+ return likertOptions.map((option, idx) => {
3037
+ const isSelected = selectedAnswer === option;
3038
+ let optionStyle = { ...defaultStyles.option };
3039
+ if (isSelected) {
3040
+ optionStyle = { ...optionStyle, ...defaultStyles.optionSelected };
3041
+ }
3042
+ return /* @__PURE__ */ jsx4(
3043
+ "div",
3044
+ {
3045
+ className: "quiz-option",
3046
+ style: {
3047
+ ...optionStyle,
3048
+ cursor: showFeedback ? "default" : "pointer",
3049
+ display: "flex",
3050
+ alignItems: "center",
3051
+ gap: "8px"
3052
+ },
3053
+ onClick: () => !showFeedback && handleAnswerChange(option),
3054
+ "data-testid": `assessment-likert-${idx}`,
3055
+ children: /* @__PURE__ */ jsx4("span", { style: { flex: 1 }, children: option })
3056
+ },
3057
+ idx
3058
+ );
3059
+ });
3060
+ })() }),
3061
+ showFeedback && currentAnswerDetail && /* @__PURE__ */ jsxs4("div", { style: {
3062
+ ...defaultStyles.feedback,
3063
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackCorrect : defaultStyles.feedbackIncorrect
3064
+ }, children: [
3065
+ /* @__PURE__ */ jsx4("div", { style: {
3066
+ ...defaultStyles.feedbackTitle,
3067
+ ...currentQuestion.type === "assessment" ? defaultStyles.feedbackTitleNeutral : currentAnswerDetail.isCorrect ? defaultStyles.feedbackTitleCorrect : defaultStyles.feedbackTitleIncorrect
3068
+ }, children: currentQuestion.type === "assessment" ? "\u2713 Response recorded" : currentAnswerDetail.isCorrect ? "\u2713 Correct!" : "\u2717 Incorrect" }),
3069
+ currentQuestion.explanation && /* @__PURE__ */ jsx4("div", { style: defaultStyles.feedbackExplanation, children: currentQuestion.explanation })
2821
3070
  ] })
2822
3071
  ] }),
2823
- /* @__PURE__ */ jsxs3("div", { style: { display: "flex", gap: "10px" }, children: [
2824
- /* @__PURE__ */ jsx3(
3072
+ showSkipModal && /* @__PURE__ */ jsx4("div", { style: {
3073
+ position: "fixed",
3074
+ top: 0,
3075
+ left: 0,
3076
+ right: 0,
3077
+ bottom: 0,
3078
+ backgroundColor: "rgba(0, 0, 0, 0.5)",
3079
+ display: "flex",
3080
+ alignItems: "center",
3081
+ justifyContent: "center",
3082
+ zIndex: 1e3
3083
+ }, children: /* @__PURE__ */ jsxs4("div", { style: {
3084
+ backgroundColor: "#ffffff",
3085
+ borderRadius: "12px",
3086
+ padding: "24px",
3087
+ maxWidth: "400px",
3088
+ width: "90%",
3089
+ boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
3090
+ }, children: [
3091
+ /* @__PURE__ */ jsx4("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Skip Question" }),
3092
+ /* @__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." }),
3093
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", flexDirection: "column", gap: "8px", marginBottom: "16px" }, children: [
3094
+ /* @__PURE__ */ jsx4(
3095
+ "button",
3096
+ {
3097
+ onClick: () => setSelectedSkipReason("question_issue"),
3098
+ disabled: isSkipping,
3099
+ style: {
3100
+ padding: "12px 16px",
3101
+ borderRadius: "8px",
3102
+ border: selectedSkipReason === "question_issue" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
3103
+ backgroundColor: selectedSkipReason === "question_issue" ? "#f5f3ff" : "#f9fafb",
3104
+ cursor: isSkipping ? "not-allowed" : "pointer",
3105
+ fontSize: "14px",
3106
+ fontWeight: "500",
3107
+ color: "#374151",
3108
+ textAlign: "left",
3109
+ opacity: isSkipping ? 0.6 : 1
3110
+ },
3111
+ "data-testid": "button-skip-reason-issue",
3112
+ children: "Question has an issue"
3113
+ }
3114
+ ),
3115
+ /* @__PURE__ */ jsx4(
3116
+ "button",
3117
+ {
3118
+ onClick: () => setSelectedSkipReason("dont_know"),
3119
+ disabled: isSkipping,
3120
+ style: {
3121
+ padding: "12px 16px",
3122
+ borderRadius: "8px",
3123
+ border: selectedSkipReason === "dont_know" ? "2px solid #8b5cf6" : "1px solid #e5e7eb",
3124
+ backgroundColor: selectedSkipReason === "dont_know" ? "#f5f3ff" : "#f9fafb",
3125
+ cursor: isSkipping ? "not-allowed" : "pointer",
3126
+ fontSize: "14px",
3127
+ fontWeight: "500",
3128
+ color: "#374151",
3129
+ textAlign: "left",
3130
+ opacity: isSkipping ? 0.6 : 1
3131
+ },
3132
+ "data-testid": "button-skip-reason-dont-know",
3133
+ children: "I don't know the answer"
3134
+ }
3135
+ )
3136
+ ] }),
3137
+ /* @__PURE__ */ jsxs4("div", { style: { marginBottom: "16px" }, children: [
3138
+ /* @__PURE__ */ jsx4("label", { style: { display: "block", fontSize: "13px", fontWeight: "500", color: "#374151", marginBottom: "6px" }, children: "Additional details (optional)" }),
3139
+ /* @__PURE__ */ jsx4(
3140
+ "textarea",
3141
+ {
3142
+ value: skipComment,
3143
+ onChange: (e) => setSkipComment(e.target.value.slice(0, 200)),
3144
+ placeholder: "Tell us more about the issue...",
3145
+ disabled: isSkipping,
3146
+ style: {
3147
+ width: "100%",
3148
+ minHeight: "80px",
3149
+ padding: "10px 12px",
3150
+ borderRadius: "8px",
3151
+ border: "1px solid #e5e7eb",
3152
+ fontSize: "14px",
3153
+ resize: "vertical",
3154
+ fontFamily: "inherit",
3155
+ boxSizing: "border-box"
3156
+ },
3157
+ "data-testid": "input-skip-comment"
3158
+ }
3159
+ ),
3160
+ /* @__PURE__ */ jsxs4("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3161
+ skipComment.length,
3162
+ "/200"
3163
+ ] })
3164
+ ] }),
3165
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", gap: "10px" }, children: [
3166
+ /* @__PURE__ */ jsx4(
3167
+ "button",
3168
+ {
3169
+ onClick: () => {
3170
+ setShowSkipModal(false);
3171
+ setSkipComment("");
3172
+ setSelectedSkipReason(null);
3173
+ },
3174
+ style: {
3175
+ flex: 1,
3176
+ padding: "10px 16px",
3177
+ borderRadius: "8px",
3178
+ border: "1px solid #e5e7eb",
3179
+ backgroundColor: "#ffffff",
3180
+ cursor: "pointer",
3181
+ fontSize: "14px",
3182
+ fontWeight: "500",
3183
+ color: "#6b7280"
3184
+ },
3185
+ "data-testid": "button-skip-cancel",
3186
+ children: "Cancel"
3187
+ }
3188
+ ),
3189
+ /* @__PURE__ */ jsx4(
3190
+ "button",
3191
+ {
3192
+ onClick: () => selectedSkipReason && handleSkipQuestion(selectedSkipReason, skipComment),
3193
+ disabled: isSkipping || !selectedSkipReason,
3194
+ style: {
3195
+ flex: 1,
3196
+ padding: "10px 16px",
3197
+ borderRadius: "8px",
3198
+ border: "none",
3199
+ backgroundColor: selectedSkipReason ? "#8b5cf6" : "#d1d5db",
3200
+ cursor: isSkipping || !selectedSkipReason ? "not-allowed" : "pointer",
3201
+ fontSize: "14px",
3202
+ fontWeight: "500",
3203
+ color: "#ffffff",
3204
+ opacity: isSkipping ? 0.6 : 1
3205
+ },
3206
+ "data-testid": "button-skip-submit",
3207
+ children: isSkipping ? "Skipping..." : "Skip Question"
3208
+ }
3209
+ )
3210
+ ] })
3211
+ ] }) }),
3212
+ showReportModal && /* @__PURE__ */ jsx4("div", { style: {
3213
+ position: "fixed",
3214
+ top: 0,
3215
+ left: 0,
3216
+ right: 0,
3217
+ bottom: 0,
3218
+ backgroundColor: "rgba(0, 0, 0, 0.5)",
3219
+ display: "flex",
3220
+ alignItems: "center",
3221
+ justifyContent: "center",
3222
+ zIndex: 1e3
3223
+ }, children: /* @__PURE__ */ jsxs4("div", { style: {
3224
+ backgroundColor: "#ffffff",
3225
+ borderRadius: "12px",
3226
+ padding: "24px",
3227
+ maxWidth: "400px",
3228
+ width: "90%",
3229
+ boxShadow: "0 20px 40px rgba(0, 0, 0, 0.2)"
3230
+ }, children: [
3231
+ /* @__PURE__ */ jsx4("h3", { style: { margin: "0 0 8px 0", fontSize: "18px", fontWeight: "600", color: "#1f2937" }, children: "Report Question" }),
3232
+ /* @__PURE__ */ jsx4("p", { style: { margin: "0 0 16px 0", fontSize: "14px", color: "#6b7280" }, children: "What's wrong with this question?" }),
3233
+ /* @__PURE__ */ jsxs4("div", { style: { marginBottom: "16px" }, children: [
3234
+ /* @__PURE__ */ jsx4(
3235
+ "textarea",
3236
+ {
3237
+ value: reportComment,
3238
+ onChange: (e) => setReportComment(e.target.value.slice(0, 300)),
3239
+ placeholder: "Describe the issue with this question...",
3240
+ disabled: isReporting,
3241
+ style: {
3242
+ width: "100%",
3243
+ minHeight: "120px",
3244
+ padding: "10px 12px",
3245
+ borderRadius: "8px",
3246
+ border: "1px solid #e5e7eb",
3247
+ fontSize: "14px",
3248
+ resize: "vertical",
3249
+ fontFamily: "inherit",
3250
+ boxSizing: "border-box"
3251
+ },
3252
+ "data-testid": "input-report-comment"
3253
+ }
3254
+ ),
3255
+ /* @__PURE__ */ jsxs4("div", { style: { fontSize: "12px", color: "#9ca3af", marginTop: "4px", textAlign: "right" }, children: [
3256
+ reportComment.length,
3257
+ "/300"
3258
+ ] })
3259
+ ] }),
3260
+ /* @__PURE__ */ jsxs4("div", { style: { display: "flex", gap: "10px" }, children: [
3261
+ /* @__PURE__ */ jsx4(
3262
+ "button",
3263
+ {
3264
+ onClick: () => {
3265
+ setShowReportModal(false);
3266
+ setReportComment("");
3267
+ },
3268
+ style: {
3269
+ flex: 1,
3270
+ padding: "10px 16px",
3271
+ borderRadius: "8px",
3272
+ border: "1px solid #e5e7eb",
3273
+ backgroundColor: "#ffffff",
3274
+ cursor: "pointer",
3275
+ fontSize: "14px",
3276
+ fontWeight: "500",
3277
+ color: "#6b7280"
3278
+ },
3279
+ "data-testid": "button-report-cancel",
3280
+ children: "Cancel"
3281
+ }
3282
+ ),
3283
+ /* @__PURE__ */ jsx4(
3284
+ "button",
3285
+ {
3286
+ onClick: () => handleReportQuestion(reportComment),
3287
+ disabled: isReporting || !reportComment.trim(),
3288
+ style: {
3289
+ flex: 1,
3290
+ padding: "10px 16px",
3291
+ borderRadius: "8px",
3292
+ border: "none",
3293
+ backgroundColor: reportComment.trim() ? "#ef4444" : "#d1d5db",
3294
+ cursor: isReporting || !reportComment.trim() ? "not-allowed" : "pointer",
3295
+ fontSize: "14px",
3296
+ fontWeight: "500",
3297
+ color: "#ffffff",
3298
+ opacity: isReporting ? 0.6 : 1
3299
+ },
3300
+ "data-testid": "button-report-submit",
3301
+ children: isReporting ? "Reporting..." : "Report"
3302
+ }
3303
+ )
3304
+ ] })
3305
+ ] }) }),
3306
+ /* @__PURE__ */ jsxs4("div", { style: defaultStyles.buttonsColumn, children: [
3307
+ showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ jsx4(
2825
3308
  "button",
2826
3309
  {
2827
- onClick: () => {
2828
- setShowReportModal(false);
2829
- setReportComment("");
2830
- },
2831
3310
  style: {
2832
- flex: 1,
2833
- padding: "10px 16px",
2834
- borderRadius: "8px",
2835
- border: "1px solid #e5e7eb",
2836
- backgroundColor: "#ffffff",
2837
- cursor: "pointer",
2838
- fontSize: "14px",
2839
- fontWeight: "500",
2840
- color: "#6b7280"
3311
+ ...defaultStyles.buttonAddMore,
3312
+ ...isGeneratingExtra ? defaultStyles.buttonAddMoreDisabled : {}
2841
3313
  },
2842
- "data-testid": "button-report-cancel",
2843
- children: "Cancel"
3314
+ onClick: handleAddMoreQuestions,
3315
+ disabled: isGeneratingExtra,
3316
+ "data-testid": "button-add-more-questions",
3317
+ children: isGeneratingExtra ? /* @__PURE__ */ jsxs4(Fragment2, { children: [
3318
+ /* @__PURE__ */ jsx4(Spinner, { size: 16, color: "#9ca3af" }),
3319
+ "Generating Questions..."
3320
+ ] }) : /* @__PURE__ */ jsxs4(Fragment2, { children: [
3321
+ "+ Add ",
3322
+ questionsToAdd,
3323
+ " More Question",
3324
+ questionsToAdd !== 1 ? "s" : ""
3325
+ ] })
2844
3326
  }
2845
3327
  ),
2846
- /* @__PURE__ */ jsx3(
2847
- "button",
2848
- {
2849
- onClick: () => handleReportQuestion(reportComment),
2850
- disabled: isReporting || !reportComment.trim(),
2851
- style: {
2852
- flex: 1,
2853
- padding: "10px 16px",
2854
- borderRadius: "8px",
2855
- border: "none",
2856
- backgroundColor: reportComment.trim() ? "#ef4444" : "#d1d5db",
2857
- cursor: isReporting || !reportComment.trim() ? "not-allowed" : "pointer",
2858
- fontSize: "14px",
2859
- fontWeight: "500",
2860
- color: "#ffffff",
2861
- opacity: isReporting ? 0.6 : 1
2862
- },
2863
- "data-testid": "button-report-submit",
2864
- children: isReporting ? "Reporting..." : "Report"
2865
- }
2866
- )
3328
+ /* @__PURE__ */ jsx4("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
3329
+ // After viewing feedback
3330
+ isLastQuestion ? /* @__PURE__ */ jsx4(
3331
+ "button",
3332
+ {
3333
+ style: {
3334
+ ...defaultStyles.button,
3335
+ ...isSubmitting || isGeneratingExtra ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
3336
+ },
3337
+ onClick: handleSubmit,
3338
+ disabled: isSubmitting || isGeneratingExtra,
3339
+ "data-testid": "button-submit-quiz",
3340
+ children: isSubmitting ? /* @__PURE__ */ jsx4(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
3341
+ }
3342
+ ) : /* @__PURE__ */ jsx4(
3343
+ "button",
3344
+ {
3345
+ style: {
3346
+ ...defaultStyles.button,
3347
+ ...defaultStyles.buttonPrimary
3348
+ },
3349
+ onClick: handleContinue,
3350
+ "data-testid": "button-continue",
3351
+ children: "Continue"
3352
+ }
3353
+ )
3354
+ ) : (
3355
+ // Before checking answer
3356
+ /* @__PURE__ */ jsx4(
3357
+ "button",
3358
+ {
3359
+ style: {
3360
+ ...defaultStyles.button,
3361
+ ...isNavigating || selectedAnswer === void 0 ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
3362
+ },
3363
+ onClick: handleCheckAnswer,
3364
+ disabled: isNavigating || selectedAnswer === void 0,
3365
+ "data-testid": "button-check-answer",
3366
+ children: isNavigating ? /* @__PURE__ */ jsx4(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
3367
+ }
3368
+ )
3369
+ ) })
2867
3370
  ] })
2868
- ] }) }),
2869
- /* @__PURE__ */ jsxs3("div", { style: defaultStyles.buttonsColumn, children: [
2870
- showFeedback && isLastQuestion && canAddMore && /* @__PURE__ */ jsx3(
2871
- "button",
2872
- {
2873
- style: {
2874
- ...defaultStyles.buttonAddMore,
2875
- ...isGeneratingExtra ? defaultStyles.buttonAddMoreDisabled : {}
2876
- },
2877
- onClick: handleAddMoreQuestions,
2878
- disabled: isGeneratingExtra,
2879
- "data-testid": "button-add-more-questions",
2880
- children: isGeneratingExtra ? /* @__PURE__ */ jsxs3(Fragment2, { children: [
2881
- /* @__PURE__ */ jsx3(Spinner, { size: 16, color: "#9ca3af" }),
2882
- "Generating Questions..."
2883
- ] }) : /* @__PURE__ */ jsxs3(Fragment2, { children: [
2884
- "+ Add ",
2885
- questionsToAdd,
2886
- " More Question",
2887
- questionsToAdd !== 1 ? "s" : ""
2888
- ] })
2889
- }
2890
- ),
2891
- /* @__PURE__ */ jsx3("div", { style: { ...defaultStyles.buttons, justifyContent: "flex-end" }, children: showFeedback ? (
2892
- // After viewing feedback
2893
- isLastQuestion ? /* @__PURE__ */ jsx3(
2894
- "button",
2895
- {
2896
- style: {
2897
- ...defaultStyles.button,
2898
- ...isSubmitting || isGeneratingExtra ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
2899
- },
2900
- onClick: handleSubmit,
2901
- disabled: isSubmitting || isGeneratingExtra,
2902
- "data-testid": "button-submit-quiz",
2903
- children: isSubmitting ? /* @__PURE__ */ jsx3(Spinner, { size: 16, color: "#9ca3af" }) : "Submit Quiz"
2904
- }
2905
- ) : /* @__PURE__ */ jsx3(
2906
- "button",
2907
- {
2908
- style: {
2909
- ...defaultStyles.button,
2910
- ...defaultStyles.buttonPrimary
2911
- },
2912
- onClick: handleContinue,
2913
- "data-testid": "button-continue",
2914
- children: "Continue"
2915
- }
2916
- )
2917
- ) : (
2918
- // Before checking answer
2919
- /* @__PURE__ */ jsx3(
2920
- "button",
2921
- {
2922
- style: {
2923
- ...defaultStyles.button,
2924
- ...isNavigating || selectedAnswer === void 0 ? defaultStyles.buttonDisabled : defaultStyles.buttonPrimary
2925
- },
2926
- onClick: handleCheckAnswer,
2927
- disabled: isNavigating || selectedAnswer === void 0,
2928
- "data-testid": "button-check-answer",
2929
- children: isNavigating ? /* @__PURE__ */ jsx3(Spinner, { size: 16, color: "#9ca3af" }) : "Check Answer"
2930
- }
2931
- )
2932
- ) })
2933
- ] })
2934
- ] }),
2935
- /* @__PURE__ */ jsx3("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ jsx3(
2936
- QuestionChatPanel,
2937
- {
2938
- apiClient: apiClient.current,
2939
- question: {
2940
- id: currentQuestion.id,
2941
- question: currentQuestion.question,
2942
- type: currentQuestion.type,
2943
- options: currentQuestion.options,
2944
- correctAnswer: currentQuestion.correctAnswer,
2945
- explanation: currentQuestion.explanation
2946
- },
2947
- quizId: quiz.id,
2948
- childId,
2949
- parentId,
2950
- lessonId,
2951
- courseId,
2952
- answerResult: showFeedback && currentAnswerDetail ? {
2953
- wasIncorrect: currentQuestion.type !== "assessment" && !currentAnswerDetail.isCorrect,
2954
- selectedAnswer: typeof selectedAnswer === "string" ? selectedAnswer : Array.isArray(selectedAnswer) ? selectedAnswer.join(", ") : void 0,
2955
- correctAnswer: typeof currentQuestion.correctAnswer === "string" ? currentQuestion.correctAnswer : Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer.join(", ") : void 0,
2956
- explanation: currentQuestion.explanation
2957
- } : void 0
2958
- }
2959
- ) })
2960
- ] }) });
3371
+ ] }),
3372
+ /* @__PURE__ */ jsx4("div", { style: defaultStyles.chatPanel, children: apiClient.current && /* @__PURE__ */ jsx4(
3373
+ QuestionChatPanel,
3374
+ {
3375
+ apiClient: apiClient.current,
3376
+ question: {
3377
+ id: currentQuestion.id,
3378
+ question: currentQuestion.question,
3379
+ type: currentQuestion.type,
3380
+ options: currentQuestion.options,
3381
+ correctAnswer: currentQuestion.correctAnswer,
3382
+ explanation: currentQuestion.explanation
3383
+ },
3384
+ quizId: quiz.id,
3385
+ childId,
3386
+ parentId,
3387
+ lessonId,
3388
+ courseId,
3389
+ answerResult: showFeedback && currentAnswerDetail ? {
3390
+ wasIncorrect: currentQuestion.type !== "assessment" && !currentAnswerDetail.isCorrect,
3391
+ selectedAnswer: typeof selectedAnswer === "string" ? selectedAnswer : Array.isArray(selectedAnswer) ? selectedAnswer.join(", ") : void 0,
3392
+ correctAnswer: typeof currentQuestion.correctAnswer === "string" ? currentQuestion.correctAnswer : Array.isArray(currentQuestion.correctAnswer) ? currentQuestion.correctAnswer.join(", ") : void 0,
3393
+ explanation: currentQuestion.explanation
3394
+ } : void 0
3395
+ }
3396
+ ) })
3397
+ ] })
3398
+ ] });
2961
3399
  }
2962
3400
 
2963
3401
  // 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";
3402
+ import { useState as useState4, useEffect as useEffect4, useRef as useRef4 } from "react";
3403
+ import { jsx as jsx5, jsxs as jsxs5 } from "react/jsx-runtime";
2966
3404
  var defaultStyles2 = {
2967
3405
  container: {
2968
3406
  fontFamily: "system-ui, -apple-system, sans-serif",
@@ -3105,7 +3543,12 @@ var defaultStyles2 = {
3105
3543
  marginTop: "12px",
3106
3544
  display: "flex",
3107
3545
  flexDirection: "column",
3108
- gap: "8px"
3546
+ gap: "8px",
3547
+ backgroundColor: "#ffffff",
3548
+ padding: "16px",
3549
+ borderRadius: "8px",
3550
+ border: "1px solid #e5e7eb",
3551
+ boxShadow: "0 1px 3px rgba(0, 0, 0, 0.05)"
3109
3552
  },
3110
3553
  chatMessage: {
3111
3554
  padding: "8px 12px",
@@ -3215,23 +3658,39 @@ function AttemptViewer({
3215
3658
  }) {
3216
3659
  const [attempt, setAttempt] = useState4(null);
3217
3660
  const [loading, setLoading] = useState4(true);
3218
- const [error, setError] = useState4(null);
3661
+ const [errorCode, setErrorCode] = useState4(null);
3219
3662
  const [chatHistories, setChatHistories] = useState4({});
3220
3663
  const [expandedChats, setExpandedChats] = useState4(/* @__PURE__ */ new Set());
3664
+ const [fetchedAttemptId, setFetchedAttemptId] = useState4(null);
3665
+ const onErrorRef = useRef4(onError);
3666
+ onErrorRef.current = onError;
3221
3667
  useEffect4(() => {
3668
+ if (fetchedAttemptId === attemptId) return;
3222
3669
  const apiClient = new QuizApiClient({
3223
3670
  baseUrl: apiBaseUrl,
3224
3671
  authToken
3225
3672
  });
3226
3673
  async function fetchAttempt() {
3227
3674
  setLoading(true);
3228
- setError(null);
3675
+ setErrorCode(null);
3229
3676
  try {
3230
3677
  const response = await fetch(`${apiBaseUrl}/api/external/quiz-attempts/${attemptId}`, {
3231
3678
  headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
3232
3679
  });
3233
3680
  if (!response.ok) {
3234
- throw new Error(`Failed to fetch attempt: ${response.statusText}`);
3681
+ const code = getErrorFromHttpStatus(response.status, "attempt");
3682
+ setErrorCode(code);
3683
+ apiClient.logError({
3684
+ errorCode: code,
3685
+ context: "attempt",
3686
+ resourceId: attemptId,
3687
+ component: "AttemptViewer",
3688
+ message: `HTTP ${response.status}: ${response.statusText}`
3689
+ });
3690
+ onErrorRef.current?.(new Error(`Failed to fetch attempt: ${response.statusText}`));
3691
+ setLoading(false);
3692
+ setFetchedAttemptId(attemptId);
3693
+ return;
3235
3694
  }
3236
3695
  const data = await response.json();
3237
3696
  setAttempt(data);
@@ -3245,14 +3704,23 @@ function AttemptViewer({
3245
3704
  }
3246
3705
  } catch (err) {
3247
3706
  const errorMessage = err instanceof Error ? err.message : "Failed to load attempt";
3248
- setError(errorMessage);
3249
- onError?.(err instanceof Error ? err : new Error(errorMessage));
3707
+ const code = getErrorFromMessage(errorMessage, "attempt");
3708
+ setErrorCode(code);
3709
+ apiClient.logError({
3710
+ errorCode: code,
3711
+ context: "attempt",
3712
+ resourceId: attemptId,
3713
+ component: "AttemptViewer",
3714
+ message: errorMessage
3715
+ });
3716
+ onErrorRef.current?.(err instanceof Error ? err : new Error(errorMessage));
3250
3717
  } finally {
3251
3718
  setLoading(false);
3719
+ setFetchedAttemptId(attemptId);
3252
3720
  }
3253
3721
  }
3254
3722
  fetchAttempt();
3255
- }, [attemptId, apiBaseUrl, authToken, onError, showConversation]);
3723
+ }, [attemptId, apiBaseUrl, authToken, showConversation, fetchedAttemptId]);
3256
3724
  const toggleChatExpanded = (questionId) => {
3257
3725
  setExpandedChats((prev) => {
3258
3726
  const newSet = new Set(prev);
@@ -3266,53 +3734,49 @@ function AttemptViewer({
3266
3734
  };
3267
3735
  const handleRetry = () => {
3268
3736
  setLoading(true);
3269
- setError(null);
3737
+ setErrorCode(null);
3270
3738
  window.location.reload();
3271
3739
  };
3272
3740
  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..." })
3741
+ return /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.container, className, children: [
3742
+ /* @__PURE__ */ jsx5("style", { children: spinnerKeyframes }),
3743
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.loading, children: [
3744
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.spinner }),
3745
+ /* @__PURE__ */ jsx5("p", { style: { marginTop: "16px", color: "#6b7280" }, children: "Loading attempt..." })
3278
3746
  ] })
3279
3747
  ] });
3280
3748
  }
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
- ] }) });
3749
+ if (errorCode || !attempt) {
3750
+ return /* @__PURE__ */ jsx5("div", { style: defaultStyles2.container, className, children: /* @__PURE__ */ jsx5(MaintenanceScreen, { errorCode: errorCode || "ATTEMPT_NOT_FOUND" }) });
3287
3751
  }
3288
3752
  const scorePercentage = attempt.score ?? 0;
3289
3753
  const correctCount = attempt.correctAnswers ?? 0;
3290
3754
  const totalQuestions = attempt.totalQuestions;
3291
3755
  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: [
3756
+ return /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.container, className, children: [
3757
+ /* @__PURE__ */ jsx5("style", { children: spinnerKeyframes }),
3758
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.header, children: /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryGrid, children: [
3759
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryCard, children: [
3760
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryValue, children: [
3297
3761
  scorePercentage,
3298
3762
  "%"
3299
3763
  ] }),
3300
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.summaryLabel, children: "Score" })
3764
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryLabel, children: "Score" })
3301
3765
  ] }),
3302
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryCard, children: [
3303
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.summaryValue, children: [
3766
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryCard, children: [
3767
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryValue, children: [
3304
3768
  correctCount,
3305
3769
  "/",
3306
3770
  totalQuestions
3307
3771
  ] }),
3308
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
3772
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryLabel, children: "Correct" })
3309
3773
  ] }),
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" })
3774
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.summaryCard, children: [
3775
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryValue, children: formatTime(timeSpent) }),
3776
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.summaryLabel, children: "Time" })
3313
3777
  ] })
3314
3778
  ] }) }),
3315
- /* @__PURE__ */ jsx4("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ jsxs4(
3779
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.questionsList, children: attempt.answers.map((answer, index) => /* @__PURE__ */ jsxs5(
3316
3780
  "div",
3317
3781
  {
3318
3782
  style: {
@@ -3320,12 +3784,12 @@ function AttemptViewer({
3320
3784
  ...answer.isCorrect ? defaultStyles2.questionCardCorrect : defaultStyles2.questionCardIncorrect
3321
3785
  },
3322
3786
  children: [
3323
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.questionHeader, children: [
3324
- /* @__PURE__ */ jsxs4("span", { style: defaultStyles2.questionNumber, children: [
3787
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.questionHeader, children: [
3788
+ /* @__PURE__ */ jsxs5("span", { style: defaultStyles2.questionNumber, children: [
3325
3789
  "Question ",
3326
3790
  index + 1
3327
3791
  ] }),
3328
- /* @__PURE__ */ jsx4(
3792
+ /* @__PURE__ */ jsx5(
3329
3793
  "span",
3330
3794
  {
3331
3795
  style: {
@@ -3336,35 +3800,35 @@ function AttemptViewer({
3336
3800
  }
3337
3801
  )
3338
3802
  ] }),
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) })
3803
+ /* @__PURE__ */ jsx5("div", { style: defaultStyles2.questionText, children: answer.questionText }),
3804
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.answerSection, children: [
3805
+ /* @__PURE__ */ jsx5("span", { style: defaultStyles2.answerLabel, children: "Your answer:" }),
3806
+ /* @__PURE__ */ jsx5("span", { style: defaultStyles2.studentAnswer, children: formatAnswer(answer.selectedAnswer, answer.questionType, answer.items, answer.leftItems) })
3343
3807
  ] }),
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) })
3808
+ !answer.isCorrect && answer.correctAnswer && /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.answerSection, children: [
3809
+ /* @__PURE__ */ jsx5("span", { style: defaultStyles2.answerLabel, children: "Correct answer:" }),
3810
+ /* @__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
3811
  ] }),
3348
- /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.points, children: [
3812
+ /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.points, children: [
3349
3813
  answer.pointsEarned,
3350
3814
  " / ",
3351
3815
  answer.points,
3352
3816
  " points"
3353
3817
  ] }),
3354
- showExplanations && answer.explanation && /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.explanation, children: [
3355
- /* @__PURE__ */ jsx4("strong", { children: "Explanation:" }),
3818
+ showExplanations && answer.explanation && /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.explanation, children: [
3819
+ /* @__PURE__ */ jsx5("strong", { children: "Explanation:" }),
3356
3820
  " ",
3357
3821
  answer.explanation
3358
3822
  ] }),
3359
- showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ jsxs4("div", { style: defaultStyles2.chatHistorySection, children: [
3360
- /* @__PURE__ */ jsxs4(
3823
+ showConversation && chatHistories[answer.questionId] && chatHistories[answer.questionId].messages.length > 0 && /* @__PURE__ */ jsxs5("div", { style: defaultStyles2.chatHistorySection, children: [
3824
+ /* @__PURE__ */ jsxs5(
3361
3825
  "button",
3362
3826
  {
3363
3827
  style: defaultStyles2.chatToggleButton,
3364
3828
  onClick: () => toggleChatExpanded(answer.questionId),
3365
3829
  "data-testid": `button-toggle-chat-${answer.questionId}`,
3366
3830
  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" }) }),
3831
+ /* @__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
3832
  expandedChats.has(answer.questionId) ? "Hide" : "View",
3369
3833
  " Chat History (",
3370
3834
  chatHistories[answer.questionId].messages.length,
@@ -3372,7 +3836,7 @@ function AttemptViewer({
3372
3836
  ]
3373
3837
  }
3374
3838
  ),
3375
- expandedChats.has(answer.questionId) && /* @__PURE__ */ jsx4("div", { style: defaultStyles2.chatMessages, children: chatHistories[answer.questionId].messages.map((msg, msgIndex) => /* @__PURE__ */ jsx4(
3839
+ expandedChats.has(answer.questionId) && /* @__PURE__ */ jsx5("div", { style: defaultStyles2.chatMessages, children: chatHistories[answer.questionId].messages.map((msg, msgIndex) => /* @__PURE__ */ jsx5(
3376
3840
  "div",
3377
3841
  {
3378
3842
  style: {
@@ -3390,14 +3854,620 @@ function AttemptViewer({
3390
3854
  )) })
3391
3855
  ] });
3392
3856
  }
3857
+
3858
+ // src/ErrorTypesPanel.tsx
3859
+ import { jsx as jsx6, jsxs as jsxs6 } from "react/jsx-runtime";
3860
+ var panelStyles2 = {
3861
+ container: {
3862
+ fontFamily: "system-ui, -apple-system, sans-serif",
3863
+ padding: "24px",
3864
+ backgroundColor: "#ffffff",
3865
+ borderRadius: "12px",
3866
+ maxWidth: "800px"
3867
+ },
3868
+ header: {
3869
+ marginBottom: "24px"
3870
+ },
3871
+ title: {
3872
+ fontSize: "20px",
3873
+ fontWeight: "600",
3874
+ color: "#111827",
3875
+ marginBottom: "8px"
3876
+ },
3877
+ subtitle: {
3878
+ fontSize: "14px",
3879
+ color: "#6b7280"
3880
+ },
3881
+ section: {
3882
+ marginBottom: "24px"
3883
+ },
3884
+ sectionTitle: {
3885
+ fontSize: "14px",
3886
+ fontWeight: "600",
3887
+ color: "#374151",
3888
+ marginBottom: "12px",
3889
+ textTransform: "uppercase",
3890
+ letterSpacing: "0.05em"
3891
+ },
3892
+ errorList: {
3893
+ display: "flex",
3894
+ flexDirection: "column",
3895
+ gap: "12px"
3896
+ },
3897
+ errorCard: {
3898
+ padding: "16px",
3899
+ backgroundColor: "#f9fafb",
3900
+ borderRadius: "8px",
3901
+ border: "1px solid #e5e7eb"
3902
+ },
3903
+ errorCardBlocking: {
3904
+ borderLeft: "4px solid #ef4444"
3905
+ },
3906
+ errorCardNonBlocking: {
3907
+ borderLeft: "4px solid #f59e0b"
3908
+ },
3909
+ errorHeader: {
3910
+ display: "flex",
3911
+ justifyContent: "space-between",
3912
+ alignItems: "flex-start",
3913
+ marginBottom: "8px"
3914
+ },
3915
+ errorCode: {
3916
+ fontSize: "13px",
3917
+ fontWeight: "600",
3918
+ color: "#1f2937",
3919
+ fontFamily: "monospace",
3920
+ backgroundColor: "#e5e7eb",
3921
+ padding: "2px 8px",
3922
+ borderRadius: "4px"
3923
+ },
3924
+ errorBadge: {
3925
+ fontSize: "11px",
3926
+ fontWeight: "500",
3927
+ padding: "2px 8px",
3928
+ borderRadius: "12px"
3929
+ },
3930
+ blockingBadge: {
3931
+ backgroundColor: "#fee2e2",
3932
+ color: "#dc2626"
3933
+ },
3934
+ nonBlockingBadge: {
3935
+ backgroundColor: "#fef3c7",
3936
+ color: "#d97706"
3937
+ },
3938
+ userMessage: {
3939
+ fontSize: "15px",
3940
+ fontWeight: "500",
3941
+ color: "#111827",
3942
+ marginBottom: "4px"
3943
+ },
3944
+ subMessage: {
3945
+ fontSize: "13px",
3946
+ color: "#6b7280",
3947
+ marginBottom: "8px"
3948
+ },
3949
+ causeLabel: {
3950
+ fontSize: "11px",
3951
+ fontWeight: "600",
3952
+ color: "#9ca3af",
3953
+ textTransform: "uppercase",
3954
+ marginBottom: "4px"
3955
+ },
3956
+ causeText: {
3957
+ fontSize: "13px",
3958
+ color: "#4b5563",
3959
+ fontStyle: "italic"
3960
+ }
3961
+ };
3962
+ function ErrorTypesPanel({ className }) {
3963
+ const blockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => e.isBlocking);
3964
+ const nonBlockingErrors = Object.values(ERROR_DEFINITIONS).filter((e) => !e.isBlocking);
3965
+ const renderErrorCard = (error) => /* @__PURE__ */ jsxs6(
3966
+ "div",
3967
+ {
3968
+ style: {
3969
+ ...panelStyles2.errorCard,
3970
+ ...error.isBlocking ? panelStyles2.errorCardBlocking : panelStyles2.errorCardNonBlocking
3971
+ },
3972
+ "data-testid": `error-card-${error.code}`,
3973
+ children: [
3974
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.errorHeader, children: [
3975
+ /* @__PURE__ */ jsx6("code", { style: panelStyles2.errorCode, children: error.code }),
3976
+ /* @__PURE__ */ jsx6(
3977
+ "span",
3978
+ {
3979
+ style: {
3980
+ ...panelStyles2.errorBadge,
3981
+ ...error.isBlocking ? panelStyles2.blockingBadge : panelStyles2.nonBlockingBadge
3982
+ },
3983
+ children: error.isBlocking ? "Blocking" : "Non-Blocking"
3984
+ }
3985
+ )
3986
+ ] }),
3987
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.userMessage, children: error.userMessage }),
3988
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.subMessage, children: error.subMessage }),
3989
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.causeLabel, children: "Why this happens:" }),
3990
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.causeText, children: error.cause })
3991
+ ]
3992
+ },
3993
+ error.code
3994
+ );
3995
+ return /* @__PURE__ */ jsxs6("div", { style: panelStyles2.container, className, "data-testid": "error-types-panel", children: [
3996
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.header, children: [
3997
+ /* @__PURE__ */ jsx6("h2", { style: panelStyles2.title, children: "Error Types Reference" }),
3998
+ /* @__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." })
3999
+ ] }),
4000
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.section, children: [
4001
+ /* @__PURE__ */ jsxs6("h3", { style: panelStyles2.sectionTitle, children: [
4002
+ "Blocking Errors (",
4003
+ blockingErrors.length,
4004
+ ")"
4005
+ ] }),
4006
+ /* @__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." }),
4007
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.errorList, children: blockingErrors.map(renderErrorCard) })
4008
+ ] }),
4009
+ /* @__PURE__ */ jsxs6("div", { style: panelStyles2.section, children: [
4010
+ /* @__PURE__ */ jsxs6("h3", { style: panelStyles2.sectionTitle, children: [
4011
+ "Non-Blocking Errors (",
4012
+ nonBlockingErrors.length,
4013
+ ")"
4014
+ ] }),
4015
+ /* @__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." }),
4016
+ /* @__PURE__ */ jsx6("div", { style: panelStyles2.errorList, children: nonBlockingErrors.map(renderErrorCard) })
4017
+ ] })
4018
+ ] });
4019
+ }
4020
+
4021
+ // src/ErrorLogsPanel.tsx
4022
+ import { useState as useState5, useEffect as useEffect5 } from "react";
4023
+ import { jsx as jsx7, jsxs as jsxs7 } from "react/jsx-runtime";
4024
+ var panelStyles3 = {
4025
+ container: {
4026
+ fontFamily: "system-ui, -apple-system, sans-serif",
4027
+ padding: "24px",
4028
+ backgroundColor: "#ffffff",
4029
+ borderRadius: "12px",
4030
+ maxWidth: "1000px"
4031
+ },
4032
+ header: {
4033
+ marginBottom: "24px"
4034
+ },
4035
+ title: {
4036
+ fontSize: "20px",
4037
+ fontWeight: "600",
4038
+ color: "#111827",
4039
+ marginBottom: "8px"
4040
+ },
4041
+ subtitle: {
4042
+ fontSize: "14px",
4043
+ color: "#6b7280"
4044
+ },
4045
+ tabs: {
4046
+ display: "flex",
4047
+ gap: "8px",
4048
+ marginBottom: "16px",
4049
+ borderBottom: "1px solid #e5e7eb",
4050
+ paddingBottom: "12px"
4051
+ },
4052
+ tab: {
4053
+ padding: "8px 16px",
4054
+ fontSize: "14px",
4055
+ fontWeight: "500",
4056
+ borderRadius: "6px",
4057
+ cursor: "pointer",
4058
+ border: "none",
4059
+ backgroundColor: "transparent",
4060
+ color: "#6b7280"
4061
+ },
4062
+ tabActive: {
4063
+ backgroundColor: "#6721b0",
4064
+ color: "#ffffff"
4065
+ },
4066
+ stats: {
4067
+ display: "flex",
4068
+ gap: "16px",
4069
+ marginBottom: "20px"
4070
+ },
4071
+ statCard: {
4072
+ padding: "12px 16px",
4073
+ backgroundColor: "#f9fafb",
4074
+ borderRadius: "8px",
4075
+ flex: 1
4076
+ },
4077
+ statValue: {
4078
+ fontSize: "24px",
4079
+ fontWeight: "700",
4080
+ color: "#111827"
4081
+ },
4082
+ statLabel: {
4083
+ fontSize: "12px",
4084
+ color: "#6b7280",
4085
+ textTransform: "uppercase",
4086
+ letterSpacing: "0.05em"
4087
+ },
4088
+ errorList: {
4089
+ display: "flex",
4090
+ flexDirection: "column",
4091
+ gap: "12px"
4092
+ },
4093
+ errorCard: {
4094
+ padding: "16px",
4095
+ backgroundColor: "#f9fafb",
4096
+ borderRadius: "8px",
4097
+ border: "1px solid #e5e7eb"
4098
+ },
4099
+ errorCardRecent: {
4100
+ borderLeft: "4px solid #ef4444",
4101
+ backgroundColor: "#fef2f2"
4102
+ },
4103
+ errorCardOld: {
4104
+ borderLeft: "4px solid #9ca3af",
4105
+ opacity: 0.7
4106
+ },
4107
+ errorHeader: {
4108
+ display: "flex",
4109
+ justifyContent: "space-between",
4110
+ alignItems: "flex-start",
4111
+ marginBottom: "8px"
4112
+ },
4113
+ errorLeft: {
4114
+ flex: 1
4115
+ },
4116
+ errorCode: {
4117
+ fontSize: "13px",
4118
+ fontWeight: "600",
4119
+ color: "#1f2937",
4120
+ fontFamily: "monospace",
4121
+ backgroundColor: "#e5e7eb",
4122
+ padding: "2px 8px",
4123
+ borderRadius: "4px",
4124
+ display: "inline-block"
4125
+ },
4126
+ countBadge: {
4127
+ fontSize: "14px",
4128
+ fontWeight: "700",
4129
+ padding: "4px 12px",
4130
+ borderRadius: "16px",
4131
+ backgroundColor: "#ef4444",
4132
+ color: "#ffffff"
4133
+ },
4134
+ countBadgeLow: {
4135
+ backgroundColor: "#f59e0b"
4136
+ },
4137
+ contextBadge: {
4138
+ fontSize: "11px",
4139
+ fontWeight: "500",
4140
+ padding: "2px 8px",
4141
+ borderRadius: "4px",
4142
+ marginLeft: "8px",
4143
+ backgroundColor: "#dbeafe",
4144
+ color: "#1d4ed8"
4145
+ },
4146
+ userMessage: {
4147
+ fontSize: "15px",
4148
+ fontWeight: "500",
4149
+ color: "#111827",
4150
+ marginTop: "8px",
4151
+ marginBottom: "4px"
4152
+ },
4153
+ resourceId: {
4154
+ fontSize: "12px",
4155
+ color: "#6b7280",
4156
+ fontFamily: "monospace",
4157
+ marginTop: "4px"
4158
+ },
4159
+ resourceLink: {
4160
+ color: "#6721b0",
4161
+ textDecoration: "underline",
4162
+ cursor: "pointer"
4163
+ },
4164
+ timestamps: {
4165
+ display: "flex",
4166
+ gap: "16px",
4167
+ marginTop: "8px",
4168
+ fontSize: "12px",
4169
+ color: "#9ca3af"
4170
+ },
4171
+ dismissButton: {
4172
+ padding: "6px 12px",
4173
+ fontSize: "12px",
4174
+ fontWeight: "500",
4175
+ borderRadius: "4px",
4176
+ cursor: "pointer",
4177
+ border: "1px solid #e5e7eb",
4178
+ backgroundColor: "#ffffff",
4179
+ color: "#6b7280",
4180
+ marginTop: "8px"
4181
+ },
4182
+ dismissButtonHover: {
4183
+ backgroundColor: "#f3f4f6"
4184
+ },
4185
+ emptyState: {
4186
+ textAlign: "center",
4187
+ padding: "40px",
4188
+ color: "#9ca3af",
4189
+ fontSize: "14px"
4190
+ },
4191
+ loading: {
4192
+ textAlign: "center",
4193
+ padding: "40px",
4194
+ color: "#6b7280",
4195
+ fontSize: "14px"
4196
+ },
4197
+ pagination: {
4198
+ display: "flex",
4199
+ justifyContent: "center",
4200
+ gap: "8px",
4201
+ marginTop: "20px"
4202
+ },
4203
+ pageButton: {
4204
+ padding: "8px 12px",
4205
+ fontSize: "14px",
4206
+ borderRadius: "4px",
4207
+ cursor: "pointer",
4208
+ border: "1px solid #e5e7eb",
4209
+ backgroundColor: "#ffffff",
4210
+ color: "#374151"
4211
+ },
4212
+ pageButtonDisabled: {
4213
+ opacity: 0.5,
4214
+ cursor: "not-allowed"
4215
+ }
4216
+ };
4217
+ function formatRelativeTime(dateStr) {
4218
+ const date = new Date(dateStr);
4219
+ const now = /* @__PURE__ */ new Date();
4220
+ const diffMs = now.getTime() - date.getTime();
4221
+ const diffMins = Math.floor(diffMs / 6e4);
4222
+ const diffHours = Math.floor(diffMs / 36e5);
4223
+ const diffDays = Math.floor(diffMs / 864e5);
4224
+ if (diffMins < 1) return "Just now";
4225
+ if (diffMins < 60) return `${diffMins}m ago`;
4226
+ if (diffHours < 24) return `${diffHours}h ago`;
4227
+ if (diffDays < 7) return `${diffDays}d ago`;
4228
+ return date.toLocaleDateString();
4229
+ }
4230
+ function isRecent(dateStr) {
4231
+ const date = new Date(dateStr);
4232
+ const now = /* @__PURE__ */ new Date();
4233
+ const diffMs = now.getTime() - date.getTime();
4234
+ const diffHours = diffMs / 36e5;
4235
+ return diffHours < 24;
4236
+ }
4237
+ function ErrorLogsPanel({ apiBaseUrl, authToken, onResourceClick }) {
4238
+ const [logs, setLogs] = useState5([]);
4239
+ const [loading, setLoading] = useState5(true);
4240
+ const [showDismissed, setShowDismissed] = useState5(false);
4241
+ const [page, setPage] = useState5(1);
4242
+ const [totalPages, setTotalPages] = useState5(1);
4243
+ const [total, setTotal] = useState5(0);
4244
+ useEffect5(() => {
4245
+ fetchLogs();
4246
+ }, [showDismissed, page]);
4247
+ async function fetchLogs() {
4248
+ setLoading(true);
4249
+ try {
4250
+ const params = new URLSearchParams();
4251
+ params.set("dismissed", showDismissed ? "true" : "false");
4252
+ params.set("page", page.toString());
4253
+ params.set("limit", "20");
4254
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs?${params}`, {
4255
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4256
+ });
4257
+ if (response.ok) {
4258
+ const data = await response.json();
4259
+ setLogs(data.items || []);
4260
+ setTotalPages(data.totalPages || 1);
4261
+ setTotal(data.total || 0);
4262
+ }
4263
+ } catch (err) {
4264
+ console.error("Failed to fetch error logs:", err);
4265
+ } finally {
4266
+ setLoading(false);
4267
+ }
4268
+ }
4269
+ async function handleDismiss(id) {
4270
+ try {
4271
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/dismiss`, {
4272
+ method: "PATCH",
4273
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4274
+ });
4275
+ if (response.ok) {
4276
+ fetchLogs();
4277
+ }
4278
+ } catch (err) {
4279
+ console.error("Failed to dismiss error:", err);
4280
+ }
4281
+ }
4282
+ async function handleUndismiss(id) {
4283
+ try {
4284
+ const response = await fetch(`${apiBaseUrl}/api/admin/error-logs/${id}/undismiss`, {
4285
+ method: "PATCH",
4286
+ headers: authToken ? { Authorization: `Bearer ${authToken}` } : {}
4287
+ });
4288
+ if (response.ok) {
4289
+ fetchLogs();
4290
+ }
4291
+ } catch (err) {
4292
+ console.error("Failed to undismiss error:", err);
4293
+ }
4294
+ }
4295
+ function getErrorDefinition(code) {
4296
+ return ERROR_DEFINITIONS[code];
4297
+ }
4298
+ const totalErrors = logs.reduce((sum, log) => sum + log.count, 0);
4299
+ const recentCount = logs.filter((log) => isRecent(log.lastSeenAt)).length;
4300
+ return /* @__PURE__ */ jsxs7("div", { style: panelStyles3.container, "data-testid": "error-logs-panel", children: [
4301
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.header, children: [
4302
+ /* @__PURE__ */ jsx7("h2", { style: panelStyles3.title, children: "Error Tracking" }),
4303
+ /* @__PURE__ */ jsx7("p", { style: panelStyles3.subtitle, children: "Aggregated errors from QuizPlayer and AttemptViewer components, sorted by occurrence count" })
4304
+ ] }),
4305
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.tabs, children: [
4306
+ /* @__PURE__ */ jsx7(
4307
+ "button",
4308
+ {
4309
+ style: {
4310
+ ...panelStyles3.tab,
4311
+ ...showDismissed ? {} : panelStyles3.tabActive
4312
+ },
4313
+ onClick: () => {
4314
+ setShowDismissed(false);
4315
+ setPage(1);
4316
+ },
4317
+ "data-testid": "tab-active-errors",
4318
+ children: "Active Errors"
4319
+ }
4320
+ ),
4321
+ /* @__PURE__ */ jsx7(
4322
+ "button",
4323
+ {
4324
+ style: {
4325
+ ...panelStyles3.tab,
4326
+ ...showDismissed ? panelStyles3.tabActive : {}
4327
+ },
4328
+ onClick: () => {
4329
+ setShowDismissed(true);
4330
+ setPage(1);
4331
+ },
4332
+ "data-testid": "tab-dismissed-errors",
4333
+ children: "Dismissed"
4334
+ }
4335
+ )
4336
+ ] }),
4337
+ !showDismissed && /* @__PURE__ */ jsxs7("div", { style: panelStyles3.stats, children: [
4338
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.statCard, children: [
4339
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statValue, children: total }),
4340
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statLabel, children: "Unique Errors" })
4341
+ ] }),
4342
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.statCard, children: [
4343
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statValue, children: totalErrors }),
4344
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statLabel, children: "Total Occurrences" })
4345
+ ] }),
4346
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.statCard, children: [
4347
+ /* @__PURE__ */ jsx7("div", { style: { ...panelStyles3.statValue, color: recentCount > 0 ? "#ef4444" : "#22c55e" }, children: recentCount }),
4348
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.statLabel, children: "Last 24h" })
4349
+ ] })
4350
+ ] }),
4351
+ 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) => {
4352
+ const def = getErrorDefinition(log.errorCode);
4353
+ const recent = isRecent(log.lastSeenAt);
4354
+ return /* @__PURE__ */ jsxs7(
4355
+ "div",
4356
+ {
4357
+ style: {
4358
+ ...panelStyles3.errorCard,
4359
+ ...recent ? panelStyles3.errorCardRecent : panelStyles3.errorCardOld
4360
+ },
4361
+ "data-testid": `error-log-${log.id}`,
4362
+ children: [
4363
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.errorHeader, children: [
4364
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.errorLeft, children: [
4365
+ /* @__PURE__ */ jsx7("span", { style: panelStyles3.errorCode, children: log.errorCode }),
4366
+ /* @__PURE__ */ jsx7("span", { style: panelStyles3.contextBadge, children: log.context }),
4367
+ /* @__PURE__ */ jsx7("span", { style: { ...panelStyles3.contextBadge, backgroundColor: "#f3e8ff", color: "#7c3aed" }, children: log.component })
4368
+ ] }),
4369
+ /* @__PURE__ */ jsxs7(
4370
+ "span",
4371
+ {
4372
+ style: {
4373
+ ...panelStyles3.countBadge,
4374
+ ...log.count < 10 ? panelStyles3.countBadgeLow : {}
4375
+ },
4376
+ children: [
4377
+ log.count,
4378
+ "x"
4379
+ ]
4380
+ }
4381
+ )
4382
+ ] }),
4383
+ /* @__PURE__ */ jsx7("div", { style: panelStyles3.userMessage, children: def?.userMessage || log.lastMessage || "Unknown error" }),
4384
+ log.resourceId && /* @__PURE__ */ jsxs7("div", { style: panelStyles3.resourceId, children: [
4385
+ "Resource:",
4386
+ " ",
4387
+ onResourceClick ? /* @__PURE__ */ jsx7(
4388
+ "span",
4389
+ {
4390
+ style: panelStyles3.resourceLink,
4391
+ onClick: () => onResourceClick(log.resourceId, log.context),
4392
+ "data-testid": `link-resource-${log.id}`,
4393
+ children: log.resourceId
4394
+ }
4395
+ ) : log.resourceId
4396
+ ] }),
4397
+ /* @__PURE__ */ jsxs7("div", { style: panelStyles3.timestamps, children: [
4398
+ /* @__PURE__ */ jsxs7("span", { children: [
4399
+ "First: ",
4400
+ formatRelativeTime(log.firstSeenAt)
4401
+ ] }),
4402
+ /* @__PURE__ */ jsxs7("span", { children: [
4403
+ "Last: ",
4404
+ formatRelativeTime(log.lastSeenAt)
4405
+ ] })
4406
+ ] }),
4407
+ /* @__PURE__ */ jsx7(
4408
+ "button",
4409
+ {
4410
+ style: panelStyles3.dismissButton,
4411
+ onClick: () => showDismissed ? handleUndismiss(log.id) : handleDismiss(log.id),
4412
+ "data-testid": `button-dismiss-${log.id}`,
4413
+ children: showDismissed ? "Restore" : "Dismiss"
4414
+ }
4415
+ )
4416
+ ]
4417
+ },
4418
+ log.id
4419
+ );
4420
+ }) }),
4421
+ totalPages > 1 && /* @__PURE__ */ jsxs7("div", { style: panelStyles3.pagination, children: [
4422
+ /* @__PURE__ */ jsx7(
4423
+ "button",
4424
+ {
4425
+ style: {
4426
+ ...panelStyles3.pageButton,
4427
+ ...page <= 1 ? panelStyles3.pageButtonDisabled : {}
4428
+ },
4429
+ onClick: () => setPage((p) => Math.max(1, p - 1)),
4430
+ disabled: page <= 1,
4431
+ "data-testid": "button-prev-page",
4432
+ children: "Previous"
4433
+ }
4434
+ ),
4435
+ /* @__PURE__ */ jsxs7("span", { style: { padding: "8px 12px", color: "#6b7280" }, children: [
4436
+ "Page ",
4437
+ page,
4438
+ " of ",
4439
+ totalPages
4440
+ ] }),
4441
+ /* @__PURE__ */ jsx7(
4442
+ "button",
4443
+ {
4444
+ style: {
4445
+ ...panelStyles3.pageButton,
4446
+ ...page >= totalPages ? panelStyles3.pageButtonDisabled : {}
4447
+ },
4448
+ onClick: () => setPage((p) => Math.min(totalPages, p + 1)),
4449
+ disabled: page >= totalPages,
4450
+ "data-testid": "button-next-page",
4451
+ children: "Next"
4452
+ }
4453
+ )
4454
+ ] })
4455
+ ] });
4456
+ }
3393
4457
  export {
3394
4458
  AttemptViewer,
4459
+ ERROR_DEFINITIONS,
4460
+ ErrorLogsPanel,
4461
+ ErrorTypesPanel,
4462
+ MaintenanceScreen,
3395
4463
  QuizApiClient,
3396
4464
  QuizPlayer,
3397
4465
  TextToSpeech,
3398
4466
  calculateScore,
3399
4467
  checkAnswer,
3400
4468
  createAnswerDetail,
3401
- formatTime
4469
+ formatTime,
4470
+ getErrorFromHttpStatus,
4471
+ getErrorFromMessage
3402
4472
  };
3403
4473
  //# sourceMappingURL=index.mjs.map