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