coursecode 0.1.0

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 (362) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +322 -0
  3. package/THIRD_PARTY_NOTICES.md +22 -0
  4. package/bin/cli.js +331 -0
  5. package/framework/assets/logo-coursecode-black.svg +14 -0
  6. package/framework/assets/logo-coursecode-white.svg +14 -0
  7. package/framework/assets/logo-coursecode.svg +14 -0
  8. package/framework/css/01-base.css +160 -0
  9. package/framework/css/02-layout.css +499 -0
  10. package/framework/css/accessibility.css +834 -0
  11. package/framework/css/components/accordions.css +710 -0
  12. package/framework/css/components/assessments.css +520 -0
  13. package/framework/css/components/audio-player.css +570 -0
  14. package/framework/css/components/badges.css +80 -0
  15. package/framework/css/components/breadcrumbs.css +87 -0
  16. package/framework/css/components/buttons.css +707 -0
  17. package/framework/css/components/callouts.css +1280 -0
  18. package/framework/css/components/cards.css +475 -0
  19. package/framework/css/components/carousel.css +193 -0
  20. package/framework/css/components/checkbox-group.css +123 -0
  21. package/framework/css/components/checklist.css +203 -0
  22. package/framework/css/components/collapse.css +96 -0
  23. package/framework/css/components/comparison.css +33 -0
  24. package/framework/css/components/content-image.css +36 -0
  25. package/framework/css/components/document-gallery.css +425 -0
  26. package/framework/css/components/dropdown.css +115 -0
  27. package/framework/css/components/embed-frame.css +142 -0
  28. package/framework/css/components/engagement.css +412 -0
  29. package/framework/css/components/features.css +35 -0
  30. package/framework/css/components/flip-cards.css +253 -0
  31. package/framework/css/components/footer.css +353 -0
  32. package/framework/css/components/forms.css +294 -0
  33. package/framework/css/components/hero.css +216 -0
  34. package/framework/css/components/images.css +528 -0
  35. package/framework/css/components/interactive-timeline.css +274 -0
  36. package/framework/css/components/intro-cards.css +30 -0
  37. package/framework/css/components/lightbox.css +666 -0
  38. package/framework/css/components/loading.css +65 -0
  39. package/framework/css/components/modals.css +235 -0
  40. package/framework/css/components/notifications.css +107 -0
  41. package/framework/css/components/quote.css +150 -0
  42. package/framework/css/components/sidebar.css +684 -0
  43. package/framework/css/components/slide-header.css +52 -0
  44. package/framework/css/components/spinner.css +62 -0
  45. package/framework/css/components/stats.css +44 -0
  46. package/framework/css/components/steps.css +232 -0
  47. package/framework/css/components/tables.css +90 -0
  48. package/framework/css/components/tabs.css +347 -0
  49. package/framework/css/components/timeline.css +154 -0
  50. package/framework/css/components/toggle.css +95 -0
  51. package/framework/css/components/tooltip.css +226 -0
  52. package/framework/css/components/video-player.css +438 -0
  53. package/framework/css/design-tokens.css +707 -0
  54. package/framework/css/framework.css +86 -0
  55. package/framework/css/interactions/accessibility.css +75 -0
  56. package/framework/css/interactions/base.css +92 -0
  57. package/framework/css/interactions/drag-drop.css +295 -0
  58. package/framework/css/interactions/fill-in-the-blank.css +236 -0
  59. package/framework/css/interactions/hotspots.css +69 -0
  60. package/framework/css/interactions/index.css +45 -0
  61. package/framework/css/interactions/interactive-image.css +359 -0
  62. package/framework/css/interactions/likert.css +126 -0
  63. package/framework/css/interactions/matching.css +354 -0
  64. package/framework/css/interactions/numeric-input.css +78 -0
  65. package/framework/css/interactions/sequencing.css +378 -0
  66. package/framework/css/interactions/true-false.css +177 -0
  67. package/framework/css/layouts/article.css +258 -0
  68. package/framework/css/layouts/base.css +30 -0
  69. package/framework/css/layouts/canvas.css +38 -0
  70. package/framework/css/layouts/focused.css +236 -0
  71. package/framework/css/layouts/index.css +29 -0
  72. package/framework/css/layouts/presentation.css +191 -0
  73. package/framework/css/layouts/traditional.css +52 -0
  74. package/framework/css/responsive.css +439 -0
  75. package/framework/css/utilities/accessibility-utils.css +59 -0
  76. package/framework/css/utilities/animations.css +419 -0
  77. package/framework/css/utilities/borders.css +72 -0
  78. package/framework/css/utilities/colors.css +76 -0
  79. package/framework/css/utilities/container.css +46 -0
  80. package/framework/css/utilities/decorative.css +442 -0
  81. package/framework/css/utilities/display.css +257 -0
  82. package/framework/css/utilities/flexbox.css +80 -0
  83. package/framework/css/utilities/grid.css +69 -0
  84. package/framework/css/utilities/icons.css +534 -0
  85. package/framework/css/utilities/lists.css +190 -0
  86. package/framework/css/utilities/spacing.css +167 -0
  87. package/framework/css/utilities/tables.css +81 -0
  88. package/framework/css/utilities/typography.css +159 -0
  89. package/framework/css/utilities/visibility.css +117 -0
  90. package/framework/docs/COURSE_AUTHORING_GUIDE.md +1773 -0
  91. package/framework/docs/COURSE_OUTLINE_GUIDE.md +725 -0
  92. package/framework/docs/COURSE_OUTLINE_TEMPLATE.md +161 -0
  93. package/framework/docs/DATA_MODEL.md +409 -0
  94. package/framework/docs/FRAMEWORK_GUIDE.md +1088 -0
  95. package/framework/docs/USER_GUIDE.md +583 -0
  96. package/framework/docs/examples/cloudflare-channel-relay.js +169 -0
  97. package/framework/docs/examples/cloudflare-data-worker.js +102 -0
  98. package/framework/docs/examples/cloudflare-error-worker.js +228 -0
  99. package/framework/index.html +175 -0
  100. package/framework/js/app/AppActions.js +410 -0
  101. package/framework/js/app/AppState.js +225 -0
  102. package/framework/js/app/AppUI.js +616 -0
  103. package/framework/js/assessment/AssessmentActions.js +615 -0
  104. package/framework/js/assessment/AssessmentFactory.js +471 -0
  105. package/framework/js/assessment/AssessmentState.js +322 -0
  106. package/framework/js/assessment/AssessmentUI.js +451 -0
  107. package/framework/js/automation/api-engagement.js +196 -0
  108. package/framework/js/automation/api-interactions.js +167 -0
  109. package/framework/js/automation/api.js +242 -0
  110. package/framework/js/automation/index.js +41 -0
  111. package/framework/js/components/interactions/drag-drop.js +884 -0
  112. package/framework/js/components/interactions/fill-in.js +535 -0
  113. package/framework/js/components/interactions/hotspot.js +702 -0
  114. package/framework/js/components/interactions/interaction-base.js +511 -0
  115. package/framework/js/components/interactions/likert.js +301 -0
  116. package/framework/js/components/interactions/matching.js +699 -0
  117. package/framework/js/components/interactions/multiple-choice.js +377 -0
  118. package/framework/js/components/interactions/numeric.js +271 -0
  119. package/framework/js/components/interactions/sequencing.js +423 -0
  120. package/framework/js/components/interactions/true-false.js +241 -0
  121. package/framework/js/components/ui-components/accordion.js +442 -0
  122. package/framework/js/components/ui-components/alert.js +88 -0
  123. package/framework/js/components/ui-components/audio-player.js +1193 -0
  124. package/framework/js/components/ui-components/callout.js +121 -0
  125. package/framework/js/components/ui-components/carousel.js +145 -0
  126. package/framework/js/components/ui-components/checkbox-group.js +87 -0
  127. package/framework/js/components/ui-components/checklist.js +40 -0
  128. package/framework/js/components/ui-components/collapse.js +114 -0
  129. package/framework/js/components/ui-components/comparison.js +30 -0
  130. package/framework/js/components/ui-components/conditional-display.js +150 -0
  131. package/framework/js/components/ui-components/content-image.js +41 -0
  132. package/framework/js/components/ui-components/dropdown.js +262 -0
  133. package/framework/js/components/ui-components/embed-frame.js +274 -0
  134. package/framework/js/components/ui-components/features.js +33 -0
  135. package/framework/js/components/ui-components/flip-card.js +230 -0
  136. package/framework/js/components/ui-components/form-validator.js +76 -0
  137. package/framework/js/components/ui-components/hero.js +49 -0
  138. package/framework/js/components/ui-components/index.js +12 -0
  139. package/framework/js/components/ui-components/interactive-image.js +235 -0
  140. package/framework/js/components/ui-components/interactive-timeline.js +285 -0
  141. package/framework/js/components/ui-components/intro-cards.js +35 -0
  142. package/framework/js/components/ui-components/lightbox.js +652 -0
  143. package/framework/js/components/ui-components/modal.js +386 -0
  144. package/framework/js/components/ui-components/notifications.js +145 -0
  145. package/framework/js/components/ui-components/progress.js +88 -0
  146. package/framework/js/components/ui-components/quote.js +41 -0
  147. package/framework/js/components/ui-components/stats.js +33 -0
  148. package/framework/js/components/ui-components/steps.js +41 -0
  149. package/framework/js/components/ui-components/tabs.js +255 -0
  150. package/framework/js/components/ui-components/timeline.js +42 -0
  151. package/framework/js/components/ui-components/toggle-group.js +73 -0
  152. package/framework/js/components/ui-components/tooltip.js +458 -0
  153. package/framework/js/components/ui-components/value-display.js +133 -0
  154. package/framework/js/components/ui-components/video-player.js +686 -0
  155. package/framework/js/core/component-catalog.js +121 -0
  156. package/framework/js/core/event-bus.js +178 -0
  157. package/framework/js/core/interaction-catalog.js +149 -0
  158. package/framework/js/dev/runtime-linter.js +1725 -0
  159. package/framework/js/drivers/cmi5-driver.js +768 -0
  160. package/framework/js/drivers/driver-factory.js +77 -0
  161. package/framework/js/drivers/driver-interface.js +110 -0
  162. package/framework/js/drivers/http-driver-base.js +241 -0
  163. package/framework/js/drivers/lti-driver.js +508 -0
  164. package/framework/js/drivers/proxy-driver.js +444 -0
  165. package/framework/js/drivers/scorm-12-driver.js +560 -0
  166. package/framework/js/drivers/scorm-2004-driver.js +775 -0
  167. package/framework/js/drivers/scorm-driver-base.js +112 -0
  168. package/framework/js/engagement/engagement-manager.js +404 -0
  169. package/framework/js/engagement/engagement-progress.js +191 -0
  170. package/framework/js/engagement/engagement-trackers.js +215 -0
  171. package/framework/js/engagement/requirement-strategies.js +268 -0
  172. package/framework/js/main.js +727 -0
  173. package/framework/js/managers/accessibility-manager.js +499 -0
  174. package/framework/js/managers/assessment-manager.js +230 -0
  175. package/framework/js/managers/audio-manager.js +944 -0
  176. package/framework/js/managers/comment-manager.js +88 -0
  177. package/framework/js/managers/flag-manager.js +86 -0
  178. package/framework/js/managers/interaction-manager.js +254 -0
  179. package/framework/js/managers/interaction-registry.js +96 -0
  180. package/framework/js/managers/objective-manager.js +423 -0
  181. package/framework/js/managers/score-manager.js +441 -0
  182. package/framework/js/managers/video-manager.js +536 -0
  183. package/framework/js/navigation/Breadcrumbs.js +234 -0
  184. package/framework/js/navigation/NavigationActions.js +1132 -0
  185. package/framework/js/navigation/NavigationState.js +276 -0
  186. package/framework/js/navigation/NavigationUI.js +574 -0
  187. package/framework/js/navigation/document-gallery.js +357 -0
  188. package/framework/js/navigation/navigation-helpers.js +175 -0
  189. package/framework/js/navigation/navigation-validators.js +174 -0
  190. package/framework/js/state/index.js +8 -0
  191. package/framework/js/state/lms-connection.js +482 -0
  192. package/framework/js/state/lms-error-utils.js +58 -0
  193. package/framework/js/state/state-commits.js +200 -0
  194. package/framework/js/state/state-domains.js +86 -0
  195. package/framework/js/state/state-manager.js +502 -0
  196. package/framework/js/state/state-validation.js +311 -0
  197. package/framework/js/state/transaction-log.js +41 -0
  198. package/framework/js/state/xapi-statement-service.js +325 -0
  199. package/framework/js/utilities/access-control.js +99 -0
  200. package/framework/js/utilities/breakpoint-manager.js +315 -0
  201. package/framework/js/utilities/canvas-slide.js +35 -0
  202. package/framework/js/utilities/conditional-display.js +388 -0
  203. package/framework/js/utilities/course-channel.js +214 -0
  204. package/framework/js/utilities/course-helpers.js +420 -0
  205. package/framework/js/utilities/data-reporter.js +273 -0
  206. package/framework/js/utilities/error-reporter.js +313 -0
  207. package/framework/js/utilities/hotspot-helper.js +341 -0
  208. package/framework/js/utilities/icons.js +348 -0
  209. package/framework/js/utilities/logger.js +92 -0
  210. package/framework/js/utilities/markdown-renderer.js +45 -0
  211. package/framework/js/utilities/scroll-tracker.js +68 -0
  212. package/framework/js/utilities/ui-initializer.js +146 -0
  213. package/framework/js/utilities/utilities.js +293 -0
  214. package/framework/js/utilities/view-manager.js +227 -0
  215. package/framework/js/validation/html-validators.js +422 -0
  216. package/framework/js/validation/scorm-validators.js +438 -0
  217. package/framework/js/vendor/pipwerks.js +931 -0
  218. package/framework/scripts/generate-narration.js +629 -0
  219. package/framework/scripts/tts-providers/azure-provider.js +178 -0
  220. package/framework/scripts/tts-providers/base-provider.js +81 -0
  221. package/framework/scripts/tts-providers/deepgram-provider.js +135 -0
  222. package/framework/scripts/tts-providers/elevenlabs-provider.js +148 -0
  223. package/framework/scripts/tts-providers/google-provider.js +272 -0
  224. package/framework/scripts/tts-providers/index.js +158 -0
  225. package/framework/scripts/tts-providers/openai-provider.js +143 -0
  226. package/framework/version.json +63 -0
  227. package/lib/authoring-api.js +919 -0
  228. package/lib/build-linter.js +450 -0
  229. package/lib/build-packaging.js +186 -0
  230. package/lib/build.js +88 -0
  231. package/lib/cloud.js +691 -0
  232. package/lib/convert.js +341 -0
  233. package/lib/course-parser.js +936 -0
  234. package/lib/course-writer.js +258 -0
  235. package/lib/create.js +248 -0
  236. package/lib/css-index.js +237 -0
  237. package/lib/dev.js +51 -0
  238. package/lib/export-content.js +1246 -0
  239. package/lib/headless-browser.js +413 -0
  240. package/lib/import.js +377 -0
  241. package/lib/index.js +80 -0
  242. package/lib/info.js +79 -0
  243. package/lib/interaction-formatters.js +568 -0
  244. package/lib/manifest/cmi5-manifest.js +63 -0
  245. package/lib/manifest/lti-tool-config.js +53 -0
  246. package/lib/manifest/manifest-factory.js +99 -0
  247. package/lib/manifest/scorm-12-manifest.js +61 -0
  248. package/lib/manifest/scorm-2004-manifest.js +94 -0
  249. package/lib/manifest/scorm-proxy-manifest.js +104 -0
  250. package/lib/manifest-parser.js +96 -0
  251. package/lib/mcp-prompts.js +753 -0
  252. package/lib/mcp-server.js +316 -0
  253. package/lib/narration.js +53 -0
  254. package/lib/pdf-structure.js +142 -0
  255. package/lib/preview-export.js +231 -0
  256. package/lib/preview-routes-api.js +662 -0
  257. package/lib/preview-routes-editing.js +159 -0
  258. package/lib/preview-routes-lms.js +230 -0
  259. package/lib/preview-server.js +564 -0
  260. package/lib/project-utils.js +269 -0
  261. package/lib/proxy-templates/proxy.html +68 -0
  262. package/lib/proxy-templates/scorm-bridge.js +112 -0
  263. package/lib/scaffold.js +193 -0
  264. package/lib/schema-extractor.js +361 -0
  265. package/lib/slide-source-editor.js +586 -0
  266. package/lib/stub-player/app-viewer.js +195 -0
  267. package/lib/stub-player/app.js +370 -0
  268. package/lib/stub-player/catalog-panel.js +312 -0
  269. package/lib/stub-player/config-panel.js +1303 -0
  270. package/lib/stub-player/content-generator.js +586 -0
  271. package/lib/stub-player/content-viewer.js +173 -0
  272. package/lib/stub-player/debug-panel.js +420 -0
  273. package/lib/stub-player/edit-mode.js +922 -0
  274. package/lib/stub-player/edit-utils.js +400 -0
  275. package/lib/stub-player/header-bar.js +354 -0
  276. package/lib/stub-player/interaction-editor.js +210 -0
  277. package/lib/stub-player/interactions-panel.js +565 -0
  278. package/lib/stub-player/lms-api.js +1094 -0
  279. package/lib/stub-player/login-screen.js +74 -0
  280. package/lib/stub-player/outline-mode.js +689 -0
  281. package/lib/stub-player/styles/_assessments-panel.css +245 -0
  282. package/lib/stub-player/styles/_base.css +89 -0
  283. package/lib/stub-player/styles/_catalog-icons.css +96 -0
  284. package/lib/stub-player/styles/_catalog-panel.css +291 -0
  285. package/lib/stub-player/styles/_config-panel.css +636 -0
  286. package/lib/stub-player/styles/_content-viewer.css +834 -0
  287. package/lib/stub-player/styles/_debug-panel.css +576 -0
  288. package/lib/stub-player/styles/_edit-mode.css +128 -0
  289. package/lib/stub-player/styles/_header-bar.css +343 -0
  290. package/lib/stub-player/styles/_interaction-editor.css +140 -0
  291. package/lib/stub-player/styles/_interactions-panel.css +1038 -0
  292. package/lib/stub-player/styles/_login-screen.css +102 -0
  293. package/lib/stub-player/styles/_outline-mode.css +752 -0
  294. package/lib/stub-player/styles.css +15 -0
  295. package/lib/stub-player.js +160 -0
  296. package/lib/test-data-reporting.js +176 -0
  297. package/lib/test-error-reporting.js +146 -0
  298. package/lib/token.js +86 -0
  299. package/lib/upgrade.js +257 -0
  300. package/lib/validation-rules.js +517 -0
  301. package/lib/vite-plugin-content-discovery.js +296 -0
  302. package/package.json +108 -0
  303. package/schemas/XMLSchema.dtd +402 -0
  304. package/schemas/adlcp_v1p3.xsd +111 -0
  305. package/schemas/adlnav_v1p3.xsd +61 -0
  306. package/schemas/adlseq_v1p3.xsd +93 -0
  307. package/schemas/common/anyElement.xsd +27 -0
  308. package/schemas/common/dataTypes.xsd +138 -0
  309. package/schemas/common/elementNames.xsd +767 -0
  310. package/schemas/common/elementTypes.xsd +786 -0
  311. package/schemas/common/rootElement.xsd +31 -0
  312. package/schemas/common/vocabTypes.xsd +345 -0
  313. package/schemas/common/vocabValues.xsd +257 -0
  314. package/schemas/datatypes.dtd +203 -0
  315. package/schemas/ims_xml.xsd +35 -0
  316. package/schemas/imscp_v1p1.xsd +368 -0
  317. package/schemas/imsss_v1p0.xsd +67 -0
  318. package/schemas/imsss_v1p0auxresource.xsd +19 -0
  319. package/schemas/imsss_v1p0control.xsd +20 -0
  320. package/schemas/imsss_v1p0delivery.xsd +17 -0
  321. package/schemas/imsss_v1p0limit.xsd +47 -0
  322. package/schemas/imsss_v1p0objective.xsd +67 -0
  323. package/schemas/imsss_v1p0random.xsd +16 -0
  324. package/schemas/imsss_v1p0rollup.xsd +46 -0
  325. package/schemas/imsss_v1p0seqrule.xsd +108 -0
  326. package/schemas/imsss_v1p0util.xsd +94 -0
  327. package/schemas/license.txt +17 -0
  328. package/schemas/lom.xsd +102 -0
  329. package/schemas/lomCustom.xsd +62 -0
  330. package/schemas/lomLoose.xsd +62 -0
  331. package/schemas/lomStrict.xsd +62 -0
  332. package/schemas/xml.xsd +81 -0
  333. package/template/.env.example +92 -0
  334. package/template/course/assets/audio/example-intro.mp3 +0 -0
  335. package/template/course/assets/audio/example-ui-demo--compact-player.mp3 +0 -0
  336. package/template/course/assets/audio/example-ui-demo--demo-modal.mp3 +0 -0
  337. package/template/course/assets/audio/example-ui-demo--full-player.mp3 +0 -0
  338. package/template/course/assets/docs/example_md_1.md +39 -0
  339. package/template/course/assets/docs/example_md_2.md +41 -0
  340. package/template/course/assets/docs/example_pdf_1_thumbnail.png +0 -0
  341. package/template/course/assets/docs/example_pdf_2.pdf +0 -0
  342. package/template/course/assets/images/course-architecture.svg +36 -0
  343. package/template/course/assets/images/logo.svg +14 -0
  344. package/template/course/assets/widgets/counter-demo.html +190 -0
  345. package/template/course/assets/widgets/gravity-painter.html +384 -0
  346. package/template/course/course-config.js +539 -0
  347. package/template/course/icons.js +19 -0
  348. package/template/course/interactions/PLUGIN_GUIDE.md +97 -0
  349. package/template/course/slides/example-course-structure.js +138 -0
  350. package/template/course/slides/example-final-exam.js +144 -0
  351. package/template/course/slides/example-finishing.js +127 -0
  352. package/template/course/slides/example-interactions-showcase.js +615 -0
  353. package/template/course/slides/example-preview-tour.js +129 -0
  354. package/template/course/slides/example-remedial.js +143 -0
  355. package/template/course/slides/example-summary.js +103 -0
  356. package/template/course/slides/example-ui-showcase.js +1805 -0
  357. package/template/course/slides/example-welcome.js +123 -0
  358. package/template/course/slides/example-workflow.js +140 -0
  359. package/template/course/theme.css +165 -0
  360. package/template/eslint.config.js +47 -0
  361. package/template/package.json +28 -0
  362. package/template/vite.config.js +339 -0
