tessera-learn 0.0.13 → 0.2.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 (68) hide show
  1. package/AGENTS.md +1794 -0
  2. package/README.md +5 -5
  3. package/dist/{validation-B-xTvM9B.js → audit-BA5o0ick.js} +605 -269
  4. package/dist/audit-BA5o0ick.js.map +1 -0
  5. package/dist/build-commands-C0OnV-Vg.js +27 -0
  6. package/dist/build-commands-C0OnV-Vg.js.map +1 -0
  7. package/dist/inline-config-CroQ-_2Y.js +31 -0
  8. package/dist/inline-config-CroQ-_2Y.js.map +1 -0
  9. package/dist/plugin/cli.d.ts +9 -1
  10. package/dist/plugin/cli.d.ts.map +1 -0
  11. package/dist/plugin/cli.js +326 -17
  12. package/dist/plugin/cli.js.map +1 -1
  13. package/dist/plugin/index.d.ts +1 -1
  14. package/dist/plugin/index.d.ts.map +1 -1
  15. package/dist/plugin/index.js +2 -763
  16. package/dist/plugin-W_rk3Pit.js +731 -0
  17. package/dist/plugin-W_rk3Pit.js.map +1 -0
  18. package/package.json +21 -9
  19. package/src/components/FillInTheBlank.svelte +2 -2
  20. package/src/components/Matching.svelte +2 -2
  21. package/src/components/MultipleChoice.svelte +2 -2
  22. package/src/components/RevealModal.svelte +48 -103
  23. package/src/components/Sorting.svelte +2 -2
  24. package/src/components/util.ts +9 -0
  25. package/src/plugin/a11y/audit.ts +40 -8
  26. package/src/plugin/a11y-cli.ts +39 -22
  27. package/src/plugin/ast.ts +276 -0
  28. package/src/plugin/build-commands.ts +31 -0
  29. package/src/plugin/cli.ts +96 -21
  30. package/src/plugin/course-root.ts +98 -0
  31. package/src/plugin/duplicate-cli.ts +74 -0
  32. package/src/plugin/index.ts +87 -122
  33. package/src/plugin/inline-config.ts +54 -0
  34. package/src/plugin/manifest.ts +103 -136
  35. package/src/plugin/new-cli.ts +51 -0
  36. package/src/plugin/package-root.ts +24 -0
  37. package/src/plugin/project-name.ts +29 -0
  38. package/src/plugin/quiz.ts +8 -9
  39. package/src/plugin/template-copy.ts +43 -0
  40. package/src/plugin/validate-cli.ts +30 -0
  41. package/src/plugin/validation.ts +152 -244
  42. package/src/runtime/App.svelte +11 -97
  43. package/src/runtime/Sidebar.svelte +3 -1
  44. package/src/runtime/adapters/cmi5.ts +6 -10
  45. package/src/runtime/adapters/format.ts +6 -0
  46. package/src/runtime/adapters/retry.ts +1 -1
  47. package/src/runtime/adapters/scorm2004.ts +2 -4
  48. package/src/runtime/branding.ts +90 -0
  49. package/src/runtime/defaults.ts +3 -0
  50. package/src/runtime/hooks.svelte.ts +16 -53
  51. package/src/runtime/interaction-format.ts +3 -8
  52. package/src/runtime/progress.svelte.ts +47 -83
  53. package/src/runtime/xapi/derive-actor.ts +41 -48
  54. package/src/runtime/xapi/publisher.ts +14 -14
  55. package/src/runtime/xapi/setup.ts +39 -46
  56. package/templates/course/course.config.js +11 -0
  57. package/templates/course/layout.svelte +116 -0
  58. package/templates/course/pages/01-getting-started/01-welcome/_meta.js +1 -0
  59. package/templates/course/pages/01-getting-started/01-welcome/welcome.svelte +19 -0
  60. package/templates/course/pages/01-getting-started/_meta.js +1 -0
  61. package/templates/course/styles/custom.css +5 -0
  62. package/dist/audit-BBJpQGqb.js +0 -204
  63. package/dist/audit-BBJpQGqb.js.map +0 -1
  64. package/dist/plugin/a11y-cli.d.ts +0 -1
  65. package/dist/plugin/a11y-cli.js +0 -36
  66. package/dist/plugin/a11y-cli.js.map +0 -1
  67. package/dist/plugin/index.js.map +0 -1
  68. package/dist/validation-B-xTvM9B.js.map +0 -1
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugin-W_rk3Pit.js","names":[],"sources":["../src/runtime/slugify.ts","../src/plugin/export.ts","../src/plugin/override-plugin.ts","../src/plugin/layout.ts","../src/plugin/quiz.ts","../src/plugin/index.ts"],"sourcesContent":["/**\n * Slugify a string for use as a URL-safe / filename-safe identifier.\n * \"My Course Title\" → \"my-course-title\"\n *\n * Shared by the runtime (`WebAdapter` localStorage key) and the build-time\n * exporter (`runExport` zip filename). Both want identical, deterministic\n * output so a course's storage key matches its package name.\n */\nexport function slugify(text: string): string {\n return text\n .toLowerCase()\n .trim()\n .replace(/[^\\w\\s-]/g, '')\n .replace(/[\\s_]+/g, '-')\n .replace(/-+/g, '-')\n .replace(/^-|-$/g, '');\n}\n","import {\n existsSync,\n readdirSync,\n statSync,\n writeFileSync,\n unlinkSync,\n} from 'node:fs';\nimport { resolve } from 'node:path';\nimport { createWriteStream } from 'node:fs';\nimport { createHash } from 'node:crypto';\nimport { ZipArchive } from 'archiver';\nimport { slugify } from '../runtime/slugify.js';\n\n// ---------- Types ----------\n\ninterface ExportConfig {\n title: string;\n description?: string;\n version?: string;\n scoring?: { passingScore?: number };\n completion?: { mode?: 'quiz' | 'percentage' };\n export?: { standard?: string };\n}\n\n// ---------- Helpers ----------\n\nfunction escapeXml(str: string): string {\n return str\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;')\n .replace(/'/g, '&apos;');\n}\n\n/**\n * Recursively collect all file paths relative to a directory.\n */\nfunction collectFiles(dir: string, base: string = ''): string[] {\n const files: string[] = [];\n if (!existsSync(dir)) return files;\n\n for (const entry of readdirSync(dir)) {\n const fullPath = resolve(dir, entry);\n const relPath = base ? `${base}/${entry}` : entry;\n if (statSync(fullPath).isDirectory()) {\n files.push(...collectFiles(fullPath, relPath));\n } else {\n files.push(relPath);\n }\n }\n return files;\n}\n\n/**\n * Derive a stable URN IRI from a seed string. cmi5 §13.1 / xs:anyURI\n * require course / AU ids to be IRIs — bare hex or UUID-shaped strings\n * (without correct version/variant bits) aren't conformant URNs and may\n * be rejected by strict LMS importers.\n *\n * Hash the seed so the id survives rebuilds, then format as\n * `urn:tessera:<kind>:<hex>`. The same seed always produces the same\n * IRI, so existing LRS records are not orphaned by re-export.\n */\nfunction stableUrn(kind: 'course' | 'au', seed: string): string {\n const h = createHash('sha256').update(seed).digest('hex');\n // 32 hex chars (128 bits of entropy) is plenty; trim to keep ids short.\n return `urn:tessera:${kind}:${h.slice(0, 32)}`;\n}\n\nfunction formatSize(bytes: number): string {\n if (bytes < 1024) return `${bytes} B`;\n if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;\n return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\n}\n\n// ---------- Manifest Generators ----------\n\n/** Per-version XML differences in imsmanifest.xml between SCORM 1.2 and 2004. */\ninterface ScormManifestDialect {\n rootNs: string;\n adlcpNs: string;\n schemaversion: string;\n /** Attribute name on <resource>: SCORM 1.2 uses lowercase, 2004 uses camelCase. */\n scormTypeAttr: 'scormtype' | 'scormType';\n /** Whitespace-separated namespace+XSD pairs for xsi:schemaLocation. */\n schemaLocation: string;\n}\n\nconst SCORM_DIALECTS: Record<'1.2' | '2004', ScormManifestDialect> = {\n '1.2': {\n rootNs: 'http://www.imsproject.org/xsd/imscp_rootv1p1p2',\n adlcpNs: 'http://www.adlnet.org/xsd/adlcp_rootv1p2',\n schemaversion: '1.2',\n scormTypeAttr: 'scormtype',\n schemaLocation:\n 'http://www.imsproject.org/xsd/imscp_rootv1p1p2 imscp_rootv1p1p2.xsd ' +\n 'http://www.imsglobal.org/xsd/imsmd_rootv1p2p1 imsmd_rootv1p2p1.xsd ' +\n 'http://www.adlnet.org/xsd/adlcp_rootv1p2 adlcp_rootv1p2.xsd',\n },\n '2004': {\n rootNs: 'http://www.imsglobal.org/xsd/imscp_v1p1',\n adlcpNs: 'http://www.adlnet.org/xsd/adlcp_v1p3',\n schemaversion: '2004 4th Edition',\n scormTypeAttr: 'scormType',\n schemaLocation:\n 'http://www.imsglobal.org/xsd/imscp_v1p1 imscp_v1p1.xsd ' +\n 'http://www.adlnet.org/xsd/adlcp_v1p3 adlcp_v1p3.xsd',\n },\n};\n\nexport function generateScormManifest(\n version: '1.2' | '2004',\n config: ExportConfig,\n distDir: string,\n): string {\n const dialect = SCORM_DIALECTS[version];\n const title = escapeXml(config.title || 'Tessera Course');\n const files = collectFiles(distDir);\n const fileElements = files\n .map((f) => ` <file href=\"${escapeXml(f)}\" />`)\n .join('\\n');\n\n return `<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<manifest identifier=\"tessera-course\" version=\"1.0\"\n xmlns=\"${dialect.rootNs}\"\n xmlns:adlcp=\"${dialect.adlcpNs}\"\n xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n xsi:schemaLocation=\"${dialect.schemaLocation}\">\n <metadata>\n <schema>ADL SCORM</schema>\n <schemaversion>${dialect.schemaversion}</schemaversion>\n </metadata>\n <organizations default=\"org-1\">\n <organization identifier=\"org-1\">\n <title>${title}</title>\n <item identifier=\"item-1\" identifierref=\"res-1\">\n <title>${title}</title>\n </item>\n </organization>\n </organizations>\n <resources>\n <resource identifier=\"res-1\" type=\"webcontent\" adlcp:${dialect.scormTypeAttr}=\"sco\" href=\"index.html\">\n${fileElements}\n </resource>\n </resources>\n</manifest>`;\n}\n\nexport function generateSCORM12Manifest(\n config: ExportConfig,\n distDir: string,\n): string {\n return generateScormManifest('1.2', config, distDir);\n}\n\nexport function generateSCORM2004Manifest(\n config: ExportConfig,\n distDir: string,\n): string {\n return generateScormManifest('2004', config, distDir);\n}\n\nexport function generateCMI5Xml(config: ExportConfig): string {\n const title = escapeXml(config.title || 'Tessera Course');\n const description = escapeXml(config.description || '');\n // Derive stable IDs from the course title so they survive rebuilds without\n // orphaning existing learner records in the LRS.\n const courseId = stableUrn('course', `tessera-course:${config.title || ''}`);\n const auId = stableUrn('au', `tessera-au:${config.title || ''}`);\n // cmi5 §10.2.4 caps masteryScore at 4 decimals; avoid float drift like 0.7000000000000001.\n const masteryScore = Number(\n ((config.scoring?.passingScore ?? 70) / 100).toFixed(4),\n );\n // cmi5 §13.1.4 — `moveOn` decides which verb(s) the LMS treats as\n // satisfying the AU. For graded courses (completion gated on a quiz)\n // a learner who completes without passing should NOT receive credit, so\n // the LMS needs both a Completed AND a Passed before satisfaction.\n // Percentage-mode courses don't surface pass/fail, so completion alone\n // is the right signal.\n const moveOn =\n config.completion?.mode === 'quiz' ? 'CompletedAndPassed' : 'Completed';\n\n return `<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<courseStructure xmlns=\"https://w3id.org/xapi/profiles/cmi5/v1/CourseStructure.xsd\">\n <course id=\"${courseId}\">\n <title><langstring lang=\"en-US\">${title}</langstring></title>\n <description><langstring lang=\"en-US\">${description}</langstring></description>\n </course>\n <au id=\"${auId}\" launchMethod=\"AnyWindow\" moveOn=\"${moveOn}\" masteryScore=\"${masteryScore}\">\n <title><langstring lang=\"en-US\">${title}</langstring></title>\n <description><langstring lang=\"en-US\">${description}</langstring></description>\n <url>index.html</url>\n </au>\n</courseStructure>`;\n}\n\n// ---------- ZIP Packaging ----------\n\nexport async function createZip(\n distDir: string,\n outputPath: string,\n): Promise<number> {\n return new Promise((res, reject) => {\n const output = createWriteStream(outputPath);\n const archive = new ZipArchive({ zlib: { level: 9 } });\n\n output.on('close', () => {\n res(archive.pointer());\n });\n output.on('error', reject);\n archive.on('error', reject);\n\n archive.pipe(output);\n archive.directory(distDir, false);\n void archive.finalize();\n });\n}\n\n// ---------- Main Export ----------\n\n/**\n * Run the export process after Vite build completes.\n * Writes manifest XML into dist/, then packages into ZIP if needed.\n */\n/** Remove any previously built zips for this package to prevent accumulation. */\nfunction cleanOldZips(projectRoot: string, slug: string): void {\n try {\n for (const f of readdirSync(projectRoot)) {\n if (f.startsWith(`${slug}-`) && f.endsWith('.zip')) {\n try {\n unlinkSync(resolve(projectRoot, f));\n } catch {}\n }\n }\n } catch {}\n}\n\n/** Packaged (zipped) export targets: which manifest file to write and how. */\nconst PACKAGED_EXPORTS: Record<\n 'scorm12' | 'scorm2004' | 'cmi5',\n {\n manifestFile: string;\n label: string;\n generate: (config: ExportConfig, distDir: string) => string;\n }\n> = {\n scorm12: {\n manifestFile: 'imsmanifest.xml',\n label: 'SCORM 1.2',\n generate: generateSCORM12Manifest,\n },\n scorm2004: {\n manifestFile: 'imsmanifest.xml',\n label: 'SCORM 2004',\n generate: generateSCORM2004Manifest,\n },\n cmi5: {\n manifestFile: 'cmi5.xml',\n label: 'CMI5',\n generate: (config) => generateCMI5Xml(config),\n },\n};\n\nexport async function runExport(\n projectRoot: string,\n config: ExportConfig,\n): Promise<void> {\n const distDir = resolve(projectRoot, 'dist');\n const standard = config.export?.standard || 'web';\n const slug = slugify(config.title || 'tessera-course') || 'tessera-course';\n const version = config.version || '1.0.0';\n const zipName = `${slug}-${version}.zip`;\n const zipPath = resolve(projectRoot, zipName);\n\n if (standard === 'web') {\n const files = collectFiles(distDir);\n let totalSize = 0;\n for (const f of files) totalSize += statSync(resolve(distDir, f)).size;\n console.log(`✓ Web export: dist/ (${formatSize(totalSize)})`);\n return;\n }\n\n const spec = PACKAGED_EXPORTS[standard as keyof typeof PACKAGED_EXPORTS];\n if (!spec) return; // unknown standard — the validator rejects these upstream\n\n writeFileSync(\n resolve(distDir, spec.manifestFile),\n spec.generate(config, distDir),\n 'utf-8',\n );\n cleanOldZips(projectRoot, slug);\n const zipSize = await createZip(distDir, zipPath);\n console.log(`✓ ${spec.label} export: ${zipName} (${formatSize(zipSize)})`);\n}\n","import type { Plugin, ResolvedConfig, ViteDevServer } from 'vite';\nimport { normalizePath } from 'vite';\nimport { existsSync } from 'node:fs';\nimport { resolve } from 'node:path';\n\nexport interface OverridePluginOptions {\n name: string;\n virtualId: string;\n projectFile: string;\n /** Built-in re-exported when the project file is absent; null export otherwise. */\n builtinFile?: string;\n}\n\n/**\n * A virtual module that resolves to a project-root override file when present,\n * and to the built-in (or a null export) otherwise. Shared by the layout and\n * quiz plugins — they differ only in the virtual id, file name, and built-in.\n */\nexport function createOverridePlugin({\n name,\n virtualId,\n projectFile,\n builtinFile,\n}: OverridePluginOptions): Plugin {\n const resolvedId = '\\0' + virtualId;\n const fallback = builtinFile\n ? `export { default } from '${normalizePath(builtinFile)}';`\n : 'export default null;';\n let filePath: string;\n\n return {\n name,\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n filePath = resolve(config.root, projectFile);\n },\n\n resolveId(id) {\n if (id === virtualId) return resolvedId;\n return null;\n },\n\n load(id) {\n if (id !== resolvedId) return null;\n if (existsSync(filePath)) {\n // Only watch when it exists — addWatchFile on a missing path makes\n // Vite's importAnalysis try to resolve it as a real import.\n this.addWatchFile(filePath);\n return `export { default } from '${normalizePath(filePath)}';`;\n }\n return fallback;\n },\n\n configureServer(server: ViteDevServer) {\n // Only add/unlink flips load()'s output between the override and the\n // fallback; a `change` leaves it identical and Svelte's own HMR handles\n // the underlying file.\n server.watcher.on('all', (event, changed) => {\n if (changed !== filePath) return;\n if (event !== 'add' && event !== 'unlink') return;\n const mod = server.moduleGraph.getModuleById(resolvedId);\n if (mod) server.moduleGraph.invalidateModule(mod);\n server.ws.send({ type: 'full-reload' });\n });\n },\n };\n}\n","import type { Plugin } from 'vite';\nimport { createOverridePlugin } from './override-plugin.js';\n\nexport function tesseraLayoutPlugin(): Plugin {\n return createOverridePlugin({\n name: 'tessera:layout',\n virtualId: 'virtual:tessera-layout',\n projectFile: 'layout.svelte',\n });\n}\n","import type { Plugin } from 'vite';\nimport { resolve } from 'node:path';\nimport { createOverridePlugin } from './override-plugin.js';\nimport { resolvePackageRoot } from './package-root.js';\n\nexport function tesseraQuizPlugin(): Plugin {\n const builtinQuiz = resolve(\n resolvePackageRoot(),\n 'src',\n 'components',\n 'Quiz.svelte',\n );\n return createOverridePlugin({\n name: 'tessera:quiz',\n virtualId: 'virtual:tessera-quiz',\n projectFile: 'quiz.svelte',\n builtinFile: builtinQuiz,\n });\n}\n","import type { Plugin, ResolvedConfig, ViteDevServer } from 'vite';\nimport { svelte } from '@sveltejs/vite-plugin-svelte';\nimport { resolve, relative, isAbsolute } from 'node:path';\nimport {\n existsSync,\n readdirSync,\n statSync,\n writeFileSync,\n unlinkSync,\n cpSync,\n mkdirSync,\n} from 'node:fs';\nimport { generateManifest, readCourseConfig } from './manifest.js';\nimport type { Manifest } from './manifest.js';\nimport type { CourseConfig } from '../runtime/types.js';\nimport {\n DEFAULT_PASSING_SCORE,\n DEFAULT_PERCENTAGE_THRESHOLD,\n} from '../runtime/defaults.js';\nimport {\n validateProject,\n reportValidationIssues,\n normalizeA11y,\n isPlausibleLanguageTag,\n isIgnored,\n type A11ySettings,\n} from './validation.js';\nimport { runExport } from './export.js';\nimport { tesseraLayoutPlugin } from './layout.js';\nimport { tesseraQuizPlugin } from './quiz.js';\nimport { resolvePackageRoot } from './package-root.js';\n\nimport { AUDIT_ENV_FLAG } from './a11y/audit.js';\n\nexport { runAudit } from './a11y/audit.js';\nexport type { AuditOptions, ImpactLevel } from './a11y/audit.js';\n\nfunction isAuditBuild(): boolean {\n return process.env[AUDIT_ENV_FLAG] === '1';\n}\n\n// Resolve the runtime directory where App.svelte lives\nfunction resolveRuntimeDir(): string {\n return resolve(resolvePackageRoot(), 'src', 'runtime');\n}\n\n// Resolve the framework styles directory\nfunction resolveStylesDir(): string {\n return resolve(resolvePackageRoot(), 'styles');\n}\n\n// Tier-1a state shared between the svelte() onwarn handler and the sibling\n// gate plugin. onwarn fires during transform (after the Tier-1b buildStart\n// gate), so a11y warnings are collected here and flushed/gated at buildEnd.\ninterface A11yCompilerState {\n warnings: string[];\n projectRoot: string;\n isBuild: boolean;\n settings: A11ySettings;\n}\n\n// Svelte's onwarn filename is relative to the vite root (e.g. `pages/x.svelte`)\n// in build and may be absolute or a virtual id elsewhere. Return the\n// project-relative path for a real author file, or null to skip framework /\n// node_modules / virtual modules — Tier 0 owns the framework's own warnings.\nfunction projectFileRel(\n filename: string | undefined,\n projectRoot: string,\n): string | null {\n if (!filename || !projectRoot) return null;\n if (\n filename.startsWith('\\0') ||\n filename.includes('virtual:') ||\n filename.includes('node_modules')\n ) {\n return null;\n }\n const abs = isAbsolute(filename) ? filename : resolve(projectRoot, filename);\n const rel = relative(projectRoot, abs);\n if (rel.startsWith('..') || isAbsolute(rel) || rel.includes('node_modules')) {\n return null;\n }\n return rel;\n}\n\ntype VirtualLoadCtx = { projectRoot: string; isBuild: boolean };\n\nfunction virtualModule(\n name: string,\n virtualId: string,\n load: (\n this: import('vite').Rollup.PluginContext,\n ctx: VirtualLoadCtx,\n ) => string | null,\n): Plugin {\n const resolvedId = '\\0' + virtualId;\n let projectRoot = '';\n let isBuild = false;\n return {\n name,\n enforce: 'pre',\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n resolveId(id) {\n return id === virtualId ? resolvedId : null;\n },\n load(id) {\n return id === resolvedId\n ? load.call(this, { projectRoot, isBuild })\n : null;\n },\n };\n}\n\nexport function tesseraPlugin() {\n const manifestRef: { current: Manifest | null; root: string } = {\n current: null,\n root: '',\n };\n const a11y: A11yCompilerState = {\n warnings: [],\n projectRoot: '',\n isBuild: false,\n settings: normalizeA11y(undefined),\n };\n return [\n svelte({\n compilerOptions: { css: 'external' },\n onwarn(warning, defaultHandler) {\n if (warning.code?.startsWith('a11y')) {\n const rel = projectFileRel(warning.filename, a11y.projectRoot);\n if (rel !== null) {\n const msg = `[${warning.code}] ${rel}: ${warning.message}`;\n if (a11y.isBuild) {\n a11y.warnings.push(msg);\n } else if (!a11y.settings.ignore.includes(warning.code)) {\n reportValidationIssues({ errors: [], warnings: [msg] });\n }\n }\n return; // suppress the raw Vite print; we re-emit via the reporter\n }\n defaultHandler?.(warning);\n },\n }),\n tesseraA11yCompilerPlugin(a11y),\n tesseraValidationPlugin(),\n tesseraEntryPlugin(),\n tesseraConfigDefaultsPlugin(),\n tesseraConfigPlugin(),\n tesseraPagesPlugin(),\n tesseraManifestPlugin(manifestRef),\n tesseraLayoutPlugin(),\n tesseraQuizPlugin(),\n tesseraAdapterPlugin(),\n tesseraXAPISetupPlugin(),\n tesseraFirstPagePreloadPlugin(manifestRef),\n tesseraExportPlugin(),\n ];\n}\n\n// ---------- Entry Plugin ----------\n\nconst VIRTUAL_ENTRY_ID = 'virtual:tessera-entry';\nconst RESOLVED_ENTRY_ID = '\\0' + VIRTUAL_ENTRY_ID;\nconst VIRTUAL_MAIN_ID = '/virtual:tessera-main';\nconst RESOLVED_MAIN_ID = '\\0virtual:tessera-main';\n\nfunction tesseraEntryPlugin(): Plugin {\n const runtimeDir = resolveRuntimeDir();\n const stylesDir = resolveStylesDir();\n const appSveltePath = resolve(runtimeDir, 'App.svelte');\n let projectRoot: string;\n let outDir: string;\n let isBuild = false;\n\n return {\n name: 'tessera:entry',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n outDir = resolve(config.root, config.build.outDir);\n isBuild = config.command === 'build';\n },\n\n // For build mode: write index.html so Rollup can find it\n buildStart() {\n if (isBuild) {\n writeFileSync(\n resolve(projectRoot, 'index.html'),\n generateIndexHtml(readLanguage(projectRoot)),\n 'utf-8',\n );\n }\n },\n\n // For build mode: clean up temporary index.html and copy assets\n closeBundle() {\n if (isBuild) {\n const htmlPath = resolve(projectRoot, 'index.html');\n if (existsSync(htmlPath)) {\n try {\n unlinkSync(htmlPath);\n } catch {}\n }\n\n // Copy assets/ into the build's assets/ so $assets/ references resolve\n const assetsDir = resolve(projectRoot, 'assets');\n const distAssetsDir = resolve(outDir, 'assets');\n if (existsSync(assetsDir)) {\n mkdirSync(distAssetsDir, { recursive: true });\n cpSync(assetsDir, distAssetsDir, { recursive: true });\n }\n }\n },\n\n // Serve index.html for the dev server\n configureServer(server: ViteDevServer) {\n return () => {\n server.middlewares.use(async (req, res, next) => {\n if (req.url === '/' || req.url === '/index.html') {\n const html = generateIndexHtml(readLanguage(projectRoot));\n const transformed = await server.transformIndexHtml(req.url, html);\n res.setHeader('Content-Type', 'text/html');\n res.statusCode = 200;\n res.end(transformed);\n return;\n }\n next();\n });\n };\n },\n\n resolveId(id) {\n if (id === VIRTUAL_ENTRY_ID) return RESOLVED_ENTRY_ID;\n if (id === VIRTUAL_MAIN_ID || id === 'virtual:tessera-main')\n return RESOLVED_MAIN_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_ENTRY_ID || id === RESOLVED_MAIN_ID) {\n return generateEntryScript(appSveltePath, stylesDir, projectRoot);\n }\n return null;\n },\n };\n}\n\n// 'en' fallback applied here: the config default-merge runs later than buildStart.\n// Only a validated BCP-47 tag is interpolated into <html lang>, so a malformed\n// value (caught separately as a warning) can't ship a broken attribute.\nfunction readLanguage(projectRoot: string): string {\n const read = readCourseConfig(projectRoot);\n const lang = read.ok ? read.config.language : undefined;\n return isPlausibleLanguageTag(lang) ? lang : 'en';\n}\n\nfunction generateIndexHtml(lang: string): string {\n return `<!DOCTYPE html>\n<html lang=\"${lang}\">\n<head>\n <meta charset=\"UTF-8\" />\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\" />\n <title>Tessera Course</title>\n</head>\n<body>\n <div id=\"tessera-root\"></div>\n <script type=\"module\" src=\"/virtual:tessera-main\"></script>\n</body>\n</html>`;\n}\n\nfunction generateEntryScript(\n appSveltePath: string,\n frameworkStylesDir: string,\n projectRoot: string,\n): string {\n const normalizedPath = appSveltePath.replace(/\\\\/g, '/');\n\n // Framework CSS imports (theme → base → layout)\n const frameworkCssOrder = ['theme.css', 'base.css', 'layout.css'];\n const frameworkImports = frameworkCssOrder\n .map((file) => resolve(frameworkStylesDir, file).replace(/\\\\/g, '/'))\n .filter((path) => existsSync(path))\n .map((path) => `import '${path}';`)\n .join('\\n');\n\n // User CSS imports from project's styles/ directory\n const userStylesDir = resolve(projectRoot, 'styles');\n let userImports = '';\n if (existsSync(userStylesDir)) {\n const userCssFiles = readdirSync(userStylesDir)\n .filter((f) => f.endsWith('.css'))\n .sort();\n userImports = userCssFiles\n .map((f) => resolve(userStylesDir, f).replace(/\\\\/g, '/'))\n .map((path) => `import '${path}';`)\n .join('\\n');\n }\n\n return `// Framework styles\n${frameworkImports}\n// User styles\n${userImports}\n\nimport { mount } from 'svelte';\nimport App from '${normalizedPath}';\n\nmount(App, {\n target: document.getElementById('tessera-root'),\n});\n`;\n}\n\n// ---------- Config Plugin ----------\n\nconst VIRTUAL_CONFIG_ID = 'virtual:tessera-config';\n\nfunction completionDefaults(mode: string | undefined): {\n completion: Record<string, unknown>;\n passingScore: number;\n} {\n if (mode === 'manual') {\n return { completion: { mode: 'manual' }, passingScore: 0 };\n }\n return {\n completion: {\n mode: 'percentage',\n percentageThreshold: DEFAULT_PERCENTAGE_THRESHOLD,\n },\n passingScore: DEFAULT_PASSING_SCORE,\n };\n}\n\nfunction tesseraConfigDefaultsPlugin(): Plugin {\n return {\n name: 'tessera:config-defaults',\n enforce: 'pre',\n config(config) {\n const root = config.root || process.cwd();\n return {\n base: './',\n build: { assetsDir: 'tessera' },\n resolve: { alias: { $assets: resolve(root, 'assets') } },\n // tessera-learn ships .ts/.svelte.ts source; Vite's dep optimizer\n // doesn't run vite-plugin-svelte's preprocessor, so skip pre-bundling.\n optimizeDeps: { exclude: ['tessera-learn'] },\n };\n },\n };\n}\n\nfunction tesseraConfigPlugin(): Plugin {\n return virtualModule(\n 'tessera:config',\n VIRTUAL_CONFIG_ID,\n function ({ projectRoot }) {\n const configPath = resolve(projectRoot, 'course.config.js');\n if (existsSync(configPath)) this.addWatchFile(configPath);\n const read = readCourseConfig(projectRoot);\n const userConfig: Partial<CourseConfig> = read.ok ? read.config : {};\n const { completion, passingScore } = completionDefaults(\n userConfig.completion?.mode,\n );\n const merged = {\n title: userConfig.title || 'Untitled Course',\n ...userConfig,\n navigation: { mode: 'free', ...userConfig.navigation },\n completion: { ...completion, ...userConfig.completion },\n scoring: { passingScore, ...userConfig.scoring },\n export: { standard: 'web', ...userConfig.export },\n };\n return `export default ${JSON.stringify(merged)};`;\n },\n );\n}\n\n// ---------- Manifest Watch Helpers ----------\n\n/** Register all _meta.js and .svelte files under pagesDir as watch files for build mode. */\nfunction addWatchFiles(\n ctx: { addWatchFile(id: string): void },\n dir: string,\n): void {\n if (!existsSync(dir)) return;\n for (const entry of readdirSync(dir)) {\n const full = resolve(dir, entry);\n if (statSync(full).isDirectory()) {\n addWatchFiles(ctx, full);\n } else if (entry.endsWith('.svelte') || entry === '_meta.js') {\n ctx.addWatchFile(full);\n }\n }\n}\n\n// ---------- Pages Plugin ----------\n\nconst VIRTUAL_PAGES_ID = 'virtual:tessera-pages';\n\n/**\n * Provides a virtual module that exports an import.meta.glob map for all .svelte\n * pages. This runs in the user's project context so the glob resolves against their\n * pages/ directory, and Vite can statically analyze it for code splitting.\n */\nfunction tesseraPagesPlugin(): Plugin {\n return virtualModule('tessera:pages', VIRTUAL_PAGES_ID, () => {\n return `export default import.meta.glob('/pages/**/*.svelte');`;\n });\n}\n\n// ---------- Validation Plugin ----------\n\nfunction tesseraValidationPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:validation',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n // Run validation during dev (configResolved fires before server starts)\n if (!isBuild) {\n runValidation(projectRoot);\n }\n },\n\n buildStart() {\n // Run validation during build (buildStart fires once before bundling)\n if (isBuild) {\n runValidation(projectRoot);\n }\n },\n };\n}\n\n// Tier 1a: flush + gate the Svelte compiler's a11y warnings at buildEnd, after\n// every module is transformed. svelte() accepts `onwarn` but not arbitrary\n// Rollup hooks, so the gate lives here and shares the onwarn closure.\nfunction tesseraA11yCompilerPlugin(a11y: A11yCompilerState): Plugin {\n return {\n name: 'tessera:a11y-compiler',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n a11y.projectRoot = config.root;\n a11y.isBuild = config.command === 'build';\n const read = readCourseConfig(config.root);\n a11y.settings = normalizeA11y(read.ok ? read.config.a11y : undefined);\n },\n\n buildEnd() {\n if (!a11y.isBuild || a11y.warnings.length === 0) return;\n const ignored = new Set(a11y.settings.ignore);\n const warnings = a11y.warnings.filter((msg) => !isIgnored(msg, ignored));\n a11y.warnings = [];\n if (warnings.length === 0) return;\n if (a11y.settings.level === 'error') {\n reportValidationIssues({ errors: warnings, warnings: [] });\n throw new Error(\n `Tessera: ${warnings.length} a11y issue(s) with a11y.level: 'error'. Fix the errors above to continue.`,\n );\n }\n reportValidationIssues({ errors: [], warnings });\n },\n };\n}\n\nfunction runValidation(projectRoot: string): void {\n const result = validateProject(projectRoot);\n reportValidationIssues(result);\n if (result.errors.length > 0) {\n throw new Error(\n `Tessera validation failed with ${result.errors.length} error(s). Fix the errors above to continue.`,\n );\n }\n}\n\n// ---------- Export Plugin ----------\n\nfunction tesseraExportPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:export',\n enforce: 'post',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n\n async closeBundle() {\n if (!isBuild) return;\n if (isAuditBuild()) return;\n\n const read = readCourseConfig(projectRoot);\n if (!read.ok) {\n // Validation already required a parseable course.config.js — getting\n // here means it vanished or broke mid-build. Surface that loudly\n // rather than shipping a bundle with no LMS export silently.\n if (read.reason === 'missing') {\n throw new Error(\n '[tessera:export] course.config.js not found at closeBundle. The file must exist for the export step to run.',\n );\n }\n if (read.reason === 'no-export') {\n throw new Error(\n '[tessera:export] course.config.js: could not locate `export default { ... }`. Cannot determine export.standard.',\n );\n }\n throw new Error(\n `[tessera:export] course.config.js: failed to parse export-default object literal — ${(read.error as Error).message}`,\n );\n }\n\n await runExport(\n projectRoot,\n read.config as Parameters<typeof runExport>[1],\n );\n },\n };\n}\n\n// ---------- Manifest Plugin ----------\n\nconst VIRTUAL_MANIFEST_ID = 'virtual:tessera-manifest';\nconst RESOLVED_MANIFEST_ID = '\\0' + VIRTUAL_MANIFEST_ID;\n\nfunction tesseraManifestPlugin(manifestRef: {\n current: Manifest | null;\n root: string;\n}): Plugin {\n let projectRoot: string;\n let pagesDir: string;\n\n function buildManifest(): Manifest {\n const m = generateManifest(pagesDir);\n manifestRef.current = m;\n return m;\n }\n\n return {\n name: 'tessera:manifest',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n pagesDir = resolve(projectRoot, 'pages');\n manifestRef.root = projectRoot;\n },\n\n configureServer(devServer: ViteDevServer) {\n // Watch the pages directory for changes\n devServer.watcher.on('all', (event, filePath) => {\n if (!filePath.startsWith(pagesDir)) return;\n\n // Rebuild manifest on relevant file changes\n const isRelevant =\n filePath.endsWith('.svelte') ||\n filePath.endsWith('_meta.js') ||\n event === 'addDir' ||\n event === 'unlinkDir';\n\n if (isRelevant) {\n manifestRef.current = null; // invalidate cache\n\n // Invalidate the virtual module to trigger HMR\n const mod = devServer.moduleGraph.getModuleById(RESOLVED_MANIFEST_ID);\n if (mod) {\n devServer.moduleGraph.invalidateModule(mod);\n devServer.ws.send({ type: 'full-reload' });\n }\n\n console.log(\n `[tessera] Manifest rebuilt (${event}: ${filePath.replace(projectRoot, '')})`,\n );\n }\n });\n },\n\n buildStart() {\n buildManifest();\n },\n\n resolveId(id) {\n if (id === VIRTUAL_MANIFEST_ID) return RESOLVED_MANIFEST_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_MANIFEST_ID) {\n if (!manifestRef.current) {\n buildManifest();\n }\n\n // Register watch files so Vite's built-in watcher (used in build --watch)\n // knows to re-trigger when pages/ content changes.\n addWatchFiles(this, pagesDir);\n\n // Encode as base64 to prevent Vite's import analysis from\n // scanning .svelte importPath strings as module imports.\n // Replace Infinity with 1e9 since JSON.stringify drops it.\n const json = JSON.stringify(manifestRef.current, (_key, value) =>\n value === Infinity ? 1e9 : value,\n );\n const b64 = Buffer.from(json).toString('base64');\n return `export default JSON.parse(atob(\"${b64}\"));`;\n }\n return null;\n },\n };\n}\n\nconst VIRTUAL_ADAPTER_ID = 'virtual:tessera-adapter';\n\nfunction tesseraAdapterPlugin(): Plugin {\n return virtualModule(\n 'tessera:adapter',\n VIRTUAL_ADAPTER_ID,\n ({ projectRoot, isBuild }) => {\n // In dev, defer to the runtime selector so its WebAdapter fallback\n // for unreachable LMS APIs keeps working.\n if (!isBuild) {\n return `export { createAdapter } from 'tessera-learn/runtime/adapters/index.js';`;\n }\n\n let standard = 'web';\n const read = readCourseConfig(projectRoot);\n if (read.ok && typeof read.config.export?.standard === 'string') {\n standard = read.config.export.standard;\n }\n\n // The audit renders headless with no LMS in the frame chain; the SCORM/\n // cmi5 adapters throw when their API is absent, so render with WebAdapter.\n if (isAuditBuild()) standard = 'web';\n\n switch (standard) {\n case 'scorm12':\n return `\nimport { SCORM12Adapter } from 'tessera-learn/runtime/adapters/scorm12.js';\nimport { findSCORM12API } from 'tessera-learn/runtime/adapters/discovery.js';\nimport { LMSAdapterError } from 'tessera-learn/runtime/adapters/index.js';\nexport function createAdapter() {\n const api = findSCORM12API();\n if (!api) throw new LMSAdapterError('scorm12', 'Tessera: SCORM 1.2 API not found in window.parent/opener chain. Course must be launched from a SCORM 1.2 LMS.');\n return new SCORM12Adapter(api);\n}\n`;\n case 'scorm2004':\n return `\nimport { SCORM2004Adapter } from 'tessera-learn/runtime/adapters/scorm2004.js';\nimport { findSCORM2004API } from 'tessera-learn/runtime/adapters/discovery.js';\nimport { LMSAdapterError } from 'tessera-learn/runtime/adapters/index.js';\nexport function createAdapter() {\n const api = findSCORM2004API();\n if (!api) throw new LMSAdapterError('scorm2004', 'Tessera: SCORM 2004 API not found in window.parent/opener chain. Course must be launched from a SCORM 2004 LMS.');\n return new SCORM2004Adapter(api);\n}\n`;\n case 'cmi5':\n return `\nimport { CMI5Adapter } from 'tessera-learn/runtime/adapters/cmi5.js';\nimport { hasCMI5LaunchParams } from 'tessera-learn/runtime/adapters/discovery.js';\nimport { LMSAdapterError } from 'tessera-learn/runtime/adapters/index.js';\nexport function createAdapter() {\n if (!hasCMI5LaunchParams()) throw new LMSAdapterError('cmi5', 'Tessera: cmi5 launch parameters not present on URL. Course must be launched from a cmi5-compliant LMS.');\n return new CMI5Adapter();\n}\n`;\n default:\n return `\nimport { WebAdapter } from 'tessera-learn/runtime/adapters/web.js';\nexport function createAdapter(config) {\n return new WebAdapter(config);\n}\n`;\n }\n },\n );\n}\n\nconst VIRTUAL_XAPI_SETUP_ID = 'virtual:tessera-xapi-setup';\n\nfunction tesseraXAPISetupPlugin(): Plugin {\n return virtualModule(\n 'tessera:xapi-setup',\n VIRTUAL_XAPI_SETUP_ID,\n ({ projectRoot, isBuild }) => {\n if (!isBuild) {\n return `export { buildXAPIClient } from 'tessera-learn/runtime/xapi/setup.js';`;\n }\n\n // The audit runs offline — don't wire real LRS destinations into it.\n if (isAuditBuild()) {\n return `export async function buildXAPIClient() { return null; }`;\n }\n\n let standard = 'web';\n let hasXapi = false;\n const read = readCourseConfig(projectRoot);\n if (read.ok) {\n if (typeof read.config.export?.standard === 'string')\n standard = read.config.export.standard;\n hasXapi = read.config.xapi != null;\n }\n\n // cmi5 needs the publisher regardless of explicit xapi config (cmi5\n // adapter shares the publisher queue for its own LMS-required statements).\n if (hasXapi || standard === 'cmi5') {\n return `export { buildXAPIClient } from 'tessera-learn/runtime/xapi/setup.js';`;\n }\n\n return `export async function buildXAPIClient() { return null; }`;\n },\n );\n}\n\nfunction tesseraFirstPagePreloadPlugin(manifestRef: {\n current: Manifest | null;\n root: string;\n}): Plugin {\n return {\n name: 'tessera:first-page-preload',\n apply: 'build',\n transformIndexHtml: {\n order: 'post',\n handler(_html, ctx) {\n const firstPagePath = manifestRef.current?.pages[0]?.importPath;\n if (!firstPagePath || !ctx.bundle) return;\n const normalized = resolve(\n manifestRef.root,\n firstPagePath.replace(/^\\//, ''),\n ).replace(/\\\\/g, '/');\n const chunk = Object.values(ctx.bundle).find(\n (c): c is import('vite').Rollup.OutputChunk =>\n c.type === 'chunk' &&\n !!c.facadeModuleId &&\n c.facadeModuleId.replace(/\\\\/g, '/') === normalized,\n );\n if (!chunk) return;\n return [\n {\n tag: 'link',\n attrs: { rel: 'modulepreload', href: `./${chunk.fileName}` },\n injectTo: 'head',\n },\n ];\n },\n },\n };\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "tessera-learn",
3
- "version": "0.0.13",
3
+ "version": "0.2.0",
4
4
  "description": "LMS tracking runtime for interactive learning content. One adapter layer (SCORM 1.2, SCORM 2004 4th Edition, cmi5, static Web), your choice of components.",
