@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.
Files changed (105) hide show
  1. package/.prettierrc.json +4 -4
  2. package/LICENSE +674 -674
  3. package/README.md +12 -12
  4. package/eslint.config.js +28 -28
  5. package/index.html +15 -15
  6. package/package.json +60 -60
  7. package/public/themes/lara-dark-blue/theme.css +7015 -7015
  8. package/public/themes/lara-light-blue/theme.css +7005 -7005
  9. package/src/App.tsx +139 -139
  10. package/src/AppRedux.css +39 -39
  11. package/src/MetaPlayer.tsx +170 -170
  12. package/src/activityJs.ts +7 -7
  13. package/src/app/createAppSlice.ts +6 -6
  14. package/src/app/hooks.ts +12 -12
  15. package/src/app/store.ts +52 -52
  16. package/src/app.module.scss +80 -80
  17. package/src/demo.tsx +87 -87
  18. package/src/external/prime.ts +5 -5
  19. package/src/features/activityData/ExitWarning.ts +28 -28
  20. package/src/features/activityData/IsAntiCheatExitDetectionDisabledSetter.tsx +19 -19
  21. package/src/features/activityData/IsDirtySetter.tsx +17 -17
  22. package/src/features/activityData/MetaPlayerOptionsSetter.tsx +41 -41
  23. package/src/features/activityData/activityDataSlice.ts +256 -256
  24. package/src/features/activityData/activityJsData.ts +82 -82
  25. package/src/features/activityData/hooks.ts +34 -34
  26. package/src/features/activityData/metaPlayerOptions.ts +23 -23
  27. package/src/features/activityData/uiState.ts +20 -20
  28. package/src/features/activityJS/ActivityJSProvider.tsx +339 -339
  29. package/src/features/activityJS/AfterResetAction.tsx +23 -23
  30. package/src/features/activityJS/AfterSaveAction.tsx +23 -23
  31. package/src/features/activityJS/BeforeResetAction.tsx +23 -23
  32. package/src/features/activityJS/BeforeSaveAction.tsx +23 -23
  33. package/src/features/activityJS/Saver.tsx +167 -167
  34. package/src/features/activityJS/hooks.ts +85 -85
  35. package/src/features/activityJS/internal-hooks.ts +96 -96
  36. package/src/features/activityJS/saverSlice.ts +96 -96
  37. package/src/features/activitySettings/ActivitySettingsSetter.tsx +21 -21
  38. package/src/features/activitySettings/hooks.ts +12 -12
  39. package/src/features/activitySettings/index.tsx +43 -43
  40. package/src/features/activitySettings/store.ts +108 -108
  41. package/src/features/activitySettings/style.module.scss +8 -8
  42. package/src/features/activitySettings/types.ts +140 -140
  43. package/src/features/activitySettings/ui.tsx +299 -299
  44. package/src/features/functionalities/AttachedFilesFunctionality.ts +23 -23
  45. package/src/features/functionalities/PreviewDialog.tsx +83 -83
  46. package/src/features/functionalities/functionalitiesSlice.ts +98 -98
  47. package/src/features/functionalities/hooks.ts +70 -70
  48. package/src/features/layout/hooks.ts +7 -7
  49. package/src/features/layout/layoutSlice.ts +90 -90
  50. package/src/features/navbar/activity-info/index.tsx +54 -54
  51. package/src/features/navbar/activity-info/style.module.scss +38 -38
  52. package/src/features/navbar/activity-menu/ActivityQuickActions.tsx +57 -57
  53. package/src/features/navbar/activity-menu/index.tsx +154 -153
  54. package/src/features/navbar/activity-menu/style.module.scss +7 -7
  55. package/src/features/navbar/capytale-menu/CloneDialog.tsx +75 -75
  56. package/src/features/navbar/capytale-menu/Countdown.tsx +312 -312
  57. package/src/features/navbar/capytale-menu/CountdownAndSaveButton.tsx +115 -115
  58. package/src/features/navbar/capytale-menu/index.tsx +260 -260
  59. package/src/features/navbar/capytale-menu/style.module.scss +8 -8
  60. package/src/features/navbar/index.tsx +39 -39
  61. package/src/features/navbar/navbarSlice.ts +79 -79
  62. package/src/features/navbar/review-navbar/GradingNav.tsx +128 -128
  63. package/src/features/navbar/review-navbar/index.tsx +18 -18
  64. package/src/features/navbar/review-navbar/style.module.scss +22 -22
  65. package/src/features/navbar/sidebars/ActivitySidebarActions.tsx +57 -57
  66. package/src/features/navbar/sidebars/AttachedFilesSidebarContent.module.scss +43 -43
  67. package/src/features/navbar/sidebars/AttachedFilesSidebarContent.tsx +181 -181
  68. package/src/features/navbar/sidebars/SettingsSidebarContent.tsx +192 -192
  69. package/src/features/navbar/sidebars/style.module.scss +15 -15
  70. package/src/features/navbar/student-utils.ts +11 -11
  71. package/src/features/navbar/style.module.scss +65 -65
  72. package/src/features/pedago/InstructionsEditor.tsx +102 -102
  73. package/src/features/pedago/PdfEditor.tsx +91 -91
  74. package/src/features/pedago/PedagoCommands.tsx +353 -353
  75. package/src/features/pedago/SharedNotesEditor.tsx +144 -144
  76. package/src/features/pedago/index.tsx +207 -204
  77. package/src/features/pedago/style.module.scss +233 -233
  78. package/src/features/theming/ThemeSwitcher.tsx +51 -51
  79. package/src/features/theming/hooks.ts +6 -6
  80. package/src/features/theming/themingSlice.ts +93 -93
  81. package/src/features/toast.tsx +38 -38
  82. package/src/hooks/index.ts +16 -16
  83. package/src/index.css +132 -132
  84. package/src/index.ts +90 -90
  85. package/src/logo.svg +1 -1
  86. package/src/my_json_data.js +4146 -4146
  87. package/src/settings.ts +6 -6
  88. package/src/types.ts +9 -9
  89. package/src/utils/ButtonDoubleIcon.tsx +35 -35
  90. package/src/utils/CardSelector.tsx +41 -41
  91. package/src/utils/ErrorBoundary.tsx +41 -41
  92. package/src/utils/PopupButton.tsx +134 -134
  93. package/src/utils/activity.ts +8 -8
  94. package/src/utils/breakpoints.ts +4 -4
  95. package/src/utils/capytale.ts +10 -10
  96. package/src/utils/clipboard.ts +11 -11
  97. package/src/utils/download.ts +7 -7
  98. package/src/utils/equality.ts +32 -32
  99. package/src/utils/server-clock.ts +42 -42
  100. package/src/utils/style.module.scss +45 -45
  101. package/src/utils/useFullscreen.ts +65 -65
  102. package/src/vite-env.d.ts +1 -1
  103. package/tsconfig.json +28 -28
  104. package/tsconfig.node.json +9 -9
  105. package/vite.config.ts +11 -11