@@ -0,0 +1,322 @@
1
+ /**
2
+ * @file AssessmentState.js
3
+ * @description Manages all state interactions for a single assessment instance.
4
+ * This module provides a factory to create a state manager for an assessment,
5
+ * encapsulating all interactions with the global StateManager.
6
+ */
7
+
8
+ import { deepClone } from '../utilities/utilities.js';
9
+ import { logger } from '../utilities/logger.js';
10
+
11
+ /**
12
+ * Utility function to get domain key for an assessment.
13
+ * Centralizes the domain key pattern used throughout the assessment system.
14
+ * @param {string} assessmentId - The unique ID of the assessment
15
+ * @returns {string} The domain key for state manager
16
+ */
17
+ function getAssessmentDomainKey(assessmentId) {
18
+ return `assessment_${assessmentId}`;
19
+ }
20
+
21
+ /**
22
+ * Creates a state management object for a single assessment instance.
23
+ * All state is persisted through the stateManager - no in-memory state cache.
24
+ * This ensures a single source of truth and prevents desynchronization.
25
+ *
26
+ * @param {object} config - The assessment's configuration object, must include an `id`.
27
+ * @param {object} stateManager - The global state manager instance.
28
+ * @returns {object} An assessment-specific state management object.
29
+ */
30
+ export function createAssessmentState(config, stateManager) {
31
+ if (!config || !config.id) {
32
+ const errorMessage = 'AssessmentState requires a config object with an id';
33
+ logger.error(errorMessage, { domain: 'assessment', operation: 'createAssessmentState' });
34
+ throw new Error(errorMessage);
35
+ }
36
+ if (!stateManager) {
37
+ const errorMessage = `[AssessmentState:${config.id}] StateManager instance is required`;
38
+ logger.error(errorMessage, { domain: 'assessment', operation: 'createAssessmentState', assessmentId: config.id });
39
+ throw new Error(errorMessage);
40
+ }
41
+
42
+ const DOMAIN_KEY = getAssessmentDomainKey(config.id);
43
+ const SOURCE = 'assessment-state';
44
+
45
+ // Default session structure (used for initialization only)
46
+ const DEFAULT_SESSION = {
47
+ currentView: 'intro', // 'intro', 'question', 'review', 'results'
48
+ currentQuestionIndex: 0,
49
+ startTime: null,
50
+ submitted: false,
51
+ attemptNumber: 1,
52
+ responses: {},
53
+ selectedQuestions: null, // Array of question IDs only (for bank mode) - NOT full configs to save space
54
+ reviewReached: false, // True once user reaches review screen (allows jump to review)
55
+ };
56
+
57
+ function _getAssessmentDomainState() {
58
+ return stateManager.getDomainState(DOMAIN_KEY) || {};
59
+ }
60
+
61
+ async function _setAssessmentDomainState(domainState, source = SOURCE) {
62
+ await stateManager.setDomainState(DOMAIN_KEY, domainState, { source });
63
+ }
64
+
65
+ function _ensureSessionExists() {
66
+ const domainState = _getAssessmentDomainState();
67
+ if (!domainState.session) {
68
+ domainState.session = { ...DEFAULT_SESSION };
69
+ }
70
+ return domainState;
71
+ }
72
+
73
+ function _resolveAttemptNumber(summaryRecord) {
74
+ if (summaryRecord?.attempts > 0) {
75
+ return summaryRecord.attempts + 1;
76
+ }
77
+ return 1;
78
+ }
79
+
80
+ // === Summary Operations ===
81
+
82
+ function getSummary() {
83
+ const domainState = _getAssessmentDomainState();
84
+ return domainState.summary ? deepClone(domainState.summary) : null;
85
+ }
86
+
87
+ async function updateSummary(patch = {}, source = SOURCE) {
88
+ const domainState = _getAssessmentDomainState();
89
+ const currentSummary = domainState.summary || {};
90
+ domainState.summary = { ...currentSummary, ...patch };
91
+ await _setAssessmentDomainState(domainState, source);
92
+ }
93
+
94
+ // === Session Operations ===
95
+
96
+ function getSession() {
97
+ const domainState = _getAssessmentDomainState();
98
+ return domainState.session ? deepClone(domainState.session) : null;
99
+ }
100
+
101
+ async function initializeSession() {
102
+ const domainState = _ensureSessionExists();
103
+ await _setAssessmentDomainState(domainState, `${SOURCE}:init`);
104
+ }
105
+
106
+ async function updateSession(patch = {}, source = SOURCE) {
107
+ const domainState = _ensureSessionExists();
108
+ domainState.session = { ...domainState.session, ...patch };
109
+ await _setAssessmentDomainState(domainState, source);
110
+ }
111
+
112
+ async function clearSession(source = `${SOURCE}:clear`) {
113
+ const domainState = _getAssessmentDomainState();
114
+ delete domainState.session;
115
+ await _setAssessmentDomainState(domainState, source);
116
+ }
117
+
118
+ // === State Property Getters ===
119
+
120
+ function getCurrentView() {
121
+ const session = getSession();
122
+ return session?.currentView || 'intro';
123
+ }
124
+
125
+ function getCurrentQuestionIndex() {
126
+ const session = getSession();
127
+ return session?.currentQuestionIndex || 0;
128
+ }
129
+
130
+ function getStartTime() {
131
+ const session = getSession();
132
+ return session?.startTime || null;
133
+ }
134
+
135
+ function isSubmitted() {
136
+ const session = getSession();
137
+ return session?.submitted || false;
138
+ }
139
+
140
+ function getAttemptNumber() {
141
+ const session = getSession();
142
+ const summary = getSummary();
143
+ if (session?.attemptNumber) {
144
+ return session.attemptNumber;
145
+ }
146
+ return _resolveAttemptNumber(summary);
147
+ }
148
+
149
+ function getLastResults() {
150
+ const summary = getSummary();
151
+ return summary?.lastResults || null;
152
+ }
153
+
154
+ // === State Property Setters ===
155
+
156
+ async function setCurrentView(view, source = `${SOURCE}:view`) {
157
+ await updateSession({ currentView: view }, source);
158
+ }
159
+
160
+ async function setCurrentQuestionIndex(index, source = `${SOURCE}:question`) {
161
+ await updateSession({ currentQuestionIndex: index }, source);
162
+ }
163
+
164
+ async function setStartTime(timestamp, source = `${SOURCE}:start`) {
165
+ await updateSession({ startTime: timestamp }, source);
166
+ }
167
+
168
+ async function setSubmitted(submitted, source = `${SOURCE}:submit`) {
169
+ await updateSession({ submitted }, source);
170
+ }
171
+
172
+ async function setAttemptNumber(attemptNumber, source = `${SOURCE}:attempt`) {
173
+ await updateSession({ attemptNumber }, source);
174
+ }
175
+
176
+ // === Response Operations ===
177
+
178
+ function getResponse(questionIndex) {
179
+ const session = getSession();
180
+ if (!session?.responses) {
181
+ return null;
182
+ }
183
+ const key = String(questionIndex);
184
+ if (!Object.prototype.hasOwnProperty.call(session.responses, key)) {
185
+ return null;
186
+ }
187
+ return deepClone(session.responses[key]);
188
+ }
189
+
190
+ async function saveResponse(questionIndex, response) {
191
+ const key = String(questionIndex);
192
+ const storedResponse = response === undefined ? null : deepClone(response);
193
+
194
+ const domainState = _ensureSessionExists();
195
+ domainState.session.responses = domainState.session.responses || {};
196
+ domainState.session.responses[key] = storedResponse;
197
+
198
+ await _setAssessmentDomainState(domainState, `${SOURCE}:response`);
199
+ }
200
+
201
+ // === Selected Questions Operations ===
202
+
203
+ function getSelectedQuestions() {
204
+ const session = getSession();
205
+ return session?.selectedQuestions ? deepClone(session.selectedQuestions) : null;
206
+ }
207
+
208
+ async function setSelectedQuestions(questions) {
209
+ const domainState = _ensureSessionExists();
210
+ if (questions === null) {
211
+ domainState.session.selectedQuestions = null;
212
+ await _setAssessmentDomainState(domainState, `${SOURCE}:selected-questions`);
213
+ return;
214
+ }
215
+
216
+ if (!Array.isArray(questions)) {
217
+ throw new Error(`[AssessmentState:${config.id}] setSelectedQuestions expects an array or null`);
218
+ }
219
+
220
+ // CRITICAL: Only store question IDs to meet SCORM suspend_data size limits
221
+ // Many LMS implementations restrict cmi.suspend_data to 4096 characters
222
+ // Storing full question configs (~7KB) exceeds this limit and causes SCORM error 409
223
+ // Question IDs (~100-200 bytes) stay well within limits
224
+ // Full configs are reconstructed from slide config on resume
225
+ domainState.session.selectedQuestions = questions.map((q, index) => {
226
+ if (typeof q === 'string') {
227
+ return q;
228
+ }
229
+ if (q && typeof q.id === 'string') {
230
+ return q.id;
231
+ }
232
+ throw new Error(`[AssessmentState:${config.id}] selectedQuestions[${index}] must be a question object with id or a string ID`);
233
+ });
234
+ await _setAssessmentDomainState(domainState, `${SOURCE}:selected-questions`);
235
+ }
236
+
237
+ async function archiveDiscardedResponses(questionConfigs, responses) {
238
+ // OPTIMIZATION: Only archive if feature is enabled in config
239
+ // Defaults to disabled to reduce suspend_data bloat
240
+ const enableArchive = config.settings?.archiveDiscardedAttempts || false;
241
+
242
+ if (!enableArchive) {
243
+ logger.debug(`[AssessmentState:${config.id}] Discarded attempt archive disabled (to enable, set settings.archiveDiscardedAttempts: true)`);
244
+ return;
245
+ }
246
+
247
+ const domainState = _getAssessmentDomainState();
248
+
249
+ // Create archive array if it doesn't exist
250
+ if (!domainState.discardedAttempts) {
251
+ domainState.discardedAttempts = [];
252
+ }
253
+
254
+ // Archive current attempt data (minimal - only IDs and responses, not full question text)
255
+ const archiveEntry = {
256
+ timestamp: Date.now(),
257
+ attemptNumber: getAttemptNumber(),
258
+ // OPTIMIZATION: Store only question IDs and metadata, not full text
259
+ questions: questionConfigs.map(q => ({
260
+ id: q.id,
261
+ type: q.type,
262
+ bankId: q._meta?.bankId,
263
+ originalIndex: q._meta?.originalIndex
264
+ })),
265
+ responses: deepClone(responses)
266
+ };
267
+
268
+ domainState.discardedAttempts.push(archiveEntry);
269
+
270
+ // Limit archive size to prevent SCORM data bloat (keep last 5)
271
+ if (domainState.discardedAttempts.length > 5) {
272
+ domainState.discardedAttempts = domainState.discardedAttempts.slice(-5);
273
+ }
274
+
275
+ await _setAssessmentDomainState(domainState, `${SOURCE}:archive`);
276
+ logger.debug(`[AssessmentState:${config.id}] Archived discarded attempt #${archiveEntry.attemptNumber}`);
277
+ }
278
+
279
+ function getDiscardedAttempts() {
280
+ const domainState = _getAssessmentDomainState();
281
+ return domainState.discardedAttempts ? deepClone(domainState.discardedAttempts) : [];
282
+ }
283
+
284
+ return {
285
+ // Summary operations
286
+ getSummary,
287
+ updateSummary,
288
+
289
+ // Session operations
290
+ getSession,
291
+ initializeSession,
292
+ updateSession,
293
+ clearSession,
294
+
295
+ // State property getters
296
+ getCurrentView,
297
+ getCurrentQuestionIndex,
298
+ getStartTime,
299
+ isSubmitted,
300
+ getAttemptNumber,
301
+ getLastResults,
302
+
303
+ // State property setters
304
+ setCurrentView,
305
+ setCurrentQuestionIndex,
306
+ setStartTime,
307
+ setSubmitted,
308
+ setAttemptNumber,
309
+
310
+ // Response operations
311
+ getResponse,
312
+ saveResponse,
313
+
314
+ // Selected questions operations
315
+ getSelectedQuestions,
316
+ setSelectedQuestions,
317
+
318
+ // Discarded attempts archive
319
+ archiveDiscardedResponses,
320
+ getDiscardedAttempts,
321
+ };
322
+ }