5
5
  "keywords": [
6
6
  "svelte",
@@ -32,6 +32,8 @@
32
32
  "dist",
33
33
  "src",
34
34
  "styles",
35
+ "templates",
36
+ "AGENTS.md",
35
37
  "README.md",
36
38
  "LICENSE"
37
39
  ],
@@ -45,6 +47,14 @@
45
47
  "import": "./dist/plugin/index.js",
46
48
  "default": "./dist/plugin/index.js"
47
49
  },
50
+ "./project-name": {
51
+ "types": "./src/plugin/project-name.ts",
52
+ "default": "./src/plugin/project-name.ts"
53
+ },
54
+ "./template-copy": {
55
+ "types": "./src/plugin/template-copy.ts",
56
+ "default": "./src/plugin/template-copy.ts"
57
+ },
48
58
  "./virtual": {
49
59
  "types": "./src/virtual.d.ts"
50
60
  },
@@ -55,17 +65,19 @@
55
65
  "**/*.css"
56
66
  ],
57
67
  "bin": {
58
- "tessera-validate": "./dist/plugin/cli.js",
59
- "tessera-a11y": "./dist/plugin/a11y-cli.js"
68
+ "tessera": "./dist/plugin/cli.js"
60
69
  },
61
70
  "engines": {
62
71
  "node": ">=24"
63
72
  },
