@longsightgroup/qti3-player 0.4.0 → 0.5.4

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 (291) hide show
  1. package/README.md +146 -13
  2. package/dist/content/content-dom.d.ts +11 -0
  3. package/dist/content/content-dom.d.ts.map +1 -0
  4. package/dist/content/content-dom.js +262 -0
  5. package/dist/content/content-dom.js.map +1 -0
  6. package/dist/content/content-renderer.d.ts +17 -0
  7. package/dist/content/content-renderer.d.ts.map +1 -0
  8. package/dist/content/content-renderer.js +82 -0
  9. package/dist/content/content-renderer.js.map +1 -0
  10. package/dist/controls/remove-button.d.ts +3 -0
  11. package/dist/controls/remove-button.d.ts.map +1 -0
  12. package/dist/controls/remove-button.js +12 -0
  13. package/dist/controls/remove-button.js.map +1 -0
  14. package/dist/index.d.ts +13 -2
  15. package/dist/index.d.ts.map +1 -1
  16. package/dist/index.js +6 -0
  17. package/dist/index.js.map +1 -1
  18. package/dist/interactions/choice-interaction.d.ts +4 -0
  19. package/dist/interactions/choice-interaction.d.ts.map +1 -0
  20. package/dist/interactions/choice-interaction.js +81 -0
  21. package/dist/interactions/choice-interaction.js.map +1 -0
  22. package/dist/interactions/drawing-interaction.d.ts +6 -0
  23. package/dist/interactions/drawing-interaction.d.ts.map +1 -0
  24. package/dist/interactions/drawing-interaction.js +401 -0
  25. package/dist/interactions/drawing-interaction.js.map +1 -0
  26. package/dist/interactions/end-attempt-interaction.d.ts +4 -0
  27. package/dist/interactions/end-attempt-interaction.d.ts.map +1 -0
  28. package/dist/interactions/end-attempt-interaction.js +13 -0
  29. package/dist/interactions/end-attempt-interaction.js.map +1 -0
  30. package/dist/interactions/gap-match-interaction.d.ts +4 -0
  31. package/dist/interactions/gap-match-interaction.d.ts.map +1 -0
  32. package/dist/interactions/gap-match-interaction.js +277 -0
  33. package/dist/interactions/gap-match-interaction.js.map +1 -0
  34. package/dist/interactions/graphic-associate-interaction.d.ts +4 -0
  35. package/dist/interactions/graphic-associate-interaction.d.ts.map +1 -0
  36. package/dist/interactions/graphic-associate-interaction.js +297 -0
  37. package/dist/interactions/graphic-associate-interaction.js.map +1 -0
  38. package/dist/interactions/graphic-context.d.ts +3 -0
  39. package/dist/interactions/graphic-context.d.ts.map +1 -0
  40. package/dist/interactions/graphic-context.js +35 -0
  41. package/dist/interactions/graphic-context.js.map +1 -0
  42. package/dist/interactions/hotspot-interaction.d.ts +4 -0
  43. package/dist/interactions/hotspot-interaction.d.ts.map +1 -0
  44. package/dist/interactions/hotspot-interaction.js +68 -0
  45. package/dist/interactions/hotspot-interaction.js.map +1 -0
  46. package/dist/interactions/hottext-interaction.d.ts +3 -0
  47. package/dist/interactions/hottext-interaction.d.ts.map +1 -0
  48. package/dist/interactions/hottext-interaction.js +66 -0
  49. package/dist/interactions/hottext-interaction.js.map +1 -0
  50. package/dist/interactions/inline-choice-interaction.d.ts +4 -0
  51. package/dist/interactions/inline-choice-interaction.d.ts.map +1 -0
  52. package/dist/interactions/inline-choice-interaction.js +31 -0
  53. package/dist/interactions/inline-choice-interaction.js.map +1 -0
  54. package/dist/interactions/inline-controls.d.ts +6 -0
  55. package/dist/interactions/inline-controls.d.ts.map +1 -0
  56. package/dist/interactions/inline-controls.js +15 -0
  57. package/dist/interactions/inline-controls.js.map +1 -0
  58. package/dist/interactions/interaction-diagnostics.d.ts +7 -0
  59. package/dist/interactions/interaction-diagnostics.d.ts.map +1 -0
  60. package/dist/interactions/interaction-diagnostics.js +137 -0
  61. package/dist/interactions/interaction-diagnostics.js.map +1 -0
  62. package/dist/interactions/interaction-dispatch.d.ts +2 -0
  63. package/dist/interactions/interaction-dispatch.d.ts.map +1 -0
  64. package/dist/interactions/interaction-dispatch.js +2 -0
  65. package/dist/interactions/interaction-dispatch.js.map +1 -0
  66. package/dist/interactions/interaction-label.d.ts +4 -0
  67. package/dist/interactions/interaction-label.d.ts.map +1 -0
  68. package/dist/interactions/interaction-label.js +8 -0
  69. package/dist/interactions/interaction-label.js.map +1 -0
  70. package/dist/interactions/interaction-registry.d.ts +24 -0
  71. package/dist/interactions/interaction-registry.d.ts.map +1 -0
  72. package/dist/interactions/interaction-registry.js +138 -0
  73. package/dist/interactions/interaction-registry.js.map +1 -0
  74. package/dist/interactions/match-interaction.d.ts +4 -0
  75. package/dist/interactions/match-interaction.d.ts.map +1 -0
  76. package/dist/interactions/match-interaction.js +188 -0
  77. package/dist/interactions/match-interaction.js.map +1 -0
  78. package/dist/interactions/object-asset.d.ts +8 -0
  79. package/dist/interactions/object-asset.d.ts.map +1 -0
  80. package/dist/interactions/object-asset.js +182 -0
  81. package/dist/interactions/object-asset.js.map +1 -0
  82. package/dist/interactions/pair-interaction.d.ts +4 -0
  83. package/dist/interactions/pair-interaction.d.ts.map +1 -0
  84. package/dist/interactions/pair-interaction.js +125 -0
  85. package/dist/interactions/pair-interaction.js.map +1 -0
  86. package/dist/interactions/point-value.d.ts +16 -0
  87. package/dist/interactions/point-value.d.ts.map +1 -0
  88. package/dist/interactions/point-value.js +37 -0
  89. package/dist/interactions/point-value.js.map +1 -0
  90. package/dist/interactions/portable-custom-interaction.d.ts +16 -0
  91. package/dist/interactions/portable-custom-interaction.d.ts.map +1 -0
  92. package/dist/interactions/portable-custom-interaction.js +97 -0
  93. package/dist/interactions/portable-custom-interaction.js.map +1 -0
  94. package/dist/interactions/position-object-interaction.d.ts +4 -0
  95. package/dist/interactions/position-object-interaction.d.ts.map +1 -0
  96. package/dist/interactions/position-object-interaction.js +177 -0
  97. package/dist/interactions/position-object-interaction.js.map +1 -0
  98. package/dist/interactions/routing.d.ts +5 -0
  99. package/dist/interactions/routing.d.ts.map +1 -0
  100. package/dist/interactions/routing.js +24 -0
  101. package/dist/interactions/routing.js.map +1 -0
  102. package/dist/interactions/select-point-interaction.d.ts +4 -0
  103. package/dist/interactions/select-point-interaction.d.ts.map +1 -0
  104. package/dist/interactions/select-point-interaction.js +166 -0
  105. package/dist/interactions/select-point-interaction.js.map +1 -0
  106. package/dist/interactions/shared.d.ts +12 -0
  107. package/dist/interactions/shared.d.ts.map +1 -0
  108. package/dist/interactions/shared.js +65 -0
  109. package/dist/interactions/shared.js.map +1 -0
  110. package/dist/interactions/text-interaction.d.ts +6 -0
  111. package/dist/interactions/text-interaction.d.ts.map +1 -0
  112. package/dist/interactions/text-interaction.js +110 -0
  113. package/dist/interactions/text-interaction.js.map +1 -0
  114. package/dist/interactions/unsupported-interaction.d.ts +4 -0
  115. package/dist/interactions/unsupported-interaction.d.ts.map +1 -0
  116. package/dist/interactions/unsupported-interaction.js +23 -0
  117. package/dist/interactions/unsupported-interaction.js.map +1 -0
  118. package/dist/interactions/upload-interaction.d.ts +4 -0
  119. package/dist/interactions/upload-interaction.d.ts.map +1 -0
  120. package/dist/interactions/upload-interaction.js +10 -0
  121. package/dist/interactions/upload-interaction.js.map +1 -0
  122. package/dist/player/attempt-availability.d.ts +6 -0
  123. package/dist/player/attempt-availability.d.ts.map +1 -0
  124. package/dist/player/attempt-availability.js +27 -0
  125. package/dist/player/attempt-availability.js.map +1 -0
  126. package/dist/player/content-state.d.ts +19 -0
  127. package/dist/player/content-state.d.ts.map +1 -0
  128. package/dist/player/content-state.js +39 -0
  129. package/dist/player/content-state.js.map +1 -0
  130. package/dist/player/dynamic-body.d.ts +7 -0
  131. package/dist/player/dynamic-body.d.ts.map +1 -0
  132. package/dist/player/dynamic-body.js +28 -0
  133. package/dist/player/dynamic-body.js.map +1 -0
  134. package/dist/player/feedback-panel.d.ts +3 -0
  135. package/dist/player/feedback-panel.d.ts.map +1 -0
  136. package/dist/player/feedback-panel.js +14 -0
  137. package/dist/player/feedback-panel.js.map +1 -0
  138. package/dist/player/fetch-xml.d.ts +2 -0
  139. package/dist/player/fetch-xml.d.ts.map +1 -0
  140. package/dist/player/fetch-xml.js +10 -0
  141. package/dist/player/fetch-xml.js.map +1 -0
  142. package/dist/player/interaction-render.d.ts +14 -0
  143. package/dist/player/interaction-render.d.ts.map +1 -0
  144. package/dist/player/interaction-render.js +51 -0
  145. package/dist/player/interaction-render.js.map +1 -0
  146. package/dist/player/render-shell.d.ts +8 -0
  147. package/dist/player/render-shell.d.ts.map +1 -0
  148. package/dist/player/render-shell.js +37 -0
  149. package/dist/player/render-shell.js.map +1 -0
  150. package/dist/player/resolve-assets.d.ts +3 -0
  151. package/dist/player/resolve-assets.d.ts.map +1 -0
  152. package/dist/player/resolve-assets.js +12 -0
  153. package/dist/player/resolve-assets.js.map +1 -0
  154. package/dist/player/validation-messages.d.ts +17 -0
  155. package/dist/player/validation-messages.d.ts.map +1 -0
  156. package/dist/player/validation-messages.js +33 -0
  157. package/dist/player/validation-messages.js.map +1 -0
  158. package/dist/player-adapter.d.ts +62 -0
  159. package/dist/player-adapter.d.ts.map +1 -0
  160. package/dist/player-adapter.js +119 -0
  161. package/dist/player-adapter.js.map +1 -0
  162. package/dist/player-dev.d.ts +4 -0
  163. package/dist/player-dev.d.ts.map +1 -0
  164. package/dist/player-dev.js +14 -0
  165. package/dist/player-dev.js.map +1 -0
  166. package/dist/player-element.d.ts +14 -1
  167. package/dist/player-element.d.ts.map +1 -1
  168. package/dist/player-element.js +57 -5
  169. package/dist/player-element.js.map +1 -1
  170. package/dist/player-locale.d.ts +8 -3
  171. package/dist/player-locale.d.ts.map +1 -1
  172. package/dist/player-locale.js +16 -175
  173. package/dist/player-locale.js.map +1 -1
  174. package/dist/player-message-catalog-default.d.ts +4 -0
  175. package/dist/player-message-catalog-default.d.ts.map +1 -0
  176. package/dist/player-message-catalog-default.js +118 -0
  177. package/dist/player-message-catalog-default.js.map +1 -0
  178. package/dist/player-message-catalog-validate.d.ts +31 -0
  179. package/dist/player-message-catalog-validate.d.ts.map +1 -0
  180. package/dist/player-message-catalog-validate.js +327 -0
  181. package/dist/player-message-catalog-validate.js.map +1 -0
  182. package/dist/player-message-catalog.d.ts +18 -0
  183. package/dist/player-message-catalog.d.ts.map +1 -0
  184. package/dist/player-message-catalog.js +40 -0
  185. package/dist/player-message-catalog.js.map +1 -0
  186. package/dist/player-message-keys.d.ts +6 -0
  187. package/dist/player-message-keys.d.ts.map +1 -0
  188. package/dist/player-message-keys.js +7 -0
  189. package/dist/player-message-keys.js.map +1 -0
  190. package/dist/player-message-manifest.d.ts +272 -0
  191. package/dist/player-message-manifest.d.ts.map +1 -0
  192. package/dist/player-message-manifest.js +83 -0
  193. package/dist/player-message-manifest.js.map +1 -0
  194. package/dist/player-message-overrides.d.ts +3 -0
  195. package/dist/player-message-overrides.d.ts.map +1 -0
  196. package/dist/player-message-overrides.js +28 -0
  197. package/dist/player-message-overrides.js.map +1 -0
  198. package/dist/player-message-resolver.d.ts +31 -0
  199. package/dist/player-message-resolver.d.ts.map +1 -0
  200. package/dist/player-message-resolver.js +110 -0
  201. package/dist/player-message-resolver.js.map +1 -0
  202. package/dist/player-messages.d.ts +0 -38
  203. package/dist/player-messages.d.ts.map +1 -1
  204. package/dist/player-types.d.ts +12 -2
  205. package/dist/player-types.d.ts.map +1 -1
  206. package/dist/reorder/a11y.d.ts +7 -0
  207. package/dist/reorder/a11y.d.ts.map +1 -0
  208. package/dist/reorder/a11y.js +34 -0
  209. package/dist/reorder/a11y.js.map +1 -0
  210. package/dist/reorder/graphic-order-interaction.d.ts +4 -0
  211. package/dist/reorder/graphic-order-interaction.d.ts.map +1 -0
  212. package/dist/reorder/graphic-order-interaction.js +205 -0
  213. package/dist/reorder/graphic-order-interaction.js.map +1 -0
  214. package/dist/reorder/list-controls.d.ts +22 -0
  215. package/dist/reorder/list-controls.d.ts.map +1 -0
  216. package/dist/reorder/list-controls.js +75 -0
  217. package/dist/reorder/list-controls.js.map +1 -0
  218. package/dist/reorder/order-interaction.d.ts +4 -0
  219. package/dist/reorder/order-interaction.d.ts.map +1 -0
  220. package/dist/reorder/order-interaction.js +54 -0
  221. package/dist/reorder/order-interaction.js.map +1 -0
  222. package/dist/styles/base-styles.d.ts +2 -0
  223. package/dist/styles/base-styles.d.ts.map +1 -0
  224. package/dist/styles/base-styles.js +133 -0
  225. package/dist/styles/base-styles.js.map +1 -0
  226. package/dist/styles/choice-hottext-styles.d.ts +2 -0
  227. package/dist/styles/choice-hottext-styles.d.ts.map +1 -0
  228. package/dist/styles/choice-hottext-styles.js +76 -0
  229. package/dist/styles/choice-hottext-styles.js.map +1 -0
  230. package/dist/styles/control-styles.d.ts +2 -0
  231. package/dist/styles/control-styles.d.ts.map +1 -0
  232. package/dist/styles/control-styles.js +114 -0
  233. package/dist/styles/control-styles.js.map +1 -0
  234. package/dist/styles/drawing-styles.d.ts +2 -0
  235. package/dist/styles/drawing-styles.d.ts.map +1 -0
  236. package/dist/styles/drawing-styles.js +30 -0
  237. package/dist/styles/drawing-styles.js.map +1 -0
  238. package/dist/styles/gap-match-styles.d.ts +2 -0
  239. package/dist/styles/gap-match-styles.d.ts.map +1 -0
  240. package/dist/styles/gap-match-styles.js +33 -0
  241. package/dist/styles/gap-match-styles.js.map +1 -0
  242. package/dist/styles/graphic-styles.d.ts +2 -0
  243. package/dist/styles/graphic-styles.d.ts.map +1 -0
  244. package/dist/styles/graphic-styles.js +289 -0
  245. package/dist/styles/graphic-styles.js.map +1 -0
  246. package/dist/styles/match-pair-styles.d.ts +2 -0
  247. package/dist/styles/match-pair-styles.d.ts.map +1 -0
  248. package/dist/styles/match-pair-styles.js +62 -0
  249. package/dist/styles/match-pair-styles.js.map +1 -0
  250. package/dist/styles/text-slider-styles.d.ts +2 -0
  251. package/dist/styles/text-slider-styles.d.ts.map +1 -0
  252. package/dist/styles/text-slider-styles.js +35 -0
  253. package/dist/styles/text-slider-styles.js.map +1 -0
  254. package/package.json +8 -8
  255. package/src/controls/remove-button.ts +8 -5
  256. package/src/index.ts +61 -5
  257. package/src/interactions/choice-interaction.ts +6 -2
  258. package/src/interactions/drawing-interaction.ts +14 -9
  259. package/src/interactions/end-attempt-interaction.ts +3 -3
  260. package/src/interactions/gap-match-interaction.ts +32 -13
  261. package/src/interactions/graphic-associate-interaction.ts +15 -10
  262. package/src/interactions/hotspot-interaction.ts +10 -6
  263. package/src/interactions/inline-choice-interaction.ts +4 -4
  264. package/src/interactions/interaction-registry.ts +12 -12
  265. package/src/interactions/match-interaction.ts +9 -6
  266. package/src/interactions/pair-interaction.ts +22 -14
  267. package/src/interactions/position-object-interaction.ts +22 -13
  268. package/src/interactions/select-point-interaction.ts +25 -13
  269. package/src/interactions/shared.ts +21 -4
  270. package/src/interactions/text-interaction.ts +14 -4
  271. package/src/interactions/upload-interaction.ts +6 -3
  272. package/src/player/interaction-render.ts +4 -4
  273. package/src/player-adapter.ts +253 -0
  274. package/src/player-dev.ts +14 -0
  275. package/src/player-element.ts +78 -8
  276. package/src/player-locale.ts +28 -199
  277. package/src/player-message-catalog-default.ts +119 -0
  278. package/src/player-message-catalog-validate.ts +425 -0
  279. package/src/player-message-catalog.ts +72 -0
  280. package/src/player-message-keys.ts +12 -0
  281. package/src/player-message-manifest.ts +103 -0
  282. package/src/player-message-overrides.ts +38 -0
  283. package/src/player-message-resolver.ts +205 -0
  284. package/src/player-messages.ts +0 -30
  285. package/src/player-types.ts +15 -4
  286. package/src/reorder/a11y.ts +22 -7
  287. package/src/reorder/graphic-order-interaction.ts +23 -16
  288. package/src/reorder/list-controls.ts +8 -6
  289. package/src/reorder/order-interaction.ts +7 -5
  290. package/src/styles/base-styles.ts +20 -5
  291. package/src/styles/graphic-styles.ts +0 -6
