@pie-element/hotspot 11.1.2-next.2 → 11.1.2

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 (234) hide show
  1. package/CHANGELOG.json +997 -0
  2. package/CHANGELOG.md +2220 -0
  3. package/LICENSE.md +5 -0
  4. package/README.md +1 -0
  5. package/configure/CHANGELOG.json +682 -0
  6. package/configure/CHANGELOG.md +1957 -0
  7. package/configure/lib/DeleteWidget.js +64 -0
  8. package/configure/lib/DeleteWidget.js.map +1 -0
  9. package/configure/lib/button.js +42 -0
  10. package/configure/lib/button.js.map +1 -0
  11. package/configure/lib/buttons/circle.js +33 -0
  12. package/configure/lib/buttons/circle.js.map +1 -0
  13. package/configure/lib/buttons/polygon.js +39 -0
  14. package/configure/lib/buttons/polygon.js.map +1 -0
  15. package/configure/lib/buttons/rectangle.js +39 -0
  16. package/configure/lib/buttons/rectangle.js.map +1 -0
  17. package/configure/lib/defaults.js +155 -0
  18. package/configure/lib/defaults.js.map +1 -0
  19. package/configure/lib/hotspot-circle.js +192 -0
  20. package/configure/lib/hotspot-circle.js.map +1 -0
  21. package/configure/lib/hotspot-container.js +320 -0
  22. package/configure/lib/hotspot-container.js.map +1 -0
  23. package/configure/lib/hotspot-drawable.js +519 -0
  24. package/configure/lib/hotspot-drawable.js.map +1 -0
  25. package/configure/lib/hotspot-palette.js +107 -0
  26. package/configure/lib/hotspot-palette.js.map +1 -0
  27. package/configure/lib/hotspot-polygon.js +293 -0
  28. package/configure/lib/hotspot-polygon.js.map +1 -0
  29. package/configure/lib/hotspot-rectangle.js +190 -0
  30. package/configure/lib/hotspot-rectangle.js.map +1 -0
  31. package/configure/lib/icons.js +7 -0
  32. package/configure/lib/icons.js.map +1 -0
  33. package/configure/lib/image-konva.js +66 -0
  34. package/configure/lib/image-konva.js.map +1 -0
  35. package/configure/lib/index.js +194 -0
  36. package/configure/lib/index.js.map +1 -0
  37. package/configure/lib/root.js +330 -0
  38. package/configure/lib/root.js.map +1 -0
  39. package/configure/lib/shapes/circle.js +84 -0
  40. package/configure/lib/shapes/circle.js.map +1 -0
  41. package/configure/lib/shapes/index.js +50 -0
  42. package/configure/lib/shapes/index.js.map +1 -0
  43. package/configure/lib/shapes/polygon.js +82 -0
  44. package/configure/lib/shapes/polygon.js.map +1 -0
  45. package/configure/lib/shapes/rectagle.js +84 -0
  46. package/configure/lib/shapes/rectagle.js.map +1 -0
  47. package/configure/lib/shapes/utils.js +21 -0
  48. package/configure/lib/shapes/utils.js.map +1 -0
  49. package/configure/lib/upload-control.js +41 -0
  50. package/configure/lib/upload-control.js.map +1 -0
  51. package/configure/lib/utils.js +185 -0
  52. package/configure/lib/utils.js.map +1 -0
  53. package/configure/package.json +26 -0
  54. package/configure/src/DeleteWidget.jsx +51 -0
  55. package/configure/src/__tests__/DeleteWidget.test.jsx +366 -0
  56. package/configure/src/__tests__/button.test.jsx +198 -0
  57. package/configure/src/__tests__/hotspot-circle.test.jsx +259 -0
  58. package/configure/src/__tests__/hotspot-container.test.js +366 -0
  59. package/configure/src/__tests__/hotspot-drawable.test.js +271 -0
  60. package/configure/src/__tests__/hotspot-palette.test.jsx +71 -0
  61. package/configure/src/__tests__/image-konva.test.jsx +226 -0
  62. package/configure/src/__tests__/index.test.js +329 -0
  63. package/configure/src/__tests__/root.test.js +400 -0
  64. package/configure/src/__tests__/utils.test.js +241 -0
  65. package/configure/src/button.jsx +35 -0
  66. package/configure/src/buttons/circle.jsx +18 -0
  67. package/configure/src/buttons/polygon.jsx +29 -0
  68. package/configure/src/buttons/rectangle.jsx +29 -0
  69. package/configure/src/defaults.js +109 -0
  70. package/configure/src/hotspot-circle.jsx +183 -0
  71. package/configure/src/hotspot-container.jsx +330 -0
  72. package/configure/src/hotspot-drawable.jsx +527 -0
  73. package/configure/src/hotspot-palette.jsx +90 -0
  74. package/configure/src/hotspot-polygon.jsx +294 -0
  75. package/configure/src/hotspot-rectangle.jsx +169 -0
  76. package/configure/src/icons.js +5 -0
  77. package/configure/src/image-konva.jsx +63 -0
  78. package/configure/src/index.js +208 -0
  79. package/configure/src/root.jsx +346 -0
  80. package/configure/src/shapes/circle.js +81 -0
  81. package/configure/src/shapes/index.js +4 -0
  82. package/configure/src/shapes/polygon.js +81 -0
  83. package/configure/src/shapes/rectagle.js +82 -0
  84. package/configure/src/shapes/utils.js +16 -0
  85. package/configure/src/upload-control.jsx +33 -0
  86. package/configure/src/utils.js +210 -0
  87. package/controller/CHANGELOG.json +362 -0
  88. package/controller/CHANGELOG.md +1304 -0
  89. package/controller/lib/defaults.js +33 -0
  90. package/controller/lib/defaults.js.map +1 -0
  91. package/controller/lib/index.js +341 -0
  92. package/controller/lib/index.js.map +1 -0
  93. package/controller/lib/utils.js +32 -0
  94. package/controller/lib/utils.js.map +1 -0
  95. package/controller/package.json +18 -0
  96. package/controller/src/__tests__/index.test.js +419 -0
  97. package/controller/src/__tests__/utils.test.js +5 -0
  98. package/controller/src/defaults.js +19 -0
  99. package/controller/src/index.js +328 -0
  100. package/controller/src/utils.js +29 -0
  101. package/docs/config-schema.json +2023 -0
  102. package/docs/config-schema.json.md +1495 -0
  103. package/docs/demo/config.js +8 -0
  104. package/docs/demo/generate.js +118 -0
  105. package/docs/demo/index.html +1 -0
  106. package/docs/demo/session.js +11 -0
  107. package/docs/pie-schema.json +1204 -0
  108. package/docs/pie-schema.json.md +851 -0
  109. package/lib/hotspot/circle.js +156 -0
  110. package/lib/hotspot/circle.js.map +1 -0
  111. package/lib/hotspot/container.js +206 -0
  112. package/lib/hotspot/container.js.map +1 -0
  113. package/lib/hotspot/icons.js +8 -0
  114. package/lib/hotspot/icons.js.map +1 -0
  115. package/lib/hotspot/image-konva-tooltip.js +86 -0
  116. package/lib/hotspot/image-konva-tooltip.js.map +1 -0
  117. package/lib/hotspot/index.js +163 -0
  118. package/lib/hotspot/index.js.map +1 -0
  119. package/lib/hotspot/polygon.js +203 -0
  120. package/lib/hotspot/polygon.js.map +1 -0
  121. package/lib/hotspot/rectangle.js +175 -0
  122. package/lib/hotspot/rectangle.js.map +1 -0
  123. package/lib/index.js +213 -0
  124. package/lib/index.js.map +1 -0
  125. package/lib/session-updater.js +42 -0
  126. package/lib/session-updater.js.map +1 -0
  127. package/package.json +18 -83
  128. package/src/__tests__/container.test.jsx +58 -0
  129. package/src/__tests__/index.test.js +123 -0
  130. package/src/__tests__/session-updater.test.jsx +69 -0
  131. package/src/hotspot/__tests__/circle.test.jsx +464 -0
  132. package/src/hotspot/__tests__/container.test.jsx +546 -0
  133. package/src/hotspot/__tests__/image-konva-tooltip.test.jsx +510 -0
  134. package/src/hotspot/__tests__/polygon.test.jsx +502 -0
  135. package/src/hotspot/__tests__/rectangle.test.jsx +418 -0
  136. package/src/hotspot/circle.jsx +152 -0
  137. package/src/hotspot/container.jsx +217 -0
  138. package/src/hotspot/icons.js +7 -0
  139. package/src/hotspot/image-konva-tooltip.jsx +76 -0
  140. package/src/hotspot/index.jsx +165 -0
  141. package/src/hotspot/polygon.jsx +195 -0
  142. package/src/hotspot/rectangle.jsx +171 -0
  143. package/src/index.js +226 -0
  144. package/src/session-updater.js +29 -0
  145. package/configure.js +0 -2
  146. package/controller.js +0 -1
  147. package/dist/author/DeleteWidget.d.ts +0 -38
  148. package/dist/author/DeleteWidget.js +0 -46
  149. package/dist/author/button.d.ts +0 -31
  150. package/dist/author/button.js +0 -27
  151. package/dist/author/buttons/circle.d.ts +0 -18
  152. package/dist/author/buttons/circle.js +0 -25
  153. package/dist/author/buttons/polygon.d.ts +0 -18
  154. package/dist/author/buttons/polygon.js +0 -36
  155. package/dist/author/buttons/rectangle.d.ts +0 -18
  156. package/dist/author/buttons/rectangle.js +0 -36
  157. package/dist/author/defaults.d.ts +0 -157
  158. package/dist/author/defaults.js +0 -119
  159. package/dist/author/hotspot-circle.d.ts +0 -21
  160. package/dist/author/hotspot-circle.js +0 -124
  161. package/dist/author/hotspot-container.d.ts +0 -29
  162. package/dist/author/hotspot-container.js +0 -210
  163. package/dist/author/hotspot-drawable.d.ts +0 -31
  164. package/dist/author/hotspot-drawable.js +0 -312
  165. package/dist/author/hotspot-palette.d.ts +0 -14
  166. package/dist/author/hotspot-palette.js +0 -72
  167. package/dist/author/hotspot-polygon.d.ts +0 -38
  168. package/dist/author/hotspot-polygon.js +0 -200
  169. package/dist/author/hotspot-rectangle.d.ts +0 -20
  170. package/dist/author/hotspot-rectangle.js +0 -119
  171. package/dist/author/icons.d.ts +0 -9
  172. package/dist/author/icons.js +0 -4
  173. package/dist/author/image-konva.d.ts +0 -19
  174. package/dist/author/image-konva.js +0 -49
  175. package/dist/author/index.d.ts +0 -52
  176. package/dist/author/index.js +0 -143
  177. package/dist/author/root.d.ts +0 -15
  178. package/dist/author/root.js +0 -215
  179. package/dist/author/shapes/circle.d.ts +0 -18
  180. package/dist/author/shapes/circle.js +0 -47
  181. package/dist/author/shapes/index.d.ts +0 -12
  182. package/dist/author/shapes/polygon.d.ts +0 -19
  183. package/dist/author/shapes/polygon.js +0 -51
  184. package/dist/author/shapes/rectagle.d.ts +0 -18
  185. package/dist/author/shapes/rectagle.js +0 -57
  186. package/dist/author/shapes/utils.d.ts +0 -19
  187. package/dist/author/shapes/utils.js +0 -16
  188. package/dist/author/upload-control.d.ts +0 -29
  189. package/dist/author/upload-control.js +0 -28
  190. package/dist/author/utils.d.ts +0 -24
  191. package/dist/author/utils.js +0 -83
  192. package/dist/browser/ReactKonva-DI5WIo8o.js +0 -19336
  193. package/dist/browser/ReactKonva-DI5WIo8o.js.map +0 -1
  194. package/dist/browser/author/index.js +0 -41646
  195. package/dist/browser/author/index.js.map +0 -1
  196. package/dist/browser/browser-CfnAFove.js +0 -219
  197. package/dist/browser/browser-CfnAFove.js.map +0 -1
  198. package/dist/browser/controller/index.js +0 -198
  199. package/dist/browser/controller/index.js.map +0 -1
  200. package/dist/browser/delivery/index.js +0 -2460
  201. package/dist/browser/delivery/index.js.map +0 -1
  202. package/dist/browser/dist-C78LDz6R.js +0 -96
  203. package/dist/browser/dist-C78LDz6R.js.map +0 -1
  204. package/dist/browser/hotspot.css +0 -2
  205. package/dist/controller/defaults.d.ts +0 -35
  206. package/dist/controller/defaults.js +0 -29
  207. package/dist/controller/index.d.ts +0 -22
  208. package/dist/controller/index.js +0 -154
  209. package/dist/controller/utils.d.ts +0 -10
  210. package/dist/controller/utils.js +0 -12
  211. package/dist/delivery/hotspot/circle.d.ts +0 -19
  212. package/dist/delivery/hotspot/circle.js +0 -100
  213. package/dist/delivery/hotspot/container.d.ts +0 -16
  214. package/dist/delivery/hotspot/container.js +0 -150
  215. package/dist/delivery/hotspot/icons.d.ts +0 -10
  216. package/dist/delivery/hotspot/icons.js +0 -4
  217. package/dist/delivery/hotspot/image-konva-tooltip.d.ts +0 -19
  218. package/dist/delivery/hotspot/image-konva-tooltip.js +0 -66
  219. package/dist/delivery/hotspot/index.d.ts +0 -17
  220. package/dist/delivery/hotspot/index.js +0 -114
  221. package/dist/delivery/hotspot/polygon.d.ts +0 -21
  222. package/dist/delivery/hotspot/polygon.js +0 -108
  223. package/dist/delivery/hotspot/rectangle.d.ts +0 -19
  224. package/dist/delivery/hotspot/rectangle.js +0 -104
  225. package/dist/delivery/index.d.ts +0 -20
  226. package/dist/delivery/index.js +0 -107
  227. package/dist/delivery/session-updater.d.ts +0 -10
  228. package/dist/delivery/session-updater.js +0 -14
  229. package/dist/index.d.ts +0 -1
  230. package/dist/index.iife.d.ts +0 -8
  231. package/dist/index.iife.js +0 -169
  232. package/dist/index.js +0 -2
  233. package/dist/runtime-support.d.ts +0 -12
  234. package/dist/runtime-support.js +0 -12