64
73
  "dependencies": {
74
+ "@sveltejs/acorn-typescript": "^1.0.10",
65
75
  "@sveltejs/vite-plugin-svelte": "^7.1.2",
76
+ "acorn": "^8.16.0",
66
77
  "archiver": "^8.0.0",
67
78
  "json5": "^2.0.0",
68
- "svelte": "^5.55.7"
79
+ "svelte": "^5.56.0",
80
+ "vite": "^8.0.14"
69
81
  },
70
82
  "peerDependencies": {
71
83
  "@axe-core/playwright": ">=4",
@@ -80,14 +92,14 @@
80
92
  }
81
93
  },
82
94
  "devDependencies": {
83
- "@types/node": "^25.8.0",
84
- "@vitest/coverage-v8": "^4.1.6",
95
+ "@types/node": "^25.9.1",
96
+ "@vitest/coverage-v8": "^4.1.7",
85
97
  "jsdom": "^29.0.1",
98
+ "scorm-again": "3.0.5",
86
99
  "svelte-check": "^4.4.8",
87
- "tsdown": "^0.22.0",
100
+ "tsdown": "^0.22.1",
88
101
  "typescript": "^6.0.3",
89
- "vite": "^8.0.13",
90
- "vitest": "^4.1.6"
102
+ "vitest": "^4.1.7"
91
103
  },