@@ -0,0 +1,277 @@
1
+ import { applyGraphicSurfaceLayout, appendGraphicObjectImage, missingChoicesMessage, objectHeight, objectWidth, placeHotspotButton, responseGroup, valueToStrings, } from "../interaction-support.js";
2
+ import { parseUnlimitedMaximum } from "../response-limits.js";
3
+ import { appendGraphicContext } from "./graphic-context.js";
4
+ import { appendInlineControl, normalizeInlineSegmentText } from "./inline-controls.js";
5
+ import { sourceChoices, targetChoices, tokenButton, tokenRegion } from "./shared.js";
6
+ function positivePixelValue(value) {
7
+ const parsed = Number(value);
8
+ return Number.isFinite(parsed) && parsed > 0 ? parsed : undefined;
9
+ }
10
+ function graphicGapLabelBlockSize(sources) {
11
+ const maxLength = Math.max(0, ...sources.map((source) => (source.text || source.identifier).trim().length));
12
+ const estimatedLines = Math.max(1, Math.ceil(maxLength / 22));
13
+ return Number((estimatedLines * 0.95 + 0.9).toFixed(2));
14
+ }
15
+ export function renderGapMatchResponse(interaction, update, currentValue, messages) {
16
+ if (interaction.type === "graphicGapMatch" &&
17
+ interaction.object &&
18
+ interaction.choices.some((choice) => choice.role === "hotspot")) {
19
+ return renderGraphicGapMatchResponse(interaction, update, currentValue, messages);
20
+ }
21
+ const group = responseGroup();
22
+ appendGraphicContext(group, interaction);
23
+ const sources = sourceChoices(interaction);
24
+ const gaps = targetChoices(interaction);
25
+ if (sources.length === 0 || gaps.length === 0) {
26
+ group.append(missingChoicesMessage(interaction));
27
+ return group;
28
+ }
29
+ const assignments = new Map();
30
+ let selectedSource;
31
+ let draggedSource;
32
+ const sourceRegion = tokenRegion(messages.message("interactionChoicesBank", { type: interaction.type }));
33
+ const gapRegion = document.createElement("div");
34
+ gapRegion.className = "qti3-gap-region qti3-gap-passage";
35
+ gapRegion.role = "group";
36
+ gapRegion.setAttribute("aria-label", messages.message("interactionGapTargets", { type: interaction.type }));
37
+ for (const pair of valueToStrings(currentValue)) {
38
+ const [sourceIdentifier, gapIdentifier] = pair.split(/\s+/);
39
+ const source = sources.find((choice) => choice.identifier === sourceIdentifier);
40
+ if (source && gapIdentifier)
41
+ assignments.set(gapIdentifier, source);
42
+ }
43
+ const commit = () => {
44
+ update([...assignments.entries()].map(([gapIdentifier, source]) => `${source.identifier} ${gapIdentifier}`));
45
+ };
46
+ const syncSources = () => {
47
+ for (const button of sourceRegion.querySelectorAll("button")) {
48
+ button.setAttribute("aria-pressed", button.dataset.choiceIdentifier === selectedSource?.identifier ? "true" : "false");
49
+ }
50
+ };
51
+ const assign = (gap, sourceIdentifier) => {
52
+ const source = sources.find((choice) => choice.identifier === sourceIdentifier);
53
+ if (!source)
54
+ return;
55
+ assignments.set(gap.identifier, source);
56
+ selectedSource = undefined;
57
+ syncSources();
58
+ renderGaps();
59
+ commit();
60
+ };
61
+ const gapControl = (gap, index) => {
62
+ const assigned = assignments.get(gap.identifier);
63
+ const gapLabel = messages.message("gapLabel", { index: index + 1 });
64
+ const target = document.createElement("span");
65
+ target.className = "qti3-gap-target";
66
+ target.dataset.gapIdentifier = gap.identifier;
67
+ target.addEventListener("dragover", (event) => {
68
+ event.preventDefault();
69
+ target.classList.add("qti3-drop-target");
70
+ });
71
+ target.addEventListener("dragleave", () => target.classList.remove("qti3-drop-target"));
72
+ target.addEventListener("drop", (event) => {
73
+ event.preventDefault();
74
+ target.classList.remove("qti3-drop-target");
75
+ assign(gap, event.dataTransfer?.getData("text/plain") || draggedSource);
76
+ });
77
+ const button = document.createElement("button");
78
+ button.type = "button";
79
+ button.className = "qti3-gap-button";
80
+ button.textContent = assigned ? assigned.text : "";
81
+ button.setAttribute("aria-label", assigned
82
+ ? messages.message("gapAssignedState", { label: gapLabel, assigned: assigned.text })
83
+ : messages.message("gapEmptyState", { label: gapLabel }));
84
+ button.addEventListener("click", () => assign(gap, selectedSource?.identifier));
85
+ button.addEventListener("keydown", (event) => {
86
+ if (event.key !== "Delete" && event.key !== "Backspace")
87
+ return;
88
+ if (!assignments.has(gap.identifier))
89
+ return;
90
+ event.preventDefault();
91
+ assignments.delete(gap.identifier);
92
+ renderGaps();
93
+ commit();
94
+ });
95
+ target.append(button);
96
+ return target;
97
+ };
98
+ const renderGaps = () => {
99
+ const segments = interaction.gapMatchSegments ?? [];
100
+ const hasInlineGaps = segments.some((segment) => segment.kind === "gap");
101
+ if (!hasInlineGaps) {
102
+ gapRegion.replaceChildren(...gaps.map((gap, index) => gapControl(gap, index)));
103
+ return;
104
+ }
105
+ const content = [];
106
+ for (const [segmentIndex, segment] of segments.entries()) {
107
+ if (segment.kind === "text") {
108
+ content.push(document.createTextNode(normalizeInlineSegmentText(segment.text)));
109
+ continue;
110
+ }
111
+ const gapIndex = gaps.findIndex((gap) => gap.identifier === segment.identifier);
112
+ const gap = gaps[gapIndex];
113
+ if (gap) {
114
+ appendInlineControl(content, gapControl(gap, gapIndex), segments[segmentIndex + 1]);
115
+ }
116
+ }
117
+ gapRegion.replaceChildren(...content);
118
+ };
119
+ for (const source of sources) {
120
+ const button = tokenButton(source);
121
+ button.draggable = true;
122
+ button.addEventListener("dragstart", (event) => {
123
+ draggedSource = source.identifier;
124
+ event.dataTransfer?.setData("text/plain", source.identifier);
125
+ });
126
+ button.addEventListener("click", () => {
127
+ selectedSource = source;
128
+ syncSources();
129
+ });
130
+ sourceRegion.append(button);
131
+ }
132
+ renderGaps();
133
+ group.append(sourceRegion, gapRegion);
134
+ return group;
135
+ }
136
+ function renderGraphicGapMatchResponse(interaction, update, currentValue, messages) {
137
+ const group = responseGroup();
138
+ const width = objectWidth(interaction);
139
+ const height = objectHeight(interaction);
140
+ const sources = sourceChoices(interaction);
141
+ const gaps = targetChoices(interaction).filter((choice) => choice.role === "hotspot");
142
+ if (sources.length === 0 || gaps.length === 0) {
143
+ group.append(missingChoicesMessage(interaction));
144
+ return group;
145
+ }
146
+ const assignments = new Map();
147
+ let selectedSource;
148
+ let draggedSource;
149
+ for (const pair of valueToStrings(currentValue)) {
150
+ const [sourceIdentifier, gapIdentifier] = pair.split(/\s+/);
151
+ const source = sources.find((choice) => choice.identifier === sourceIdentifier);
152
+ if (source && gapIdentifier)
153
+ assignments.set(gapIdentifier, source);
154
+ }
155
+ const surface = document.createElement("div");
156
+ applyGraphicSurfaceLayout(surface, width, height, "qti3-graphic-context", "qti3-graphic-gap-match-surface");
157
+ surface.role = "group";
158
+ surface.setAttribute("aria-label", messages.message("interactionTargetImage", { type: interaction.type }));
159
+ surface.style.overflow = "visible";
160
+ surface.style.setProperty("--qti3-graphic-gap-label-block-size", `${graphicGapLabelBlockSize(sources)}rem`);
161
+ if (interaction.object) {
162
+ appendGraphicObjectImage(surface, interaction.object, interaction.object.text ||
163
+ messages.message("interactionImageAlt", { type: interaction.type }));
164
+ }
165
+ const sourceRegion = tokenRegion(messages.message("interactionChoicesBank", { type: interaction.type }));
166
+ sourceRegion.classList.add("qti3-graphic-gap-source-region");
167
+ const choicesWidth = positivePixelValue(interaction.attributes["data-choices-container-width"]);
168
+ if (choicesWidth !== undefined)
169
+ sourceRegion.style.maxInlineSize = `${choicesWidth}px`;
170
+ const summary = document.createElement("p");
171
+ summary.className = "qti3-selection-summary";
172
+ summary.setAttribute("aria-live", "polite");
173
+ const commit = () => {
174
+ update([...assignments.entries()].map(([gapIdentifier, source]) => `${source.identifier} ${gapIdentifier}`));
175
+ };
176
+ const syncSources = () => {
177
+ for (const button of sourceRegion.querySelectorAll("button")) {
178
+ button.setAttribute("aria-pressed", button.dataset.choiceIdentifier === selectedSource?.identifier ? "true" : "false");
179
+ }
180
+ };
181
+ const clearSourceIfSingleUse = (source, keepGapIdentifier) => {
182
+ if (parseUnlimitedMaximum(source.attributes["match-max"]) !== 1)
183
+ return;
184
+ for (const [gapIdentifier, assigned] of assignments.entries()) {
185
+ if (gapIdentifier !== keepGapIdentifier && assigned.identifier === source.identifier) {
186
+ assignments.delete(gapIdentifier);
187
+ }
188
+ }
189
+ };
190
+ const assign = (gap, sourceIdentifier) => {
191
+ const source = sources.find((choice) => choice.identifier === sourceIdentifier);
192
+ if (!source)
193
+ return;
194
+ clearSourceIfSingleUse(source, gap.identifier);
195
+ assignments.set(gap.identifier, source);
196
+ selectedSource = undefined;
197
+ syncSources();
198
+ renderTargets();
199
+ commit();
200
+ };
201
+ const targetLabel = (gap, index) => gap.attributes["aria-label"] ||
202
+ gap.attributes["hotspot-label"] ||
203
+ messages.message("graphicGapTargetLabel", { index: index + 1 });
204
+ const renderTargetButton = (gap, index) => {
205
+ const assigned = assignments.get(gap.identifier);
206
+ const label = targetLabel(gap, index);
207
+ const button = document.createElement("button");
208
+ button.type = "button";
209
+ button.className = "qti3-hotspot-button qti3-graphic-gap-hotspot";
210
+ button.dataset.gapIdentifier = gap.identifier;
211
+ button.dataset.selected = assigned ? "true" : "false";
212
+ button.setAttribute("aria-label", assigned
213
+ ? messages.message("gapAssignedState", { label, assigned: assigned.text })
214
+ : messages.message("gapEmptyState", { label }));
215
+ button.addEventListener("dragover", (event) => {
216
+ event.preventDefault();
217
+ button.classList.add("qti3-drop-target");
218
+ });
219
+ button.addEventListener("dragleave", () => button.classList.remove("qti3-drop-target"));
220
+ button.addEventListener("drop", (event) => {
221
+ event.preventDefault();
222
+ button.classList.remove("qti3-drop-target");
223
+ assign(gap, event.dataTransfer?.getData("text/plain") || draggedSource);
224
+ });
225
+ button.addEventListener("click", () => assign(gap, selectedSource?.identifier));
226
+ button.addEventListener("keydown", (event) => {
227
+ if (event.key !== "Delete" && event.key !== "Backspace")
228
+ return;
229
+ if (!assignments.has(gap.identifier))
230
+ return;
231
+ event.preventDefault();
232
+ assignments.delete(gap.identifier);
233
+ renderTargets();
234
+ commit();
235
+ });
236
+ placeHotspotButton(button, gap, width, height);
237
+ if (assigned) {
238
+ const assignedLabel = document.createElement("span");
239
+ assignedLabel.className = "qti3-graphic-gap-label";
240
+ assignedLabel.textContent = assigned.text;
241
+ button.append(assignedLabel);
242
+ }
243
+ return button;
244
+ };
245
+ const renderTargets = () => {
246
+ surface.querySelectorAll(".qti3-graphic-gap-hotspot").forEach((target) => target.remove());
247
+ for (const [index, gap] of gaps.entries()) {
248
+ surface.append(renderTargetButton(gap, index));
249
+ }
250
+ summary.textContent =
251
+ assignments.size > 0
252
+ ? messages.message("gapLabelsPlacedCount", { count: assignments.size })
253
+ : messages.message("gapNoLabelsPlaced");
254
+ };
255
+ for (const source of sources) {
256
+ const button = tokenButton(source);
257
+ button.draggable = true;
258
+ button.addEventListener("dragstart", (event) => {
259
+ draggedSource = source.identifier;
260
+ event.dataTransfer?.setData("text/plain", source.identifier);
261
+ event.dataTransfer?.setDragImage(button, 8, 8);
262
+ });
263
+ button.addEventListener("dragend", () => {
264
+ draggedSource = undefined;
265
+ syncSources();
266
+ });
267
+ button.addEventListener("click", () => {
268
+ selectedSource = source;
269
+ syncSources();
270
+ });
271
+ sourceRegion.append(button);
272
+ }
273
+ renderTargets();
274
+ group.append(surface, sourceRegion, summary);
275
+ return group;
276
+ }
277
+ //# sourceMappingURL=gap-match-interaction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"gap-match-interaction.js","sourceRoot":"","sources":["../../src/interactions/gap-match-interaction.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { QtiInteraction, QtiValue } from "@longsightgroup/qti3-core";
2
+ import type { PlayerMessageResolver } from "../player-message-resolver.js";
3
+ export declare function renderGraphicAssociateResponse(interaction: QtiInteraction, update: (value: QtiValue) => void, currentValue: QtiValue, messages: PlayerMessageResolver): HTMLElement;
4
+ //# sourceMappingURL=graphic-associate-interaction.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graphic-associate-interaction.d.ts","sourceRoot":"","sources":["../../src/interactions/graphic-associate-interaction.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAa,cAAc,EAAE,QAAQ,EAAE,MAAM,2BAA2B,CAAC;AAgBrF,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,+BAA+B,CAAC;AAG3E,wBAAgB,8BAA8B,CAC5C,WAAW,EAAE,cAAc,EAC3B,MAAM,EAAE,CAAC,KAAK,EAAE,QAAQ,KAAK,IAAI,EACjC,YAAY,EAAE,QAAQ,EACtB,QAAQ,EAAE,qBAAqB,GAC9B,WAAW,CAgTb"}
@@ -0,0 +1,297 @@
1
+ import { removeButton } from "../controls/remove-button.js";
2
+ import { applyGraphicSurfaceLayout, appendGraphicObjectImage, interactionChoices, missingChoicesMessage, hotspotAccessibleLabel, hotspotCenter, hotspotDisplayLabel, objectHeight, objectWidth, placeHotspotButton, responseGroup, valueToStrings, } from "../interaction-support.js";
3
+ import { exceedsHotspotMatchMax, maximumAllowedResponses } from "../response-limits.js";
4
+ export function renderGraphicAssociateResponse(interaction, update, currentValue, messages) {
5
+ const group = responseGroup();
6
+ const width = objectWidth(interaction);
7
+ const height = objectHeight(interaction);
8
+ const choices = interactionChoices(interaction).filter((choice) => choice.role === "hotspot");
9
+ if (choices.length === 0) {
10
+ group.append(missingChoicesMessage(interaction));
11
+ return group;
12
+ }
13
+ const selectedPairs = valueToStrings(currentValue);
14
+ const maximumAssociations = interaction.responseCardinality === "single" ? 1 : maximumAllowedResponses(interaction);
15
+ let selectedHotspot;
16
+ let draggedHotspot;
17
+ let dragPointerId;
18
+ let dragStart;
19
+ let dragStarted = false;
20
+ let suppressNextClick = false;
21
+ let previewLine;
22
+ const surface = document.createElement("div");
23
+ applyGraphicSurfaceLayout(surface, width, height, "qti3-graphic-associate-surface");
24
+ surface.role = "group";
25
+ surface.setAttribute("aria-label", messages.message("interactionHotspots", { type: interaction.type }));
26
+ const object = interaction.object;
27
+ if (object) {
28
+ appendGraphicObjectImage(surface, object, object.text || messages.message("interactionImageAlt", { type: interaction.type }));
29
+ }
30
+ const connections = document.createElementNS("http://www.w3.org/2000/svg", "svg");
31
+ connections.classList.add("qti3-graphic-associate-lines");
32
+ connections.setAttribute("viewBox", `0 0 ${width} ${height}`);
33
+ connections.setAttribute("aria-hidden", "true");
34
+ surface.append(connections);
35
+ const summary = document.createElement("p");
36
+ summary.className = "qti3-selection-summary";
37
+ summary.setAttribute("aria-live", "polite");
38
+ const pairList = document.createElement("ul");
39
+ pairList.className = "qti3-pair-list";
40
+ pairList.setAttribute("aria-label", messages.message("interactionSelectedPairsList", { type: interaction.type }));
41
+ const commit = () => {
42
+ if (interaction.responseCardinality === "single")
43
+ update(selectedPairs[0] ?? null);
44
+ else
45
+ update([...selectedPairs]);
46
+ };
47
+ const removePair = (pair) => {
48
+ const index = selectedPairs.indexOf(pair);
49
+ if (index < 0)
50
+ return;
51
+ selectedPairs.splice(index, 1);
52
+ renderState();
53
+ commit();
54
+ };
55
+ const removePairsForHotspot = (identifier) => {
56
+ let removed = false;
57
+ for (let index = selectedPairs.length - 1; index >= 0; index -= 1) {
58
+ const [source, target] = selectedPairs[index]?.split(" ") ?? [];
59
+ if (source === identifier || target === identifier) {
60
+ selectedPairs.splice(index, 1);
61
+ removed = true;
62
+ }
63
+ }
64
+ if (!removed)
65
+ return;
66
+ renderState();
67
+ commit();
68
+ };
69
+ const addPair = (source, target) => {
70
+ if (source.identifier === target.identifier) {
71
+ selectedHotspot = undefined;
72
+ renderState();
73
+ return;
74
+ }
75
+ const pair = `${source.identifier} ${target.identifier}`;
76
+ if (!selectedPairs.includes(pair)) {
77
+ if (interaction.responseCardinality === "single")
78
+ selectedPairs.splice(0);
79
+ if (maximumAssociations !== undefined &&
80
+ selectedPairs.length >= maximumAssociations &&
81
+ interaction.responseCardinality !== "single") {
82
+ selectedHotspot = undefined;
83
+ renderState();
84
+ return;
85
+ }
86
+ if (exceedsHotspotMatchMax(source, selectedPairs) ||
87
+ exceedsHotspotMatchMax(target, selectedPairs)) {
88
+ selectedHotspot = undefined;
89
+ renderState();
90
+ return;
91
+ }
92
+ selectedPairs.push(pair);
93
+ }
94
+ selectedHotspot = undefined;
95
+ renderState();
96
+ commit();
97
+ };
98
+ const authoredPointFromPointer = (event) => {
99
+ const rect = surface.getBoundingClientRect();
100
+ return {
101
+ x: Math.max(0, Math.min(width, ((event.clientX - rect.left) / rect.width) * width)),
102
+ y: Math.max(0, Math.min(height, ((event.clientY - rect.top) / rect.height) * height)),
103
+ };
104
+ };
105
+ const removePreviewLine = () => {
106
+ previewLine?.remove();
107
+ previewLine = undefined;
108
+ };
109
+ const suppressFollowingClick = () => {
110
+ suppressNextClick = true;
111
+ setTimeout(() => {
112
+ suppressNextClick = false;
113
+ }, 0);
114
+ };
115
+ const updatePreviewLine = (source, event) => {
116
+ const start = hotspotCenter(source, width, height);
117
+ const end = authoredPointFromPointer(event);
118
+ if (!previewLine) {
119
+ previewLine = document.createElementNS("http://www.w3.org/2000/svg", "line");
120
+ previewLine.dataset.preview = "true";
121
+ connections.append(previewLine);
122
+ }
123
+ previewLine.setAttribute("x1", String(start.x));
124
+ previewLine.setAttribute("y1", String(start.y));
125
+ previewLine.setAttribute("x2", String(end.x));
126
+ previewLine.setAttribute("y2", String(end.y));
127
+ };
128
+ const hotspotFromPointer = (event) => {
129
+ const element = document.elementFromPoint(event.clientX, event.clientY);
130
+ const button = element?.closest(".qti3-graphic-associate-hotspot");
131
+ const identifier = button?.dataset.choiceIdentifier;
132
+ return choices.find((choice) => choice.identifier === identifier);
133
+ };
134
+ const finishDrag = (event, source) => {
135
+ const target = hotspotFromPointer(event);
136
+ removePreviewLine();
137
+ if (target) {
138
+ addPair(source, target);
139
+ return;
140
+ }
141
+ selectedHotspot = undefined;
142
+ renderState();
143
+ };
144
+ const chooseHotspot = (choice) => {
145
+ if (!selectedHotspot) {
146
+ selectedHotspot = choice;
147
+ renderState();
148
+ return;
149
+ }
150
+ addPair(selectedHotspot, choice);
151
+ };
152
+ const focusRelativeHotspot = (choice, delta) => {
153
+ const index = choices.findIndex((entry) => entry.identifier === choice.identifier);
154
+ const next = choices[(index + delta + choices.length) % choices.length];
155
+ if (!next)
156
+ return;
157
+ surface
158
+ .querySelector(`[data-choice-identifier="${next.identifier}"]`)
159
+ ?.focus();
160
+ };
161
+ const renderState = () => {
162
+ connections.replaceChildren(...selectedPairs.flatMap((pair) => {
163
+ const [sourceIdentifier, targetIdentifier] = pair.split(" ");
164
+ const source = choices.find((choice) => choice.identifier === sourceIdentifier);
165
+ const target = choices.find((choice) => choice.identifier === targetIdentifier);
166
+ if (!source || !target)
167
+ return [];
168
+ const start = hotspotCenter(source, width, height);
169
+ const end = hotspotCenter(target, width, height);
170
+ const line = document.createElementNS("http://www.w3.org/2000/svg", "line");
171
+ line.setAttribute("x1", String(start.x));
172
+ line.setAttribute("y1", String(start.y));
173
+ line.setAttribute("x2", String(end.x));
174
+ line.setAttribute("y2", String(end.y));
175
+ return [line];
176
+ }));
177
+ for (const button of surface.querySelectorAll(".qti3-hotspot-button")) {
178
+ const identifier = button.dataset.choiceIdentifier ?? "";
179
+ const isActive = identifier === selectedHotspot?.identifier;
180
+ const isPaired = selectedPairs.some((pair) => pair.split(" ").includes(identifier));
181
+ button.setAttribute("aria-pressed", isActive ? "true" : "false");
182
+ button.dataset.selected = isActive || isPaired ? "true" : "false";
183
+ }
184
+ summary.textContent = selectedHotspot
185
+ ? messages.message("hotspotSelectedChooseAnother", {
186
+ label: hotspotDisplayLabel(selectedHotspot, choices),
187
+ })
188
+ : selectedPairs.length > 0
189
+ ? messages.message("associationsMade", { count: selectedPairs.length })
190
+ : messages.message("noAssociationsMade");
191
+ pairList.replaceChildren(...selectedPairs.map((pair) => {
192
+ const [source = "", target = ""] = pair.split(" ");
193
+ const sourceChoice = choices.find((choice) => choice.identifier === source);
194
+ const targetChoice = choices.find((choice) => choice.identifier === target);
195
+ const pairLabel = messages.message("associationPairLabel", {
196
+ source: sourceChoice ? hotspotDisplayLabel(sourceChoice, choices) : source,
197
+ target: targetChoice ? hotspotDisplayLabel(targetChoice, choices) : target,
198
+ });
199
+ const item = document.createElement("li");
200
+ item.className = "qti3-pair-chip";
201
+ const text = document.createElement("span");
202
+ text.textContent = pairLabel;
203
+ const remove = removeButton(pairLabel, messages);
204
+ remove.addEventListener("click", () => removePair(pair));
205
+ item.append(text, remove);
206
+ return item;
207
+ }));
208
+ };
209
+ for (const [index, choice] of choices.entries()) {
210
+ const button = document.createElement("button");
211
+ button.type = "button";
212
+ button.className = "qti3-hotspot-button qti3-graphic-associate-hotspot";
213
+ button.dataset.choiceIdentifier = choice.identifier;
214
+ button.textContent = hotspotDisplayLabel(choice, choices);
215
+ button.title = hotspotAccessibleLabel(choice, index);
216
+ button.setAttribute("aria-pressed", "false");
217
+ button.setAttribute("aria-label", hotspotAccessibleLabel(choice, index));
218
+ placeHotspotButton(button, choice, width, height);
219
+ button.addEventListener("click", (event) => {
220
+ if (suppressNextClick) {
221
+ suppressNextClick = false;
222
+ event.preventDefault();
223
+ return;
224
+ }
225
+ chooseHotspot(choice);
226
+ });
227
+ button.addEventListener("pointerdown", (event) => {
228
+ if (event.button !== 0)
229
+ return;
230
+ draggedHotspot = choice;
231
+ dragPointerId = event.pointerId;
232
+ dragStart = { x: event.clientX, y: event.clientY };
233
+ dragStarted = false;
234
+ button.setPointerCapture(event.pointerId);
235
+ });
236
+ button.addEventListener("pointermove", (event) => {
237
+ if (dragPointerId !== event.pointerId || !draggedHotspot || !dragStart)
238
+ return;
239
+ const moved = Math.hypot(event.clientX - dragStart.x, event.clientY - dragStart.y);
240
+ if (!dragStarted && moved < 4)
241
+ return;
242
+ if (!dragStarted) {
243
+ dragStarted = true;
244
+ suppressFollowingClick();
245
+ selectedHotspot = draggedHotspot;
246
+ renderState();
247
+ }
248
+ updatePreviewLine(draggedHotspot, event);
249
+ event.preventDefault();
250
+ });
251
+ button.addEventListener("pointerup", (event) => {
252
+ if (dragPointerId !== event.pointerId || !draggedHotspot)
253
+ return;
254
+ const source = draggedHotspot;
255
+ draggedHotspot = undefined;
256
+ dragPointerId = undefined;
257
+ dragStart = undefined;
258
+ button.releasePointerCapture(event.pointerId);
259
+ if (!dragStarted)
260
+ return;
261
+ dragStarted = false;
262
+ suppressFollowingClick();
263
+ finishDrag(event, source);
264
+ event.preventDefault();
265
+ });
266
+ button.addEventListener("pointercancel", (event) => {
267
+ if (dragPointerId !== event.pointerId)
268
+ return;
269
+ draggedHotspot = undefined;
270
+ dragPointerId = undefined;
271
+ dragStart = undefined;
272
+ dragStarted = false;
273
+ removePreviewLine();
274
+ selectedHotspot = undefined;
275
+ renderState();
276
+ });
277
+ button.addEventListener("keydown", (event) => {
278
+ if (event.key === "ArrowRight" || event.key === "ArrowDown") {
279
+ event.preventDefault();
280
+ focusRelativeHotspot(choice, 1);
281
+ }
282
+ else if (event.key === "ArrowLeft" || event.key === "ArrowUp") {
283
+ event.preventDefault();
284
+ focusRelativeHotspot(choice, -1);
285
+ }
286
+ else if (event.key === "Delete" || event.key === "Backspace") {
287
+ event.preventDefault();
288
+ removePairsForHotspot(choice.identifier);
289
+ }
290
+ });
291
+ surface.append(button);
292
+ }
293
+ renderState();
294
+ group.append(surface, summary, pairList);
295
+ return group;
296
+ }
297
+ //# sourceMappingURL=graphic-associate-interaction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graphic-associate-interaction.js","sourceRoot":"","sources":["../../src/interactions/graphic-associate-interaction.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { QtiInteraction } from "@longsightgroup/qti3-core";
2
+ export declare function appendGraphicContext(group: HTMLElement, interaction: QtiInteraction): void;
3
+ //# sourceMappingURL=graphic-context.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graphic-context.d.ts","sourceRoot":"","sources":["../../src/interactions/graphic-context.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAGhE,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,WAAW,EAAE,WAAW,EAAE,cAAc,GAAG,IAAI,CA6B1F"}