@@ -1,102 +1,102 @@
1
- import { useCapytaleRichTextEditor } from "@capytale/capytale-rich-text-editor";
2
- import { useAppDispatch, useAppSelector } from "../../app/hooks";
3
- import {
4
- selectInstructions,
5
- selectInstructionsType,
6
- selectMode,
7
- setCanSaveInstructions,
8
- setInstructionsType,
9
- setIsMPDirty,
10
- setLexicalInstructionsState,
11
- } from "../activityData/activityDataSlice";
12
- import { forwardRef, useImperativeHandle, useMemo, useRef } from "react";
13
- import settings from "../../settings";
14
- import { Button } from "primereact/button";
15
- import { useToast } from "../toast";
16
-
17
- const InstructionsEditor: React.FC = forwardRef((_props, ref) => {
18
- const [Editor, getState, _canSave] = useCapytaleRichTextEditor();
19
- const dispatch = useAppDispatch();
20
- const toast = useToast();
21
- const mode = useAppSelector(selectMode);
22
- const isEditable = mode === "create";
23
- const initialInstructions = useAppSelector(selectInstructions);
24
- const htmlInitialContent = useMemo(() => {
25
- return initialInstructions?.format === "lexical"
26
- ? undefined
27
- : initialInstructions?.htmlValue === null
28
- ? undefined
29
- : (initialInstructions?.htmlValue as string) || ""; // HTML content in value
30
- }, [initialInstructions]);
31
- const initialEditorState = useMemo(() => {
32
- return initialInstructions?.format === "lexical"
33
- ? initialInstructions?.value
34
- : undefined;
35
- }, [initialInstructions]);
36
- const initialStateOnChangeDone = useRef<boolean>(false);
37
- const instructionsType = useAppSelector(selectInstructionsType);
38
-
39
- useImperativeHandle(ref, () => {
40
- return {
41
- save: async () => {
42
- const state = await getState();
43
- dispatch(setLexicalInstructionsState(state.json));
44
- },
45
- };
46
- });
47
-
48
- return (
49
- <>
50
- <>
51
- {instructionsType === "none" && (
52
- <>
53
- <p>Pas de consigne.</p>
54
- {mode === "create" && (
55
- <Button
56
- label="Créer une consigne"
57
- onClick={() => dispatch(setInstructionsType("rich"))}
58
- />
59
- )}
60
- </>
61
- )}
62
- {instructionsType === "rich" && (
63
- <Editor
64
- placeholderText="Écrivez la consigne ici..."
65
- htmlInitialContent={htmlInitialContent}
66
- initialEditorState={initialEditorState}
67
- jsonSizeLimit={settings.STATEMENT_MAX_SIZE}
68
- onChange={(editorState) => {
69
- dispatch(setLexicalInstructionsState(editorState));
70
- if (initialStateOnChangeDone.current) {
71
- dispatch(setIsMPDirty(true));
72
- } else {
73
- initialStateOnChangeDone.current = true;
74
- }
75
- }}
76
- onJsonSizeLimitExceeded={() => {
77
- dispatch(setCanSaveInstructions(false));
78
- toast.show({
79
- summary: "Erreur",
80
- detail: `Le contenu de la consigne est trop volumineux. Veuillez le réduire avant de l'enregistrer.\nPeut-être avez-vous inséré une image trop volumineuse ?`,
81
- severity: "error",
82
- life: 10000,
83
- });
84
- }}
85
- onJsonSizeLimitMet={() => {
86
- dispatch(setCanSaveInstructions(true));
87
- toast.show({
88
- summary: "Succès",
89
- detail: `Le contenu de la consigne ne dépasse plus la taille limite.`,
90
- severity: "success",
91
- life: 4000,
92
- });
93
- }}
94
- isEditable={isEditable}
95
- />
96
- )}
97
- </>
98
- </>
99
- );
100
- });
101
-
102
- export default InstructionsEditor;
1
+ import { useCapytaleRichTextEditor } from "@capytale/capytale-rich-text-editor";
2
+ import { useAppDispatch, useAppSelector } from "../../app/hooks";
3
+ import {
4
+ selectInstructions,
5
+ selectInstructionsType,
6
+ selectMode,
7
+ setCanSaveInstructions,
8
+ setInstructionsType,
9
+ setIsMPDirty,
10
+ setLexicalInstructionsState,
11
+ } from "../activityData/activityDataSlice";
12
+ import { forwardRef, useImperativeHandle, useMemo, useRef } from "react";
13
+ import settings from "../../settings";
14
+ import { Button } from "primereact/button";
15
+ import { useToast } from "../toast";
16
+
17
+ const InstructionsEditor: React.FC = forwardRef((_props, ref) => {
18
+ const [Editor, getState, _canSave] = useCapytaleRichTextEditor();
19
+ const dispatch = useAppDispatch();
20
+ const toast = useToast();
21
+ const mode = useAppSelector(selectMode);
22
+ const isEditable = mode === "create";
23
+ const initialInstructions = useAppSelector(selectInstructions);
24
+ const htmlInitialContent = useMemo(() => {
25
+ return initialInstructions?.format === "lexical"
26
+ ? undefined
27
+ : initialInstructions?.htmlValue === null
28
+ ? undefined
29
+ : (initialInstructions?.htmlValue as string) || ""; // HTML content in value
30
+ }, [initialInstructions]);
31
+ const initialEditorState = useMemo(() => {
32
+ return initialInstructions?.format === "lexical"
33
+ ? initialInstructions?.value
34
+ : undefined;
35
+ }, [initialInstructions]);
36
+ const initialStateOnChangeDone = useRef<boolean>(false);
37
+ const instructionsType = useAppSelector(selectInstructionsType);
38
+
39
+ useImperativeHandle(ref, () => {
40
+ return {
41
+ save: async () => {
42
+ const state = await getState();
43
+ dispatch(setLexicalInstructionsState(state.json));
44
+ },
45
+ };
46
+ });
47
+
48
+ return (
49
+ <>
50
+ <>
51
+ {instructionsType === "none" && (
52
+ <>
53
+ <p>Pas de consigne.</p>
54
+ {mode === "create" && (
55
+ <Button
56
+ label="Créer une consigne"
57
+ onClick={() => dispatch(setInstructionsType("rich"))}
58
+ />
59
+ )}
60
+ </>
61
+ )}
62
+ {instructionsType === "rich" && (
63
+ <Editor
64
+ placeholderText="Écrivez la consigne ici..."
65
+ htmlInitialContent={htmlInitialContent}
66
+ initialEditorState={initialEditorState}
67
+ jsonSizeLimit={settings.STATEMENT_MAX_SIZE}
68
+ onChange={(editorState) => {
69
+ dispatch(setLexicalInstructionsState(editorState));
70
+ if (initialStateOnChangeDone.current) {
71
+ dispatch(setIsMPDirty(true));
72
+ } else {
73
+ initialStateOnChangeDone.current = true;
74
+ }
75
+ }}
76
+ onJsonSizeLimitExceeded={() => {
77
+ dispatch(setCanSaveInstructions(false));
78
+ toast.show({
79
+ summary: "Erreur",
80
+ detail: `Le contenu de la consigne est trop volumineux. Veuillez le réduire avant de l'enregistrer.\nPeut-être avez-vous inséré une image trop volumineuse ?`,
81
+ severity: "error",
82
+ life: 10000,
83
+ });
84
+ }}
85
+ onJsonSizeLimitMet={() => {
86
+ dispatch(setCanSaveInstructions(true));
87
+ toast.show({
88
+ summary: "Succès",
89
+ detail: `Le contenu de la consigne ne dépasse plus la taille limite.`,
90
+ severity: "success",
91
+ life: 4000,
92
+ });
93
+ }}
94
+ isEditable={isEditable}
95
+ />
96
+ )}
97
+ </>
98
+ </>
99
+ );
100
+ });
101
+
102
+ export default InstructionsEditor;
@@ -1,91 +1,91 @@
1
- import { FC, useEffect, useMemo } from "react";
2
- import { useDropzone } from "react-dropzone";
3
- import styles from "./style.module.scss";
4
- import { useAppDispatch, useAppSelector } from "../../app/hooks";
5
- import {
6
- selectMode,
7
- selectPdfInstructions,
8
- setPdfInstructions,
9
- } from "../activityData/activityDataSlice";
10
- import { Button } from "primereact/button";
11
-
12
- export const PdfEditor: FC = () => {
13
- const dispatch = useAppDispatch();
14
- const {
15
- getRootProps,
16
- getInputProps,
17
- isFocused,
18
- isDragAccept,
19
- isDragReject,
20
- acceptedFiles,
21
- } = useDropzone({
22
- accept: {
23
- "application/pdf": [".pdf"],
24
- },
25
- });
26
- useEffect(() => {
27
- if (acceptedFiles.length) {
28
- dispatch(setPdfInstructions(acceptedFiles[0]));
29
- }
30
- }, [acceptedFiles]);
31
- const mode = useAppSelector(selectMode);
32
- const pdfInstructions = useAppSelector(selectPdfInstructions);
33
- const pdfData = useMemo(
34
- () => (pdfInstructions ? URL.createObjectURL(pdfInstructions) : null),
35
- [pdfInstructions],
36
- );
37
- return (
38
- <>
39
- {pdfData && (
40
- <div
41
- style={{
42
- width: "100%",
43
- height: "100%",
44
- position: "relative",
45
- overflow: "hidden",
46
- }}
47
- >
48
- <object
49
- data={pdfData + "#?navpanes=0"}
50
- type="application/pdf"
51
- width="100%"
52
- height="100%"
53
- >
54
- <p>
55
- Votre navigateur ne supporte pas les PDFs, vous pouvez{" "}
56
- <a href={pdfData}>télécharger le fichier</a> à la place.
57
- </p>
58
- </object>
59
- <div className="meta-player-content-cover"></div>
60
- {mode === "create" && (
61
- <Button
62
- label="Supprimer le PDF"
63
- severity="danger"
64
- icon="pi pi-trash"
65
- className={styles.deleteButton}
66
- onClick={() => dispatch(setPdfInstructions(null))}
67
- style={{ position: "absolute", bottom: "20px", right: "20px" }}
68
- />
69
- )}
70
- </div>
71
- )}
72
- {!pdfData && (
73
- <div className={styles.dropzone} {...getRootProps()}>
74
- <div
75
- className={styles.dropzoneRectangle}
76
- data-is-focused={isFocused}
77
- data-is-drag-accept={isDragAccept}
78
- data-is-drag-reject={isDragReject}
79
- >
80
- <input {...getInputProps()} />
81
- <p>Déposez un PDF ici ou cliquez pour sélectionner un fichier...</p>
82
- <i
83
- className="pi pi-file-arrow-up"
84
- style={{ fontSize: "2.5rem" }}
85
- ></i>
86
- </div>
87
- </div>
88
- )}
89
- </>
90
- );
91
- };
1
+ import { FC, useEffect, useMemo } from "react";
2
+ import { useDropzone } from "react-dropzone";
3
+ import styles from "./style.module.scss";
4
+ import { useAppDispatch, useAppSelector } from "../../app/hooks";
5
+ import {
6
+ selectMode,
7
+ selectPdfInstructions,
8
+ setPdfInstructions,
9
+ } from "../activityData/activityDataSlice";
10
+ import { Button } from "primereact/button";
11
+
12
+ export const PdfEditor: FC = () => {
13
+ const dispatch = useAppDispatch();
14
+ const {
15
+ getRootProps,
16
+ getInputProps,
17
+ isFocused,
18
+ isDragAccept,
19
+ isDragReject,
20
+ acceptedFiles,
21
+ } = useDropzone({
22
+ accept: {
23
+ "application/pdf": [".pdf"],
24
+ },
25
+ });
26
+ useEffect(() => {
27
+ if (acceptedFiles.length) {
28
+ dispatch(setPdfInstructions(acceptedFiles[0]));
29
+ }
30
+ }, [acceptedFiles]);
31
+ const mode = useAppSelector(selectMode);
32
+ const pdfInstructions = useAppSelector(selectPdfInstructions);
33
+ const pdfData = useMemo(
34
+ () => (pdfInstructions ? URL.createObjectURL(pdfInstructions) : null),
35
+ [pdfInstructions],
36
+ );
37
+ return (
38
+ <>
39
+ {pdfData && (
40
+ <div
41
+ style={{
42
+ width: "100%",
43
+ height: "100%",
44
+ position: "relative",
45
+ overflow: "hidden",
46
+ }}
47
+ >
48
+ <object
49
+ data={pdfData + "#?navpanes=0"}
50
+ type="application/pdf"
51
+ width="100%"
52
+ height="100%"
53
+ >
54
+ <p>
55
+ Votre navigateur ne supporte pas les PDFs, vous pouvez{" "}
56
+ <a href={pdfData}>télécharger le fichier</a> à la place.
57
+ </p>
58
+ </object>
59
+ <div className="meta-player-content-cover"></div>
60
+ {mode === "create" && (
61
+ <Button
62
+ label="Supprimer le PDF"
63
+ severity="danger"
64
+ icon="pi pi-trash"
65
+ className={styles.deleteButton}
66
+ onClick={() => dispatch(setPdfInstructions(null))}
67
+ style={{ position: "absolute", bottom: "20px", right: "20px" }}
68
+ />
69
+ )}
70
+ </div>
71
+ )}
72
+ {!pdfData && (
73
+ <div className={styles.dropzone} {...getRootProps()}>
74
+ <div
75
+ className={styles.dropzoneRectangle}
76
+ data-is-focused={isFocused}
77
+ data-is-drag-accept={isDragAccept}
78
+ data-is-drag-reject={isDragReject}
79
+ >
80
+ <input {...getInputProps()} />
81
+ <p>Déposez un PDF ici ou cliquez pour sélectionner un fichier...</p>
82
+ <i
83
+ className="pi pi-file-arrow-up"
84
+ style={{ fontSize: "2.5rem" }}
85
+ ></i>
86
+ </div>
87
+ </div>
88
+ )}
89
+ </>
90
+ );
91
+ };