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