92
104
  "scripts": {
93
105
  "build": "tsdown",
@@ -1,7 +1,7 @@
1
1
  <script>
2
2
  import { onMount } from 'svelte';
3
3
  import { useQuestion } from '../runtime/hooks.svelte.js';
4
- import { slugFromQuestion } from './util.js';
4
+ import { questionId } from './util.js';
5
5
  import LockedBanner from './LockedBanner.svelte';
6
6
  import ResultIcon from './ResultIcon.svelte';
7
7
  import RetryButton from './RetryButton.svelte';
@@ -34,7 +34,7 @@
34
34
 
35
35
  const q = useQuestion({
36
36
  get id() {
37
- return id ?? `fitb-${slugFromQuestion(question)}`;
37
+ return questionId(id, 'fitb', question);
38
38
  },
39
39
  get weight() {
40
40
  return weight;
@@ -2,7 +2,7 @@
2
2
  import { onMount } from 'svelte';
3
3
  import { SvelteMap } from 'svelte/reactivity';
4
4
  import { useQuestion } from '../runtime/hooks.svelte.js';
5
- import { slugFromQuestion, shuffle } from './util.js';
5
+ import { questionId, shuffle } from './util.js';
6
6
  import LockedBanner from './LockedBanner.svelte';
7
7
  import ResultIcon from './ResultIcon.svelte';
8
8
  import RetryButton from './RetryButton.svelte';
@@ -62,7 +62,7 @@
62
62
 
63
63
  const q = useQuestion({
64
64
  get id() {
65
- return id ?? `matching-${slugFromQuestion(question)}`;
65
+ return questionId(id, 'matching', question);
66
66
  },
67
67
  get weight() {
68
68
  return weight;
@@ -1,7 +1,7 @@
1
1
  <script>
2
2
  import { onMount } from 'svelte';
3
3
  import { useQuestion } from '../runtime/hooks.svelte.js';
4
- import { slugFromQuestion } from './util.js';
4
+ import { questionId } from './util.js';
5
5
  import LockedBanner from './LockedBanner.svelte';
6
6
  import RetryButton from './RetryButton.svelte';
7
7
 
@@ -24,7 +24,7 @@
24
24
 
25
25
  const q = useQuestion({
26
26
  get id() {
27
- return id ?? `mc-${slugFromQuestion(question)}`;
27
+ return questionId(id, 'mc', question);
28
28
  },
29
29
  get weight() {
30
30
  return weight;
@@ -8,75 +8,26 @@
8
8
  * @prop {import('svelte').Snippet} content - Modal body snippet
9
9
  */
10
10
  let { trigger, content, title = '' } = $props();
11
- let open = $state(false);
12
- let modalRef = $state(null);
13
- let previousFocus = null;
11
+ let dialogRef = $state(null);
14
12
 
15
13
  function openModal() {
16
- previousFocus = document.activeElement;
17
- open = true;
14
+ dialogRef?.showModal();
15
+ document.body.style.overflow = 'hidden';
18
16
  }
19
17
 
20
18
  function closeModal() {
21
- open = false;
22
- if (previousFocus && typeof previousFocus.focus === 'function') {
23
- previousFocus.focus();
24
- }
19
+ dialogRef?.close();
25
20
  }
26
21
 
27
- function handleOverlayClick(e) {
28
- if (e.target === e.currentTarget) {
29
- closeModal();
30
- }
22
+ function handleClose() {
23
+ document.body.style.overflow = '';
31
24
  }
32
25
 
33
- function handleKeydown(e) {
34
- if (e.key === 'Escape') {
35
- closeModal();
36
- return;
37
- }
38
-
39
- // Focus trap
40
- if (e.key === 'Tab' && modalRef) {
41
- const focusable = modalRef.querySelectorAll(
42
- 'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])',
43
- );
44
- if (focusable.length === 0) return;
45
-
46
- const first = focusable[0];
47
- const last = focusable[focusable.length - 1];
48
-
49
- if (e.shiftKey) {
50
- if (document.activeElement === first) {
51
- e.preventDefault();
52
- last.focus();
53
- }
54
- } else {
55
- if (document.activeElement === last) {
56
- e.preventDefault();
57
- first.focus();
58
- }
59
- }
60
- }
26
+ // A click whose target is the dialog element itself (not its content) is a
27
+ // backdrop click.
28
+ function handleClick(e) {
29
+ if (e.target === dialogRef) closeModal();
61
30
  }
62
-
63
- $effect(() => {
64
- if (open) {
65
- document.body.style.overflow = 'hidden';
66
- // Focus the modal after render
67
- queueMicrotask(() => {
68
- if (modalRef) {
69
- const firstFocusable = modalRef.querySelector(
70
- 'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])',
71
- );
72
- if (firstFocusable) firstFocusable.focus();
73
- else modalRef.focus();
74
- }
75
- });
76
- } else {
77
- document.body.style.overflow = '';
78
- }
79
- });
80
31
  </script>
81
32
 
82
33
  <!-- svelte-ignore a11y_click_events_have_key_events -->
@@ -85,37 +36,29 @@
85
36
  {@render trigger()}
86
37
  </div>
87
38
 
88
- {#if open}
89
- <!-- svelte-ignore a11y_no_static_element_interactions -->
90
- <div
91
- class="tessera-modal-overlay"
92
- onclick={handleOverlayClick}
93
- onkeydown={handleKeydown}
94
- >
95
- <div
96
- class="tessera-modal-content"
97
- role="dialog"
98
- aria-modal="true"
99
- aria-label={title || 'Modal'}
100
- bind:this={modalRef}
101
- tabindex="-1"
102
- >
103
- {#if title}
104
- <h2 class="tessera-modal-title">{title}</h2>
105
- {/if}
106
- <div class="tessera-modal-body">
107
- {@render content()}
108
- </div>
109
- <button
110
- class="tessera-modal-close"
111
- onclick={closeModal}
112
- aria-label="Close modal"
113
- >
114
-
115
- </button>
39
+ <dialog
40
+ class="tessera-modal"
41
+ bind:this={dialogRef}
42
+ aria-label={title || 'Modal'}
43
+ onclick={handleClick}
44
+ onclose={handleClose}
45
+ >
46
+ <div class="tessera-modal-content">
47
+ {#if title}
48
+ <h2 class="tessera-modal-title">{title}</h2>
49
+ {/if}
50
+ <div class="tessera-modal-body">
51
+ {@render content()}
116
52
  </div>
53
+ <button
54
+ class="tessera-modal-close"
55
+ onclick={closeModal}
56
+ aria-label="Close modal"
57
+ >
58
+
59
+ </button>
117
60
  </div>
118
- {/if}
61
+ </dialog>
119
62
 
120
63
  <style>
121
64
  .tessera-reveal-trigger {
@@ -129,15 +72,18 @@
129
72
  border-radius: 4px;
130
73
  }
131
74
 
132
- .tessera-modal-overlay {
133
- position: fixed;
134
- inset: 0;
75
+ .tessera-modal {
76
+ border: none;
77
+ padding: 0;
78
+ background: transparent;
79
+ max-width: 600px;
80
+ width: 100%;
81
+ max-height: 80vh;
82
+ margin: auto;
83
+ }
84
+
85
+ .tessera-modal::backdrop {
135
86
  background-color: rgba(0, 0, 0, 0.5);
136
- display: flex;
137
- align-items: center;
138
- justify-content: center;
139
- z-index: 2000;
140
- padding: var(--tessera-spacing-lg);
141
87
  animation: tessera-modal-fade-in 200ms ease;
142
88
  }
143
89
 
@@ -146,18 +92,12 @@
146
92
  background: var(--tessera-bg);
147
93
  border-radius: 12px;
148
94
  padding: var(--tessera-spacing-xl);
149
- max-width: 600px;
150
- width: 100%;
151
95
  max-height: 80vh;
152
96
  overflow-y: auto;
153
97
  box-shadow: 0 20px 60px rgba(0, 0, 0, 0.2);
154
98
  animation: tessera-modal-slide-in 200ms ease;
155
99
  }
156
100
 
157
- .tessera-modal-content:focus {
158
- outline: none;
159
- }
160
-
161
101
  .tessera-modal-title {
162
102
  font-size: 1.25rem;
163
103
  font-weight: 700;
@@ -222,10 +162,15 @@
222
162
  }
223
163
 
224
164
  @media (max-width: 640px) {
165
+ .tessera-modal {
166
+ max-height: 90vh;
167
+ margin-top: auto;
168
+ margin-bottom: 0;
169
+ }
170
+
225
171
  .tessera-modal-content {
226
172
  max-height: 90vh;
227
173
  border-radius: 12px 12px 0 0;
228
- align-self: flex-end;
229
174
  }
230
175
  }
231
176
  </style>
@@ -2,7 +2,7 @@
2
2
  import { onMount } from 'svelte';
3
3
  import { SvelteMap } from 'svelte/reactivity';
4
4
  import { useQuestion } from '../runtime/hooks.svelte.js';
5
- import { slugFromQuestion, shuffle } from './util.js';
5
+ import { questionId, shuffle } from './util.js';
6
6
  import LockedBanner from './LockedBanner.svelte';
7
7
  import ResultIcon from './ResultIcon.svelte';
8
8
  import RetryButton from './RetryButton.svelte';
@@ -51,7 +51,7 @@
51
51
  // pairs as stringified ids.
52
52
  const q = useQuestion({
53
53
  get id() {
54
- return id ?? `sorting-${slugFromQuestion(question)}`;
54
+ return questionId(id, 'sorting', question);
55
55
  },
56
56
  get weight() {
57
57
  return weight;
@@ -26,6 +26,15 @@ export function slugFromQuestion(text: unknown): string {
26
26
  .slice(0, 40);
27
27
  }
28
28
 
29
+ /** Author-supplied `id`, or a `prefix-<slug>` fallback derived from the prompt. */
30
+ export function questionId(
31
+ id: string | undefined,
32
+ prefix: string,
33
+ question: unknown,
34
+ ): string {
35
+ return id ?? `${prefix}-${slugFromQuestion(question)}`;
36
+ }
37
+
29
38
  /** Fisher-Yates shuffle returning a fresh array. */
30
39
  export function shuffle<T>(arr: readonly T[]): T[] {
31
40
  const result = arr.slice();
@@ -66,6 +66,10 @@ export function axeIgnoreRules(ignore: string[]): string[] {
66
66
  );
67
67
  }
68
68
 
69
+ export function isMissingBrowserError(message: string): boolean {
70
+ return /Executable doesn't exist|playwright install/i.test(message);
71
+ }
72
+
69
73
  // A violation with no impact is treated as failing rather than slipping the
70
74
  // gate at every threshold.
71
75
  function isFailing(v: AxeViolation, thresholdRank: number): boolean {
@@ -121,6 +125,7 @@ async function loadDeps(): Promise<
121
125
  */
122
126
  export async function runAudit(
123
127
  projectRoot: string,
128
+ workspaceRoot: string,
124
129
  options: AuditOptions = {},
125
130
  ): Promise<number> {
126
131
  const threshold: ImpactLevel = options.threshold ?? 'serious';
@@ -130,8 +135,8 @@ export async function runAudit(
130
135
  console.error(
131
136
  `\x1b[31m[tessera a11y]\x1b[0m Tier 2 needs Playwright + axe-core, which aren't installed.\n` +
132
137
  ` Install them to run the runtime audit:\n` +
133
- ` npm i -D playwright @axe-core/playwright\n` +
134
- ` npx playwright install chromium`,
138
+ ` pnpm add -D playwright @axe-core/playwright\n` +
139
+ ` pnpm exec playwright install chromium`,
135
140
  );
136
141
  return 1;
137
142
  }
@@ -146,6 +151,17 @@ export async function runAudit(
146
151
 
147
152
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
148
153
  const vite = (await import('vite')) as any;
154
+ const { resolveTesseraConfig } = await import('../inline-config.js');
155
+ // Carries tesseraPlugin() + the Svelte compiler; without it the plugin-less
156
+ // build would silently produce a broken bundle (there is no vite.config.js).
157
+ const auditBaseConfig = await resolveTesseraConfig(
158
+ projectRoot,
159
+ workspaceRoot,
160
+ {
161
+ command: 'build',
162
+ mode: 'production',
163
+ },
164
+ );
149
165
 
150
166
  // A throwaway web build, kept out of dist/ so a real LMS export is untouched.
151
167
  const auditDist = resolve(projectRoot, 'node_modules', '.tessera-a11y');
@@ -159,15 +175,17 @@ export async function runAudit(
159
175
  try {
160
176
  if (options.rebuild || !existsSync(distHtml)) {
161
177
  console.log('[tessera a11y] Building course…');
162
- await vite.build({
163
- root: projectRoot,
164
- build: { outDir: auditDist, emptyOutDir: true },
165
- logLevel: 'warn',
166
- });
178
+ await vite.build(
179
+ vite.mergeConfig(auditBaseConfig, {
180
+ build: { outDir: auditDist, emptyOutDir: true },
181
+ logLevel: 'warn',
182
+ }),
183
+ );
167
184
  }
168
185
 
169
186
  server = await vite.preview({
170
187
  root: projectRoot,
188
+ base: auditBaseConfig.base,
171
189
  build: { outDir: auditDist },
172
190
  preview: { port: 0, host: '127.0.0.1' },
173
191
  logLevel: 'warn',
@@ -178,7 +196,21 @@ export async function runAudit(
178
196
  return 1;
179
197
  }
180
198
 
181
- const browser = await chromium.launch();
199
+ let browser;
200
+ try {
201
+ browser = await chromium.launch();
202
+ } catch (err) {
203
+ const message = err instanceof Error ? err.message : String(err);
204
+ if (isMissingBrowserError(message)) {
205
+ console.error(
206
+ `\x1b[31m[tessera a11y]\x1b[0m Chromium isn't installed for Playwright.\n` +
207
+ ` Install it once:\n` +
208
+ ` pnpm exec playwright install chromium`,
209
+ );
210
+ return 1;
211
+ }
212
+ throw err;
213
+ }
182
214
  const pages: PageAuditResult[] = [];
183
215
  try {
184
216
  // axe-core/playwright requires a page from an explicit context.