@@ -0,0 +1,328 @@
1
+ import debug from 'debug';
2
+ import { isEmpty } from 'lodash-es';
3
+ import { partialScoring } from '@pie-lib/controller-utils';
4
+
5
+ import { isResponseCorrect } from './utils';
6
+
7
+ import defaults from './defaults';
8
+
9
+ const log = debug('pie-elements:hotspot:controller');
10
+
11
+ export const normalize = (question) => ({
12
+ ...defaults,
13
+ ...question,
14
+ });
15
+
16
+ export function model(question, session, env) {
17
+ const normalizedQuestion = normalize(question);
18
+ const {
19
+ imageUrl,
20
+ dimensions,
21
+ hotspotColor,
22
+ hoverOutlineColor,
23
+ selectedHotspotColor,
24
+ multipleCorrect,
25
+ outlineColor,
26
+ partialScoring,
27
+ prompt,
28
+ shapes,
29
+ language,
30
+ fontSizeFactor,
31
+ autoplayAudioEnabled,
32
+ completeAudioEnabled,
33
+ customAudioButton,
34
+ } = normalizedQuestion;
35
+ const { rectangles, polygons, circles } = shapes || {};
36
+
37
+ const shouldIncludeCorrectResponse = env.mode === 'evaluate' || (env.role === 'instructor' && env.mode === 'view');
38
+
39
+ return new Promise((resolve) => {
40
+ const out = {
41
+ disabled: env.mode !== 'gather',
42
+ mode: env.mode,
43
+ dimensions,
44
+ imageUrl,
45
+ outlineColor,
46
+ hotspotColor,
47
+ hoverOutlineColor,
48
+ selectedHotspotColor,
49
+ multipleCorrect,
50
+ partialScoring,
51
+ language,
52
+ fontSizeFactor,
53
+ autoplayAudioEnabled,
54
+ completeAudioEnabled,
55
+ customAudioButton,
56
+ shapes: {
57
+ ...shapes,
58
+ // eslint-disable-next-line no-unused-vars
59
+ rectangles: (rectangles || []).map(({ index, correct, ...rectProps }) =>
60
+ shouldIncludeCorrectResponse ? { correct, ...rectProps } : { ...rectProps },
61
+ ),
62
+ // eslint-disable-next-line no-unused-vars
63
+ polygons: (polygons || []).map(({ index, correct, ...polyProps }) =>
64
+ shouldIncludeCorrectResponse ? { correct, ...polyProps } : { ...polyProps },
65
+ ),
66
+ // eslint-disable-next-line no-unused-vars
67
+ circles: (circles || []).map(({ index, correct, ...circleProps }) =>
68
+ shouldIncludeCorrectResponse ? { correct, ...circleProps } : { ...circleProps },
69
+ ),
70
+ },
71
+ responseCorrect: env.mode === 'evaluate' ? isResponseCorrect(normalizedQuestion, session) : undefined,
72
+ extraCSSRules: normalizedQuestion.extraCSSRules,
73
+ };
74
+
75
+ if (env.role === 'instructor' && (env.mode === 'view' || env.mode === 'evaluate')) {
76
+ out.rationale = normalizedQuestion.rationaleEnabled ? normalizedQuestion.rationale : null;
77
+ out.teacherInstructions = normalizedQuestion.teacherInstructionsEnabled
78
+ ? normalizedQuestion.teacherInstructions
79
+ : null;
80
+ } else {
81
+ out.rationale = null;
82
+ out.teacherInstructions = null;
83
+ }
84
+
85
+ out.prompt = normalizedQuestion.promptEnabled ? prompt : null;
86
+ out.strokeWidth = normalizedQuestion.strokeWidth;
87
+
88
+ resolve(out);
89
+ });
90
+ }
91
+
92
+ export const createDefaultModel = (model = {}) =>
93
+ new Promise((resolve) => {
94
+ resolve({
95
+ ...defaults,
96
+ ...model,
97
+ });
98
+ });
99
+
100
+ const getScore = (config, session, env = {}) => {
101
+ const { answers } = session || {};
102
+
103
+ if (!config.shapes || (!config.shapes.rectangles && !config.shapes.polygons && !config.shapes.circles)) {
104
+ return 0;
105
+ }
106
+
107
+ const { shapes: { rectangles = [], polygons = [], circles = [] } = {} } = config;
108
+ const partialScoringEnabled = partialScoring.enabled(config, env);
109
+
110
+ if (!partialScoringEnabled) {
111
+ return isResponseCorrect(config, session) ? 1 : 0;
112
+ }
113
+
114
+ let correctAnswers = 0;
115
+ let selectedChoices = 0;
116
+
117
+ const choices = [...rectangles, ...polygons, ...circles];
118
+
119
+ const correctChoices = choices.filter((choice) => choice.correct);
120
+
121
+ choices.forEach((shape) => {
122
+ const selected = answers && answers.filter((answer) => answer.id === shape.id)[0];
123
+ const correctlySelected = shape.correct && selected;
124
+
125
+ if (selected) {
126
+ selectedChoices += 1;
127
+ }
128
+
129
+ if (correctlySelected) {
130
+ correctAnswers += 1;
131
+ }
132
+ });
133
+
134
+ const extraAnswers = selectedChoices > correctChoices.length ? selectedChoices - correctChoices.length : 0;
135
+
136
+ const total = correctChoices.length === 0 ? 1 : correctChoices.length;
137
+ const str = ((correctAnswers - extraAnswers) / total).toFixed(2);
138
+
139
+ return str < 0 ? 0 : parseFloat(str);
140
+ };
141
+
142
+ /**
143
+ * Generates detailed trace log for scoring evaluation
144
+ * @param {Object} model - the question model
145
+ * @param {Object} session - the student session
146
+ * @param {Object} env - the environment
147
+ * @returns {Array} traceLog - array of trace messages
148
+ */
149
+ export const getLogTrace = (model, session, env) => {
150
+ const traceLog = [];
151
+ const { answers } = session || {};
152
+ const { shapes } = model || {};
153
+
154
+ const allShapes = [];
155
+ if (shapes) {
156
+ if (shapes.rectangles) allShapes.push(...shapes.rectangles);
157
+ if (shapes.polygons) allShapes.push(...shapes.polygons);
158
+ if (shapes.circles) allShapes.push(...shapes.circles);
159
+ }
160
+
161
+ const correctShapes = allShapes.filter(shape => shape.correct);
162
+ const totalShapes = allShapes.length;
163
+
164
+ traceLog.push(`Total of ${totalShapes} hotspot(s) defined, ${correctShapes.length} correct.`);
165
+
166
+ if (answers && answers.length > 0) {
167
+ traceLog.push(`Student selected ${answers.length} hotspot(s).`);
168
+
169
+ let correctSelections = 0;
170
+ let incorrectSelections = 0;
171
+
172
+ answers.forEach(answer => {
173
+ const shape = allShapes.find(s => s.id === answer.id);
174
+ if (shape && shape.correct) {
175
+ correctSelections++;
176
+ } else {
177
+ incorrectSelections++;
178
+ }
179
+ });
180
+
181
+ const missedCorrect = correctShapes.filter(correctShape =>
182
+ !answers.some(answer => answer.id === correctShape.id)
183
+ ).length;
184
+
185
+ if (correctSelections > 0) {
186
+ traceLog.push(`${correctSelections} correct hotspot(s) selected.`);
187
+ }
188
+ if (incorrectSelections > 0) {
189
+ traceLog.push(`${incorrectSelections} incorrect hotspot(s) selected.`);
190
+ }
191
+ if (missedCorrect > 0) {
192
+ traceLog.push(`${missedCorrect} correct hotspot(s) missed.`);
193
+ }
194
+ } else {
195
+ traceLog.push('No hotspots selected.');
196
+ }
197
+
198
+ const partialScoringEnabled = partialScoring.enabled(model, env);
199
+
200
+ if (partialScoringEnabled) {
201
+ traceLog.push(`Score calculated using partial scoring.`);
202
+
203
+ if (answers && answers.length > 0) {
204
+ let correctSelections = 0;
205
+ let incorrectSelections = 0;
206
+
207
+ answers.forEach(answer => {
208
+ const shape = allShapes.find(s => s.id === answer.id);
209
+ if (shape && shape.correct) {
210
+ correctSelections++;
211
+ } else {
212
+ incorrectSelections++;
213
+ }
214
+ });
215
+
216
+ const totalCorrectAvailable = correctShapes.length;
217
+ traceLog.push(`Partial scoring calculation: ${correctSelections} correct selections out of ${totalCorrectAvailable} available.`);
218
+
219
+ if (incorrectSelections > totalCorrectAvailable) {
220
+ const extraSelections = incorrectSelections - (totalCorrectAvailable - correctSelections);
221
+ traceLog.push(`${extraSelections} extra incorrect selection(s) beyond required amount are deducted from score.`);
222
+ }
223
+ }
224
+ } else {
225
+ traceLog.push(`Score calculated using all-or-nothing scoring.`);
226
+ }
227
+
228
+ const score = getScore(model, session, env);
229
+ traceLog.push(`Score: ${score}.`);
230
+
231
+ return traceLog;
232
+ }
233
+
234
+ export function outcome(config, session, env = {}) {
235
+ return new Promise((resolve) => {
236
+ log('outcome...');
237
+
238
+ if (!session || isEmpty(session)) {
239
+ resolve({
240
+ score: 0,
241
+ empty: true,
242
+ traceLog: ['No hotspots selected. Score: 0.']
243
+ });
244
+ }
245
+
246
+ if (session.answers) {
247
+ const traceLog = getLogTrace(config, session, env);
248
+ const score = getScore(config, session, env);
249
+
250
+ resolve({ score, empty: false, traceLog });
251
+ } else {
252
+ resolve({
253
+ score: 0,
254
+ empty: true,
255
+ traceLog: ['No hotspots selected. Score: 0.']
256
+ });
257
+ }
258
+ });
259
+ }
260
+
261
+ const returnShapesCorrect = (shapes) => {
262
+ let answers = [];
263
+
264
+ shapes.forEach((i) => {
265
+ const { correct, id } = i;
266
+ if (correct) {
267
+ answers.push({ id });
268
+ }
269
+ });
270
+ return answers;
271
+ };
272
+
273
+ export const createCorrectResponseSession = (question, env) => {
274
+ return new Promise((resolve) => {
275
+ if (env.mode !== 'evaluate' && env.role === 'instructor') {
276
+ const { shapes: { rectangles = [], circles = [], polygons = {} } = {} } = question;
277
+
278
+ const rectangleCorrect = returnShapesCorrect(rectangles);
279
+ const polygonsCorrect = returnShapesCorrect(polygons);
280
+ const circlesCorrect = returnShapesCorrect(circles);
281
+
282
+ resolve({
283
+ answers: [...rectangleCorrect, ...polygonsCorrect, ...circlesCorrect],
284
+ id: '1',
285
+ });
286
+ } else {
287
+ resolve(null);
288
+ }
289
+ });
290
+ };
291
+
292
+ // remove all html tags
293
+ const getInnerText = (html) => (html || '').replaceAll(/<[^>]*>/g, '');
294
+
295
+ // remove all html tags except img, iframe and source tag for audio
296
+ const getContent = (html) => (html || '').replace(/(<(?!img|iframe|source)([^>]+)>)/gi, '');
297
+
298
+ export const validate = (model = {}, config = {}) => {
299
+ const { shapes } = model;
300
+ const { minShapes = 2, maxShapes, maxSelections } = config;
301
+ const errors = {};
302
+
303
+ ['teacherInstructions', 'prompt', 'rationale'].forEach((field) => {
304
+ if (config[field]?.required && !getContent(model[field])) {
305
+ errors[field] = 'This field is required.';
306
+ }
307
+ });
308
+
309
+ const allShapes = Object.values(shapes || {}).reduce((acc, shape) => [...acc, ...shape], []);
310
+
311
+ const nbOfSelections = (allShapes || []).reduce((acc, shape) => (shape.correct ? acc + 1 : acc), 0);
312
+
313
+ const nbOfShapes = (allShapes || []).length;
314
+
315
+ if (nbOfShapes < minShapes) {
316
+ errors.shapes = `There should be at least ${minShapes} shapes defined.`;
317
+ } else if (nbOfShapes > maxShapes) {
318
+ errors.shapes = `No more than ${maxShapes} shapes should be defined.`;
319
+ }
320
+
321
+ if (nbOfSelections < 1) {
322
+ errors.selections = 'There should be at least 1 shape selected.';
323
+ } else if (nbOfSelections > maxSelections) {
324
+ errors.selections = `No more than ${maxSelections} shapes should be selected.`;
325
+ }
326
+
327
+ return errors;
328
+ };
@@ -0,0 +1,29 @@
1
+ import { isEmpty, isEqual } from 'lodash-es';
2
+
3
+ export const getCorrectResponse = (choices) =>
4
+ choices
5
+ .filter((c) => c.correct)
6
+ .map((c) => c.id)
7
+ .sort();
8
+
9
+ export const isResponseCorrect = (question, session) => {
10
+ const {
11
+ shapes: { rectangles = [], polygons = [], circles = [] },
12
+ } = question;
13
+ const choices = [...rectangles, ...polygons, ...circles];
14
+ let correctResponseIds = getCorrectResponse(choices);
15
+
16
+ if (!session || isEmpty(session)) {
17
+ return false;
18
+ }
19
+
20
+ if (session.answers && session.answers.length) {
21
+ let answerIds = (session.answers || []).map((a) => a.id);
22
+
23
+ return isEqual(answerIds.sort(), correctResponseIds);
24
+ } else if (!(correctResponseIds && correctResponseIds.length)) {
25
+ return true;
26
+ }
27
+
28
+ return false;
29
+ };