@capytale/meta-player 0.8.2 → 0.8.3
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/.prettierrc.json +4 -4
- package/LICENSE +674 -674
- package/README.md +12 -12
- package/eslint.config.js +28 -28
- package/index.html +15 -15
- package/package.json +60 -60
- package/public/themes/lara-dark-blue/theme.css +7015 -7015
- package/public/themes/lara-light-blue/theme.css +7005 -7005
- package/src/App.tsx +139 -139
- package/src/AppRedux.css +39 -39
- package/src/MetaPlayer.tsx +170 -170
- package/src/activityJs.ts +7 -7
- package/src/app/createAppSlice.ts +6 -6
- package/src/app/hooks.ts +12 -12
- package/src/app/store.ts +52 -52
- package/src/app.module.scss +80 -80
- package/src/demo.tsx +87 -87
- package/src/external/prime.ts +5 -5
- package/src/features/activityData/ExitWarning.ts +28 -28
- package/src/features/activityData/IsAntiCheatExitDetectionDisabledSetter.tsx +19 -19
- package/src/features/activityData/IsDirtySetter.tsx +17 -17
- package/src/features/activityData/MetaPlayerOptionsSetter.tsx +41 -41
- package/src/features/activityData/activityDataSlice.ts +256 -256
- package/src/features/activityData/activityJsData.ts +82 -82
- package/src/features/activityData/hooks.ts +34 -34
- package/src/features/activityData/metaPlayerOptions.ts +23 -23
- package/src/features/activityData/uiState.ts +20 -20
- package/src/features/activityJS/ActivityJSProvider.tsx +339 -339
- package/src/features/activityJS/AfterResetAction.tsx +23 -23
- package/src/features/activityJS/AfterSaveAction.tsx +23 -23
- package/src/features/activityJS/BeforeResetAction.tsx +23 -23
- package/src/features/activityJS/BeforeSaveAction.tsx +23 -23
- package/src/features/activityJS/Saver.tsx +167 -167
- package/src/features/activityJS/hooks.ts +85 -85
- package/src/features/activityJS/internal-hooks.ts +96 -96
- package/src/features/activityJS/saverSlice.ts +96 -96
- package/src/features/activitySettings/ActivitySettingsSetter.tsx +21 -21
- package/src/features/activitySettings/hooks.ts +12 -12
- package/src/features/activitySettings/index.tsx +43 -43
- package/src/features/activitySettings/store.ts +108 -108
- package/src/features/activitySettings/style.module.scss +8 -8
- package/src/features/activitySettings/types.ts +140 -140
- package/src/features/activitySettings/ui.tsx +299 -299
- package/src/features/functionalities/AttachedFilesFunctionality.ts +23 -23
- package/src/features/functionalities/PreviewDialog.tsx +83 -83
- package/src/features/functionalities/functionalitiesSlice.ts +98 -98
- package/src/features/functionalities/hooks.ts +70 -70
- package/src/features/layout/hooks.ts +7 -7
- package/src/features/layout/layoutSlice.ts +90 -90
- package/src/features/navbar/activity-info/index.tsx +54 -54
- package/src/features/navbar/activity-info/style.module.scss +38 -38
- package/src/features/navbar/activity-menu/ActivityQuickActions.tsx +57 -57
- package/src/features/navbar/activity-menu/index.tsx +154 -153
- package/src/features/navbar/activity-menu/style.module.scss +7 -7
- package/src/features/navbar/capytale-menu/CloneDialog.tsx +75 -75
- package/src/features/navbar/capytale-menu/Countdown.tsx +312 -312
- package/src/features/navbar/capytale-menu/CountdownAndSaveButton.tsx +115 -115
- package/src/features/navbar/capytale-menu/index.tsx +260 -260
- package/src/features/navbar/capytale-menu/style.module.scss +8 -8
- package/src/features/navbar/index.tsx +39 -39
- package/src/features/navbar/navbarSlice.ts +79 -79
- package/src/features/navbar/review-navbar/GradingNav.tsx +128 -128
- package/src/features/navbar/review-navbar/index.tsx +18 -18
- package/src/features/navbar/review-navbar/style.module.scss +22 -22
- package/src/features/navbar/sidebars/ActivitySidebarActions.tsx +57 -57
- package/src/features/navbar/sidebars/AttachedFilesSidebarContent.module.scss +43 -43
- package/src/features/navbar/sidebars/AttachedFilesSidebarContent.tsx +181 -181
- package/src/features/navbar/sidebars/SettingsSidebarContent.tsx +192 -192
- package/src/features/navbar/sidebars/style.module.scss +15 -15
- package/src/features/navbar/student-utils.ts +11 -11
- package/src/features/navbar/style.module.scss +65 -65
- package/src/features/pedago/InstructionsEditor.tsx +102 -102
- package/src/features/pedago/PdfEditor.tsx +91 -91
- package/src/features/pedago/PedagoCommands.tsx +353 -353
- package/src/features/pedago/SharedNotesEditor.tsx +144 -144
- package/src/features/pedago/index.tsx +207 -204
- package/src/features/pedago/style.module.scss +233 -233
- package/src/features/theming/ThemeSwitcher.tsx +51 -51
- package/src/features/theming/hooks.ts +6 -6
- package/src/features/theming/themingSlice.ts +93 -93
- package/src/features/toast.tsx +38 -38
- package/src/hooks/index.ts +16 -16
- package/src/index.css +132 -132
- package/src/index.ts +90 -90
- package/src/logo.svg +1 -1
- package/src/my_json_data.js +4146 -4146
- package/src/settings.ts +6 -6
- package/src/types.ts +9 -9
- package/src/utils/ButtonDoubleIcon.tsx +35 -35
- package/src/utils/CardSelector.tsx +41 -41
- package/src/utils/ErrorBoundary.tsx +41 -41
- package/src/utils/PopupButton.tsx +134 -134
- package/src/utils/activity.ts +8 -8
- package/src/utils/breakpoints.ts +4 -4
- package/src/utils/capytale.ts +10 -10
- package/src/utils/clipboard.ts +11 -11
- package/src/utils/download.ts +7 -7
- package/src/utils/equality.ts +32 -32
- package/src/utils/server-clock.ts +42 -42
- package/src/utils/style.module.scss +45 -45
- package/src/utils/useFullscreen.ts +65 -65
- package/src/vite-env.d.ts +1 -1
- package/tsconfig.json +28 -28
- package/tsconfig.node.json +9 -9
- package/vite.config.ts +11 -11
|
@@ -1,339 +1,339 @@
|
|
|
1
|
-
import {
|
|
2
|
-
FC,
|
|
3
|
-
PropsWithChildren,
|
|
4
|
-
ReactNode,
|
|
5
|
-
createContext,
|
|
6
|
-
useCallback,
|
|
7
|
-
useContext,
|
|
8
|
-
useEffect,
|
|
9
|
-
useMemo,
|
|
10
|
-
useState,
|
|
11
|
-
} from "react";
|
|
12
|
-
|
|
13
|
-
import {
|
|
14
|
-
ActivitySessionLoaderReturnValue,
|
|
15
|
-
LoadOptions,
|
|
16
|
-
useActivitySessionLoader,
|
|
17
|
-
} from "./internal-hooks";
|
|
18
|
-
import { getIdFromUrl } from "../../utils/activity";
|
|
19
|
-
import { useAppDispatch } from "../../app/hooks";
|
|
20
|
-
import ActivitySession from "@capytale/activity.js/activity/activitySession";
|
|
21
|
-
import { setActivityJSData } from "../activityData/activityDataSlice";
|
|
22
|
-
|
|
23
|
-
import tracker from "@capytale/activity.js/backend/capytale/tracker";
|
|
24
|
-
|
|
25
|
-
const ActivityJSContext = createContext<ActivitySessionLoaderReturnValue>({
|
|
26
|
-
state: "initial",
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
export type BinaryDataMode =
|
|
30
|
-
| "json"
|
|
31
|
-
| "text"
|
|
32
|
-
| "blob"
|
|
33
|
-
| "arrayBuffer"
|
|
34
|
-
| "stream";
|
|
35
|
-
|
|
36
|
-
type ActivityJSProviderProps = PropsWithChildren<{
|
|
37
|
-
options?: LoadOptions;
|
|
38
|
-
loadingFallback?: ReactNode;
|
|
39
|
-
errorFallback?:
|
|
40
|
-
| keyof JSX.IntrinsicElements
|
|
41
|
-
| React.JSXElementConstructor<{ children: ReactNode }>;
|
|
42
|
-
}>;
|
|
43
|
-
|
|
44
|
-
export const ActivityJSProvider: FC<ActivityJSProviderProps> = (props) => {
|
|
45
|
-
const [loaded, setLoaded] = useState(false);
|
|
46
|
-
const activityId = getIdFromUrl();
|
|
47
|
-
const dispatch = useAppDispatch();
|
|
48
|
-
|
|
49
|
-
useEffect(() => {
|
|
50
|
-
tracker.trackPageView();
|
|
51
|
-
}, []);
|
|
52
|
-
|
|
53
|
-
const callback = (data: ActivitySession) => {
|
|
54
|
-
(window as any).capy = data;
|
|
55
|
-
|
|
56
|
-
const ab = data.activityBunch;
|
|
57
|
-
|
|
58
|
-
const sharedNotesContent = ab.shared_notes.value;
|
|
59
|
-
|
|
60
|
-
dispatch(
|
|
61
|
-
setActivityJSData({
|
|
62
|
-
title: ab.title.value || "",
|
|
63
|
-
mode: data.mode,
|
|
64
|
-
returnUrl: data.returnUrl,
|
|
65
|
-
helpUrl: data.type.helpUrl,
|
|
66
|
-
nid: ab.mainNode.nid,
|
|
67
|
-
activityNid: ab.activityNode.nid,
|
|
68
|
-
code: data.code,
|
|
69
|
-
codeLink: data.codeLink,
|
|
70
|
-
accessTrMode: ab.access_tr_mode.value || "",
|
|
71
|
-
accessTimerange: ab.access_timerange.value,
|
|
72
|
-
studentInfo: data.student
|
|
73
|
-
? {
|
|
74
|
-
firstName: data.student.prenom || "",
|
|
75
|
-
lastName: data.student.nom || "",
|
|
76
|
-
class: data.student.classe || "",
|
|
77
|
-
}
|
|
78
|
-
: null,
|
|
79
|
-
instructions: ab.instructions.value.lexical
|
|
80
|
-
? {
|
|
81
|
-
value: ab.instructions.value.lexical,
|
|
82
|
-
htmlValue: ab.instructions.value.html,
|
|
83
|
-
format: "lexical",
|
|
84
|
-
}
|
|
85
|
-
: {
|
|
86
|
-
value: null,
|
|
87
|
-
htmlValue: ab.instructions.value.html || "",
|
|
88
|
-
format: "html",
|
|
89
|
-
},
|
|
90
|
-
instructionsType:
|
|
91
|
-
ab.instructions.value.lexical ||
|
|
92
|
-
ab.instructions.value.html ||
|
|
93
|
-
(data.mode === "create" && ab.activityNode.isNew)
|
|
94
|
-
? "rich"
|
|
95
|
-
: "none",
|
|
96
|
-
pdfInstructions: null,
|
|
97
|
-
sharedNotesContent: sharedNotesContent,
|
|
98
|
-
sharedNotesType: sharedNotesContent == null ? "none" : "rich",
|
|
99
|
-
icon: data.icon || null,
|
|
100
|
-
friendlyType: data.friendlyType,
|
|
101
|
-
comments: ab.assignmentNode?.comments.value || null,
|
|
102
|
-
grading: ab.assignmentNode?.grading.value || null,
|
|
103
|
-
workflow: ab.assignmentNode?.workflow,
|
|
104
|
-
antiCheat: ab.hasAntiCheat
|
|
105
|
-
? {
|
|
106
|
-
passwordHash: ab.antiCheatPasswd,
|
|
107
|
-
startLocked: !ab.assignmentNode?.isNew,
|
|
108
|
-
}
|
|
109
|
-
: null,
|
|
110
|
-
hasEvaluations: ab.activityNode.has_evaluation.value,
|
|
111
|
-
}),
|
|
112
|
-
);
|
|
113
|
-
tracker.trackActivity(data);
|
|
114
|
-
setLoaded(true);
|
|
115
|
-
};
|
|
116
|
-
|
|
117
|
-
if (!activityId) {
|
|
118
|
-
throw new Error("No activity id found in the URL");
|
|
119
|
-
}
|
|
120
|
-
const value = useActivitySessionLoader(activityId, props.options, callback);
|
|
121
|
-
if (!loaded || value.state !== "loaded") {
|
|
122
|
-
if (loaded) {
|
|
123
|
-
console.warn("Should not happen: loaded but not loaded state");
|
|
124
|
-
}
|
|
125
|
-
if (value.state === "loading" || value.state === "initial") {
|
|
126
|
-
return <>{props.loadingFallback}</>;
|
|
127
|
-
} else if (value.state === "error") {
|
|
128
|
-
console.error(value.error);
|
|
129
|
-
if (props.errorFallback) {
|
|
130
|
-
return (
|
|
131
|
-
<>
|
|
132
|
-
{<props.errorFallback>{String(value.error)}</props.errorFallback>}
|
|
133
|
-
</>
|
|
134
|
-
);
|
|
135
|
-
}
|
|
136
|
-
return (
|
|
137
|
-
<div>
|
|
138
|
-
<h2>Error loading ActivityJS data</h2>
|
|
139
|
-
<p>{String(value.error)}</p>
|
|
140
|
-
</div>
|
|
141
|
-
);
|
|
142
|
-
}
|
|
143
|
-
console.error("Unknown state:", value.state);
|
|
144
|
-
return (
|
|
145
|
-
<div>
|
|
146
|
-
<h2>Unknown state</h2>
|
|
147
|
-
<p>{String(value.state)}</p>
|
|
148
|
-
</div>
|
|
149
|
-
);
|
|
150
|
-
}
|
|
151
|
-
return (
|
|
152
|
-
<ActivityJSContext.Provider value={value}>
|
|
153
|
-
{props.children}
|
|
154
|
-
</ActivityJSContext.Provider>
|
|
155
|
-
);
|
|
156
|
-
};
|
|
157
|
-
|
|
158
|
-
export const useActivityJS = () => {
|
|
159
|
-
const activityJs = useContext(ActivityJSContext);
|
|
160
|
-
if (!activityJs) {
|
|
161
|
-
throw new Error("useActivityJS must be used within an ActivityJSProvider");
|
|
162
|
-
}
|
|
163
|
-
if (activityJs.state === "initial") {
|
|
164
|
-
throw new Error("ActivityJS data not loaded (initial)");
|
|
165
|
-
}
|
|
166
|
-
if (activityJs.state === "loading") {
|
|
167
|
-
throw new Error("ActivityJS data not loaded (loading)");
|
|
168
|
-
}
|
|
169
|
-
if (activityJs.state === "error") {
|
|
170
|
-
throw new Error("Error loading ActivityJS data: " + activityJs.error);
|
|
171
|
-
}
|
|
172
|
-
return activityJs;
|
|
173
|
-
};
|
|
174
|
-
|
|
175
|
-
export const useActivitySession = () => {
|
|
176
|
-
const activityJs = useActivityJS();
|
|
177
|
-
return activityJs.activitySession;
|
|
178
|
-
};
|
|
179
|
-
|
|
180
|
-
export const useActivityBunch = () => {
|
|
181
|
-
const activitySession = useActivitySession();
|
|
182
|
-
const ab = activitySession.activityBunch;
|
|
183
|
-
return ab;
|
|
184
|
-
};
|
|
185
|
-
|
|
186
|
-
export const useActivityJsEssentials = () => {
|
|
187
|
-
const activitySession = useActivitySession();
|
|
188
|
-
const mode = activitySession.mode;
|
|
189
|
-
const ab = activitySession.activityBunch;
|
|
190
|
-
const hasAssignment = !!ab.assignmentNode;
|
|
191
|
-
const getActivityContent = useCallback(() => {
|
|
192
|
-
return ab.activityNode.content.value as string | null;
|
|
193
|
-
}, [ab.activityNode.content]);
|
|
194
|
-
const getAssignmentContent = useCallback(() => {
|
|
195
|
-
if (!ab.assignmentNode) {
|
|
196
|
-
throw new Error("No assignment node");
|
|
197
|
-
}
|
|
198
|
-
return ab.assignmentNode.content.value as string | null;
|
|
199
|
-
}, [ab.assignmentNode?.content]);
|
|
200
|
-
const getContent = useCallback(() => {
|
|
201
|
-
if (!hasAssignment) {
|
|
202
|
-
return getActivityContent();
|
|
203
|
-
}
|
|
204
|
-
const assignmentContent = getAssignmentContent();
|
|
205
|
-
if (assignmentContent != null) {
|
|
206
|
-
return assignmentContent;
|
|
207
|
-
}
|
|
208
|
-
return getActivityContent();
|
|
209
|
-
}, [getActivityContent, getAssignmentContent]);
|
|
210
|
-
const getActivityBinaryData = useCallback(() => {
|
|
211
|
-
return ab.activityNode.binaryData.value as any;
|
|
212
|
-
}, [ab.activityNode.binaryData]);
|
|
213
|
-
const getAssignmentBinaryData = useCallback(() => {
|
|
214
|
-
if (!ab.assignmentNode) {
|
|
215
|
-
throw new Error("No assignment node");
|
|
216
|
-
}
|
|
217
|
-
return ab.assignmentNode.binaryData.value as any;
|
|
218
|
-
}, [ab.assignmentNode?.binaryData]);
|
|
219
|
-
const getBinaryData = useCallback(() => {
|
|
220
|
-
if (!hasAssignment) {
|
|
221
|
-
return getActivityBinaryData();
|
|
222
|
-
}
|
|
223
|
-
const assignmentBinaryData = getAssignmentBinaryData();
|
|
224
|
-
if (assignmentBinaryData != null) {
|
|
225
|
-
return assignmentBinaryData;
|
|
226
|
-
}
|
|
227
|
-
return getActivityBinaryData();
|
|
228
|
-
}, [getActivityBinaryData, getAssignmentBinaryData]);
|
|
229
|
-
|
|
230
|
-
const retrieveAndGetActivityBinaryData = useCallback(
|
|
231
|
-
async (mode: BinaryDataMode) => {
|
|
232
|
-
await ab.activityNode.binaryData.loadAsync(mode);
|
|
233
|
-
return ab.activityNode.binaryData.value as any;
|
|
234
|
-
},
|
|
235
|
-
[ab.activityNode.binaryData],
|
|
236
|
-
);
|
|
237
|
-
const retrieveAndGetAssignmentBinaryData = useCallback(
|
|
238
|
-
async (mode: BinaryDataMode) => {
|
|
239
|
-
if (!ab.assignmentNode) {
|
|
240
|
-
throw new Error("No assignment node");
|
|
241
|
-
}
|
|
242
|
-
await ab.assignmentNode.binaryData.loadAsync(mode);
|
|
243
|
-
return ab.assignmentNode.binaryData.value as any;
|
|
244
|
-
},
|
|
245
|
-
[ab.assignmentNode?.binaryData],
|
|
246
|
-
);
|
|
247
|
-
const retrieveAndGetBinaryData = useCallback(
|
|
248
|
-
async (mode: BinaryDataMode) => {
|
|
249
|
-
if (!hasAssignment) {
|
|
250
|
-
return await retrieveAndGetActivityBinaryData(mode);
|
|
251
|
-
}
|
|
252
|
-
const assignmentBinaryData =
|
|
253
|
-
await retrieveAndGetAssignmentBinaryData(mode);
|
|
254
|
-
if (assignmentBinaryData != null) {
|
|
255
|
-
return assignmentBinaryData;
|
|
256
|
-
}
|
|
257
|
-
return await retrieveAndGetActivityBinaryData(mode);
|
|
258
|
-
},
|
|
259
|
-
[
|
|
260
|
-
hasAssignment,
|
|
261
|
-
retrieveAndGetActivityBinaryData,
|
|
262
|
-
retrieveAndGetAssignmentBinaryData,
|
|
263
|
-
],
|
|
264
|
-
);
|
|
265
|
-
|
|
266
|
-
const setActivityContent = useCallback(
|
|
267
|
-
(content: string | null) => {
|
|
268
|
-
ab.activityNode.content.value = content;
|
|
269
|
-
},
|
|
270
|
-
[ab.activityNode.content],
|
|
271
|
-
);
|
|
272
|
-
const setAssignmentContent = useCallback(
|
|
273
|
-
(content: string | null) => {
|
|
274
|
-
if (!ab.assignmentNode) {
|
|
275
|
-
throw new Error("No assignment node");
|
|
276
|
-
}
|
|
277
|
-
ab.assignmentNode.content.value = content;
|
|
278
|
-
},
|
|
279
|
-
[ab.assignmentNode?.content],
|
|
280
|
-
);
|
|
281
|
-
const setContent = useCallback(
|
|
282
|
-
(content: string | null) => {
|
|
283
|
-
if (!hasAssignment) {
|
|
284
|
-
setActivityContent(content);
|
|
285
|
-
} else {
|
|
286
|
-
setAssignmentContent(content);
|
|
287
|
-
}
|
|
288
|
-
},
|
|
289
|
-
[hasAssignment, setActivityContent, setAssignmentContent],
|
|
290
|
-
);
|
|
291
|
-
const setActivityBinaryData = useCallback(
|
|
292
|
-
(data: any | null) => {
|
|
293
|
-
ab.activityNode.binaryData.value = data;
|
|
294
|
-
},
|
|
295
|
-
[ab.activityNode.binaryData],
|
|
296
|
-
);
|
|
297
|
-
const setAssignmentBinaryData = useCallback(
|
|
298
|
-
(data: any | null) => {
|
|
299
|
-
if (!ab.assignmentNode) {
|
|
300
|
-
throw new Error("No assignment node");
|
|
301
|
-
}
|
|
302
|
-
ab.assignmentNode.binaryData.value = data;
|
|
303
|
-
},
|
|
304
|
-
[ab.assignmentNode?.binaryData],
|
|
305
|
-
);
|
|
306
|
-
const setBinaryData = useCallback(
|
|
307
|
-
(data: any | null) => {
|
|
308
|
-
if (!hasAssignment) {
|
|
309
|
-
setActivityBinaryData(data);
|
|
310
|
-
} else {
|
|
311
|
-
setAssignmentBinaryData(data);
|
|
312
|
-
}
|
|
313
|
-
},
|
|
314
|
-
[hasAssignment, setActivityBinaryData, setAssignmentBinaryData],
|
|
315
|
-
);
|
|
316
|
-
const title = useMemo(() => ab.title.value, [ab.title.value]);
|
|
317
|
-
const nid = useMemo(() => ab.mainNode.nid, [ab.mainNode.nid]);
|
|
318
|
-
return {
|
|
319
|
-
mode,
|
|
320
|
-
title,
|
|
321
|
-
hasAssignment,
|
|
322
|
-
nid,
|
|
323
|
-
getActivityContent,
|
|
324
|
-
getAssignmentContent,
|
|
325
|
-
getContent,
|
|
326
|
-
getActivityBinaryData,
|
|
327
|
-
getAssignmentBinaryData,
|
|
328
|
-
getBinaryData,
|
|
329
|
-
retrieveAndGetActivityBinaryData,
|
|
330
|
-
retrieveAndGetAssignmentBinaryData,
|
|
331
|
-
retrieveAndGetBinaryData,
|
|
332
|
-
setActivityContent,
|
|
333
|
-
setAssignmentContent,
|
|
334
|
-
setContent,
|
|
335
|
-
setActivityBinaryData,
|
|
336
|
-
setAssignmentBinaryData,
|
|
337
|
-
setBinaryData,
|
|
338
|
-
};
|
|
339
|
-
};
|
|
1
|
+
import {
|
|
2
|
+
FC,
|
|
3
|
+
PropsWithChildren,
|
|
4
|
+
ReactNode,
|
|
5
|
+
createContext,
|
|
6
|
+
useCallback,
|
|
7
|
+
useContext,
|
|
8
|
+
useEffect,
|
|
9
|
+
useMemo,
|
|
10
|
+
useState,
|
|
11
|
+
} from "react";
|
|
12
|
+
|
|
13
|
+
import {
|
|
14
|
+
ActivitySessionLoaderReturnValue,
|
|
15
|
+
LoadOptions,
|
|
16
|
+
useActivitySessionLoader,
|
|
17
|
+
} from "./internal-hooks";
|
|
18
|
+
import { getIdFromUrl } from "../../utils/activity";
|
|
19
|
+
import { useAppDispatch } from "../../app/hooks";
|
|
20
|
+
import ActivitySession from "@capytale/activity.js/activity/activitySession";
|
|
21
|
+
import { setActivityJSData } from "../activityData/activityDataSlice";
|
|
22
|
+
|
|
23
|
+
import tracker from "@capytale/activity.js/backend/capytale/tracker";
|
|
24
|
+
|
|
25
|
+
const ActivityJSContext = createContext<ActivitySessionLoaderReturnValue>({
|
|
26
|
+
state: "initial",
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
export type BinaryDataMode =
|
|
30
|
+
| "json"
|
|
31
|
+
| "text"
|
|
32
|
+
| "blob"
|
|
33
|
+
| "arrayBuffer"
|
|
34
|
+
| "stream";
|
|
35
|
+
|
|
36
|
+
type ActivityJSProviderProps = PropsWithChildren<{
|
|
37
|
+
options?: LoadOptions;
|
|
38
|
+
loadingFallback?: ReactNode;
|
|
39
|
+
errorFallback?:
|
|
40
|
+
| keyof JSX.IntrinsicElements
|
|
41
|
+
| React.JSXElementConstructor<{ children: ReactNode }>;
|
|
42
|
+
}>;
|
|
43
|
+
|
|
44
|
+
export const ActivityJSProvider: FC<ActivityJSProviderProps> = (props) => {
|
|
45
|
+
const [loaded, setLoaded] = useState(false);
|
|
46
|
+
const activityId = getIdFromUrl();
|
|
47
|
+
const dispatch = useAppDispatch();
|
|
48
|
+
|
|
49
|
+
useEffect(() => {
|
|
50
|
+
tracker.trackPageView();
|
|
51
|
+
}, []);
|
|
52
|
+
|
|
53
|
+
const callback = (data: ActivitySession) => {
|
|
54
|
+
(window as any).capy = data;
|
|
55
|
+
|
|
56
|
+
const ab = data.activityBunch;
|
|
57
|
+
|
|
58
|
+
const sharedNotesContent = ab.shared_notes.value;
|
|
59
|
+
|
|
60
|
+
dispatch(
|
|
61
|
+
setActivityJSData({
|
|
62
|
+
title: ab.title.value || "",
|
|
63
|
+
mode: data.mode,
|
|
64
|
+
returnUrl: data.returnUrl,
|
|
65
|
+
helpUrl: data.type.helpUrl,
|
|
66
|
+
nid: ab.mainNode.nid,
|
|
67
|
+
activityNid: ab.activityNode.nid,
|
|
68
|
+
code: data.code,
|
|
69
|
+
codeLink: data.codeLink,
|
|
70
|
+
accessTrMode: ab.access_tr_mode.value || "",
|
|
71
|
+
accessTimerange: ab.access_timerange.value,
|
|
72
|
+
studentInfo: data.student
|
|
73
|
+
? {
|
|
74
|
+
firstName: data.student.prenom || "",
|
|
75
|
+
lastName: data.student.nom || "",
|
|
76
|
+
class: data.student.classe || "",
|
|
77
|
+
}
|
|
78
|
+
: null,
|
|
79
|
+
instructions: ab.instructions.value.lexical
|
|
80
|
+
? {
|
|
81
|
+
value: ab.instructions.value.lexical,
|
|
82
|
+
htmlValue: ab.instructions.value.html,
|
|
83
|
+
format: "lexical",
|
|
84
|
+
}
|
|
85
|
+
: {
|
|
86
|
+
value: null,
|
|
87
|
+
htmlValue: ab.instructions.value.html || "",
|
|
88
|
+
format: "html",
|
|
89
|
+
},
|
|
90
|
+
instructionsType:
|
|
91
|
+
ab.instructions.value.lexical ||
|
|
92
|
+
ab.instructions.value.html ||
|
|
93
|
+
(data.mode === "create" && ab.activityNode.isNew)
|
|
94
|
+
? "rich"
|
|
95
|
+
: "none",
|
|
96
|
+
pdfInstructions: null,
|
|
97
|
+
sharedNotesContent: sharedNotesContent,
|
|
98
|
+
sharedNotesType: sharedNotesContent == null ? "none" : "rich",
|
|
99
|
+
icon: data.icon || null,
|
|
100
|
+
friendlyType: data.friendlyType,
|
|
101
|
+
comments: ab.assignmentNode?.comments.value || null,
|
|
102
|
+
grading: ab.assignmentNode?.grading.value || null,
|
|
103
|
+
workflow: ab.assignmentNode?.workflow,
|
|
104
|
+
antiCheat: ab.hasAntiCheat
|
|
105
|
+
? {
|
|
106
|
+
passwordHash: ab.antiCheatPasswd,
|
|
107
|
+
startLocked: !ab.assignmentNode?.isNew,
|
|
108
|
+
}
|
|
109
|
+
: null,
|
|
110
|
+
hasEvaluations: ab.activityNode.has_evaluation.value,
|
|
111
|
+
}),
|
|
112
|
+
);
|
|
113
|
+
tracker.trackActivity(data);
|
|
114
|
+
setLoaded(true);
|
|
115
|
+
};
|
|
116
|
+
|
|
117
|
+
if (!activityId) {
|
|
118
|
+
throw new Error("No activity id found in the URL");
|
|
119
|
+
}
|
|
120
|
+
const value = useActivitySessionLoader(activityId, props.options, callback);
|
|
121
|
+
if (!loaded || value.state !== "loaded") {
|
|
122
|
+
if (loaded) {
|
|
123
|
+
console.warn("Should not happen: loaded but not loaded state");
|
|
124
|
+
}
|
|
125
|
+
if (value.state === "loading" || value.state === "initial") {
|
|
126
|
+
return <>{props.loadingFallback}</>;
|
|
127
|
+
} else if (value.state === "error") {
|
|
128
|
+
console.error(value.error);
|
|
129
|
+
if (props.errorFallback) {
|
|
130
|
+
return (
|
|
131
|
+
<>
|
|
132
|
+
{<props.errorFallback>{String(value.error)}</props.errorFallback>}
|
|
133
|
+
</>
|
|
134
|
+
);
|
|
135
|
+
}
|
|
136
|
+
return (
|
|
137
|
+
<div>
|
|
138
|
+
<h2>Error loading ActivityJS data</h2>
|
|
139
|
+
<p>{String(value.error)}</p>
|
|
140
|
+
</div>
|
|
141
|
+
);
|
|
142
|
+
}
|
|
143
|
+
console.error("Unknown state:", value.state);
|
|
144
|
+
return (
|
|
145
|
+
<div>
|
|
146
|
+
<h2>Unknown state</h2>
|
|
147
|
+
<p>{String(value.state)}</p>
|
|
148
|
+
</div>
|
|
149
|
+
);
|
|
150
|
+
}
|
|
151
|
+
return (
|
|
152
|
+
<ActivityJSContext.Provider value={value}>
|
|
153
|
+
{props.children}
|
|
154
|
+
</ActivityJSContext.Provider>
|
|
155
|
+
);
|
|
156
|
+
};
|
|
157
|
+
|
|
158
|
+
export const useActivityJS = () => {
|
|
159
|
+
const activityJs = useContext(ActivityJSContext);
|
|
160
|
+
if (!activityJs) {
|
|
161
|
+
throw new Error("useActivityJS must be used within an ActivityJSProvider");
|
|
162
|
+
}
|
|
163
|
+
if (activityJs.state === "initial") {
|
|
164
|
+
throw new Error("ActivityJS data not loaded (initial)");
|
|
165
|
+
}
|
|
166
|
+
if (activityJs.state === "loading") {
|
|
167
|
+
throw new Error("ActivityJS data not loaded (loading)");
|
|
168
|
+
}
|
|
169
|
+
if (activityJs.state === "error") {
|
|
170
|
+
throw new Error("Error loading ActivityJS data: " + activityJs.error);
|
|
171
|
+
}
|
|
172
|
+
return activityJs;
|
|
173
|
+
};
|
|
174
|
+
|
|
175
|
+
export const useActivitySession = () => {
|
|
176
|
+
const activityJs = useActivityJS();
|
|
177
|
+
return activityJs.activitySession;
|
|
178
|
+
};
|
|
179
|
+
|
|
180
|
+
export const useActivityBunch = () => {
|
|
181
|
+
const activitySession = useActivitySession();
|
|
182
|
+
const ab = activitySession.activityBunch;
|
|
183
|
+
return ab;
|
|
184
|
+
};
|
|
185
|
+
|
|
186
|
+
export const useActivityJsEssentials = () => {
|
|
187
|
+
const activitySession = useActivitySession();
|
|
188
|
+
const mode = activitySession.mode;
|
|
189
|
+
const ab = activitySession.activityBunch;
|
|
190
|
+
const hasAssignment = !!ab.assignmentNode;
|
|
191
|
+
const getActivityContent = useCallback(() => {
|
|
192
|
+
return ab.activityNode.content.value as string | null;
|
|
193
|
+
}, [ab.activityNode.content]);
|
|
194
|
+
const getAssignmentContent = useCallback(() => {
|
|
195
|
+
if (!ab.assignmentNode) {
|
|
196
|
+
throw new Error("No assignment node");
|
|
197
|
+
}
|
|
198
|
+
return ab.assignmentNode.content.value as string | null;
|
|
199
|
+
}, [ab.assignmentNode?.content]);
|
|
200
|
+
const getContent = useCallback(() => {
|
|
201
|
+
if (!hasAssignment) {
|
|
202
|
+
return getActivityContent();
|
|
203
|
+
}
|
|
204
|
+
const assignmentContent = getAssignmentContent();
|
|
205
|
+
if (assignmentContent != null) {
|
|
206
|
+
return assignmentContent;
|
|
207
|
+
}
|
|
208
|
+
return getActivityContent();
|
|
209
|
+
}, [getActivityContent, getAssignmentContent]);
|
|
210
|
+
const getActivityBinaryData = useCallback(() => {
|
|
211
|
+
return ab.activityNode.binaryData.value as any;
|
|
212
|
+
}, [ab.activityNode.binaryData]);
|
|
213
|
+
const getAssignmentBinaryData = useCallback(() => {
|
|
214
|
+
if (!ab.assignmentNode) {
|
|
215
|
+
throw new Error("No assignment node");
|
|
216
|
+
}
|
|
217
|
+
return ab.assignmentNode.binaryData.value as any;
|
|
218
|
+
}, [ab.assignmentNode?.binaryData]);
|
|
219
|
+
const getBinaryData = useCallback(() => {
|
|
220
|
+
if (!hasAssignment) {
|
|
221
|
+
return getActivityBinaryData();
|
|
222
|
+
}
|
|
223
|
+
const assignmentBinaryData = getAssignmentBinaryData();
|
|
224
|
+
if (assignmentBinaryData != null) {
|
|
225
|
+
return assignmentBinaryData;
|
|
226
|
+
}
|
|
227
|
+
return getActivityBinaryData();
|
|
228
|
+
}, [getActivityBinaryData, getAssignmentBinaryData]);
|
|
229
|
+
|
|
230
|
+
const retrieveAndGetActivityBinaryData = useCallback(
|
|
231
|
+
async (mode: BinaryDataMode) => {
|
|
232
|
+
await ab.activityNode.binaryData.loadAsync(mode);
|
|
233
|
+
return ab.activityNode.binaryData.value as any;
|
|
234
|
+
},
|
|
235
|
+
[ab.activityNode.binaryData],
|
|
236
|
+
);
|
|
237
|
+
const retrieveAndGetAssignmentBinaryData = useCallback(
|
|
238
|
+
async (mode: BinaryDataMode) => {
|
|
239
|
+
if (!ab.assignmentNode) {
|
|
240
|
+
throw new Error("No assignment node");
|
|
241
|
+
}
|
|
242
|
+
await ab.assignmentNode.binaryData.loadAsync(mode);
|
|
243
|
+
return ab.assignmentNode.binaryData.value as any;
|
|
244
|
+
},
|
|
245
|
+
[ab.assignmentNode?.binaryData],
|
|
246
|
+
);
|
|
247
|
+
const retrieveAndGetBinaryData = useCallback(
|
|
248
|
+
async (mode: BinaryDataMode) => {
|
|
249
|
+
if (!hasAssignment) {
|
|
250
|
+
return await retrieveAndGetActivityBinaryData(mode);
|
|
251
|
+
}
|
|
252
|
+
const assignmentBinaryData =
|
|
253
|
+
await retrieveAndGetAssignmentBinaryData(mode);
|
|
254
|
+
if (assignmentBinaryData != null) {
|
|
255
|
+
return assignmentBinaryData;
|
|
256
|
+
}
|
|
257
|
+
return await retrieveAndGetActivityBinaryData(mode);
|
|
258
|
+
},
|
|
259
|
+
[
|
|
260
|
+
hasAssignment,
|
|
261
|
+
retrieveAndGetActivityBinaryData,
|
|
262
|
+
retrieveAndGetAssignmentBinaryData,
|
|
263
|
+
],
|
|
264
|
+
);
|
|
265
|
+
|
|
266
|
+
const setActivityContent = useCallback(
|
|
267
|
+
(content: string | null) => {
|
|
268
|
+
ab.activityNode.content.value = content;
|
|
269
|
+
},
|
|
270
|
+
[ab.activityNode.content],
|
|
271
|
+
);
|
|
272
|
+
const setAssignmentContent = useCallback(
|
|
273
|
+
(content: string | null) => {
|
|
274
|
+
if (!ab.assignmentNode) {
|
|
275
|
+
throw new Error("No assignment node");
|
|
276
|
+
}
|
|
277
|
+
ab.assignmentNode.content.value = content;
|
|
278
|
+
},
|
|
279
|
+
[ab.assignmentNode?.content],
|
|
280
|
+
);
|
|
281
|
+
const setContent = useCallback(
|
|
282
|
+
(content: string | null) => {
|
|
283
|
+
if (!hasAssignment) {
|
|
284
|
+
setActivityContent(content);
|
|
285
|
+
} else {
|
|
286
|
+
setAssignmentContent(content);
|
|
287
|
+
}
|
|
288
|
+
},
|
|
289
|
+
[hasAssignment, setActivityContent, setAssignmentContent],
|
|
290
|
+
);
|
|
291
|
+
const setActivityBinaryData = useCallback(
|
|
292
|
+
(data: any | null) => {
|
|
293
|
+
ab.activityNode.binaryData.value = data;
|
|
294
|
+
},
|
|
295
|
+
[ab.activityNode.binaryData],
|
|
296
|
+
);
|
|
297
|
+
const setAssignmentBinaryData = useCallback(
|
|
298
|
+
(data: any | null) => {
|
|
299
|
+
if (!ab.assignmentNode) {
|
|
300
|
+
throw new Error("No assignment node");
|
|
301
|
+
}
|
|
302
|
+
ab.assignmentNode.binaryData.value = data;
|
|
303
|
+
},
|
|
304
|
+
[ab.assignmentNode?.binaryData],
|
|
305
|
+
);
|
|
306
|
+
const setBinaryData = useCallback(
|
|
307
|
+
(data: any | null) => {
|
|
308
|
+
if (!hasAssignment) {
|
|
309
|
+
setActivityBinaryData(data);
|
|
310
|
+
} else {
|
|
311
|
+
setAssignmentBinaryData(data);
|
|
312
|
+
}
|
|
313
|
+
},
|
|
314
|
+
[hasAssignment, setActivityBinaryData, setAssignmentBinaryData],
|
|
315
|
+
);
|
|
316
|
+
const title = useMemo(() => ab.title.value, [ab.title.value]);
|
|
317
|
+
const nid = useMemo(() => ab.mainNode.nid, [ab.mainNode.nid]);
|
|
318
|
+
return {
|
|
319
|
+
mode,
|
|
320
|
+
title,
|
|
321
|
+
hasAssignment,
|
|
322
|
+
nid,
|
|
323
|
+
getActivityContent,
|
|
324
|
+
getAssignmentContent,
|
|
325
|
+
getContent,
|
|
326
|
+
getActivityBinaryData,
|
|
327
|
+
getAssignmentBinaryData,
|
|
328
|
+
getBinaryData,
|
|
329
|
+
retrieveAndGetActivityBinaryData,
|
|
330
|
+
retrieveAndGetAssignmentBinaryData,
|
|
331
|
+
retrieveAndGetBinaryData,
|
|
332
|
+
setActivityContent,
|
|
333
|
+
setAssignmentContent,
|
|
334
|
+
setContent,
|
|
335
|
+
setActivityBinaryData,
|
|
336
|
+
setAssignmentBinaryData,
|
|
337
|
+
setBinaryData,
|
|
338
|
+
};
|
|
339
|
+
};
|