@etus/bhono-app 0.1.1

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 (688) hide show
  1. package/dist/cli.d.ts +13 -0
  2. package/dist/cli.js +46 -0
  3. package/dist/cli.js.map +1 -0
  4. package/dist/cli.test.d.ts +1 -0
  5. package/dist/cli.test.js +26 -0
  6. package/dist/cli.test.js.map +1 -0
  7. package/dist/generator.d.ts +14 -0
  8. package/dist/generator.js +142 -0
  9. package/dist/generator.js.map +1 -0
  10. package/dist/generator.test.d.ts +1 -0
  11. package/dist/generator.test.js +127 -0
  12. package/dist/generator.test.js.map +1 -0
  13. package/dist/index.d.ts +2 -0
  14. package/dist/index.js +97 -0
  15. package/dist/index.js.map +1 -0
  16. package/dist/prompts.d.ts +25 -0
  17. package/dist/prompts.js +83 -0
  18. package/dist/prompts.js.map +1 -0
  19. package/dist/prompts.test.d.ts +1 -0
  20. package/dist/prompts.test.js +24 -0
  21. package/dist/prompts.test.js.map +1 -0
  22. package/dist/providers/cloudflare.d.ts +37 -0
  23. package/dist/providers/cloudflare.js +61 -0
  24. package/dist/providers/cloudflare.js.map +1 -0
  25. package/dist/providers/cloudflare.test.d.ts +1 -0
  26. package/dist/providers/cloudflare.test.js +29 -0
  27. package/dist/providers/cloudflare.test.js.map +1 -0
  28. package/dist/providers/github.d.ts +16 -0
  29. package/dist/providers/github.js +57 -0
  30. package/dist/providers/github.js.map +1 -0
  31. package/dist/providers/github.test.d.ts +1 -0
  32. package/dist/providers/github.test.js +16 -0
  33. package/dist/providers/github.test.js.map +1 -0
  34. package/dist/templates.d.ts +8 -0
  35. package/dist/templates.js +88 -0
  36. package/dist/templates.js.map +1 -0
  37. package/dist/templates.test.d.ts +1 -0
  38. package/dist/templates.test.js +26 -0
  39. package/dist/templates.test.js.map +1 -0
  40. package/package.json +36 -0
  41. package/templates/base/.claude/agents/architect-review.md +160 -0
  42. package/templates/base/.claude/agents/backend-architect.md +308 -0
  43. package/templates/base/.claude/agents/code-reviewer.md +170 -0
  44. package/templates/base/.claude/agents/performance-engineer.md +166 -0
  45. package/templates/base/.claude/agents/test-automator.md +219 -0
  46. package/templates/base/.claude/commands/check-skill-rules.md +53 -0
  47. package/templates/base/.claude/commands/claude-md.md +250 -0
  48. package/templates/base/.claude/commands/code-prompt.md +212 -0
  49. package/templates/base/.claude/commands/explain-code.md +194 -0
  50. package/templates/base/.claude/commands/init-projec.md +89 -0
  51. package/templates/base/.claude/commands/linear/README.md +297 -0
  52. package/templates/base/.claude/commands/linear/create-issue.md +190 -0
  53. package/templates/base/.claude/commands/linear/implement-issue.md +248 -0
  54. package/templates/base/.claude/commands/linear/process-triage.md +399 -0
  55. package/templates/base/.claude/commands/linear/setup.md +180 -0
  56. package/templates/base/.claude/commands/prime.md +9 -0
  57. package/templates/base/.claude/commands/review-gap.md +10 -0
  58. package/templates/base/.claude/commands/setup-aa.md +311 -0
  59. package/templates/base/.claude/commands/ship.md +262 -0
  60. package/templates/base/.claude/commands/tools.md +3 -0
  61. package/templates/base/.claude/docs/claude-progress.txt +107 -0
  62. package/templates/base/.claude/hooks/package-lock.json +556 -0
  63. package/templates/base/.claude/hooks/package.json +16 -0
  64. package/templates/base/.claude/hooks/skill-activation-prompt.sh +7 -0
  65. package/templates/base/.claude/hooks/skill-activation-prompt.ts +142 -0
  66. package/templates/base/.claude/hooks/tsconfig.json +19 -0
  67. package/templates/base/.claude/scripts/check-updates.sh +85 -0
  68. package/templates/base/.claude/scripts/install_pkgs.sh +66 -0
  69. package/templates/base/.claude/scripts/setup-project.sh +177 -0
  70. package/templates/base/.claude/scripts/validate-skill-rules.sh +94 -0
  71. package/templates/base/.claude/settings.json +113 -0
  72. package/templates/base/.claude/settings.local.json +11 -0
  73. package/templates/base/.claude/skills/architecture-analyzer/SKILL.md +531 -0
  74. package/templates/base/.claude/skills/architecture-analyzer/assets/report-template.md +215 -0
  75. package/templates/base/.claude/skills/architecture-analyzer/references/c4-templates.md +234 -0
  76. package/templates/base/.claude/skills/architecture-analyzer/references/confidence-levels.md +203 -0
  77. package/templates/base/.claude/skills/architecture-analyzer/scripts/analyze_structure.py +266 -0
  78. package/templates/base/.claude/skills/architecture-analyzer/scripts/analyze_tech_debt.py +776 -0
  79. package/templates/base/.claude/skills/architecture-analyzer/scripts/extract_apis.py +338 -0
  80. package/templates/base/.claude/skills/architecture-analyzer/scripts/generate_c4.py +283 -0
  81. package/templates/base/.claude/skills/architecture-analyzer/scripts/generate_erd.py +935 -0
  82. package/templates/base/.claude/skills/architecture-analyzer/scripts/map_dependencies.py +555 -0
  83. package/templates/base/.claude/skills/dev-browser/SKILL.md +318 -0
  84. package/templates/base/.claude/skills/dev-browser/bun.lock +443 -0
  85. package/templates/base/.claude/skills/dev-browser/package-lock.json +2927 -0
  86. package/templates/base/.claude/skills/dev-browser/package.json +27 -0
  87. package/templates/base/.claude/skills/dev-browser/scripts/start-server.ts +117 -0
  88. package/templates/base/.claude/skills/dev-browser/server.sh +24 -0
  89. package/templates/base/.claude/skills/dev-browser/src/client.ts +403 -0
  90. package/templates/base/.claude/skills/dev-browser/src/index.ts +281 -0
  91. package/templates/base/.claude/skills/dev-browser/src/snapshot/__tests__/snapshot.test.ts +223 -0
  92. package/templates/base/.claude/skills/dev-browser/src/snapshot/browser-script.ts +877 -0
  93. package/templates/base/.claude/skills/dev-browser/src/snapshot/index.ts +14 -0
  94. package/templates/base/.claude/skills/dev-browser/src/snapshot/inject.ts +13 -0
  95. package/templates/base/.claude/skills/dev-browser/src/types.ts +27 -0
  96. package/templates/base/.claude/skills/dev-browser/tsconfig.json +36 -0
  97. package/templates/base/.claude/skills/dev-browser/vitest.config.ts +12 -0
  98. package/templates/base/.claude/skills/linear/SKILL.md +440 -0
  99. package/templates/base/.claude/skills/linear/examples.md +262 -0
  100. package/templates/base/.claude/skills/linear/lib/client.ts +51 -0
  101. package/templates/base/.claude/skills/linear/lib/config.ts +106 -0
  102. package/templates/base/.claude/skills/linear/lib/output.ts +34 -0
  103. package/templates/base/.claude/skills/linear/package-lock.json +698 -0
  104. package/templates/base/.claude/skills/linear/package.json +27 -0
  105. package/templates/base/.claude/skills/linear/reference.md +263 -0
  106. package/templates/base/.claude/skills/linear/scripts/comments/create.ts +47 -0
  107. package/templates/base/.claude/skills/linear/scripts/comments/list.ts +47 -0
  108. package/templates/base/.claude/skills/linear/scripts/issues/archive.ts +30 -0
  109. package/templates/base/.claude/skills/linear/scripts/issues/create.ts +279 -0
  110. package/templates/base/.claude/skills/linear/scripts/issues/get.ts +68 -0
  111. package/templates/base/.claude/skills/linear/scripts/issues/list.ts +67 -0
  112. package/templates/base/.claude/skills/linear/scripts/issues/update.ts +281 -0
  113. package/templates/base/.claude/skills/linear/scripts/labels/add-to-issue.ts +63 -0
  114. package/templates/base/.claude/skills/linear/scripts/labels/create.ts +45 -0
  115. package/templates/base/.claude/skills/linear/scripts/labels/list.ts +30 -0
  116. package/templates/base/.claude/skills/linear/scripts/list-teams.ts +52 -0
  117. package/templates/base/.claude/skills/linear/scripts/setup/setup-credentials.ts +96 -0
  118. package/templates/base/.claude/skills/linear/scripts/status/list.ts +31 -0
  119. package/templates/base/.claude/skills/linear/scripts/status/set-by-name.ts +78 -0
  120. package/templates/base/.claude/skills/linear/scripts/status/update.ts +44 -0
  121. package/templates/base/.claude/skills/linear/scripts/users/list.ts +59 -0
  122. package/templates/base/.claude/skills/linear/scripts/users/me.ts +20 -0
  123. package/templates/base/.claude/skills/linear/templates/README.md +203 -0
  124. package/templates/base/.claude/skills/linear/templates/api-reference.md +258 -0
  125. package/templates/base/.claude/skills/linear/templates/bug-report.md +99 -0
  126. package/templates/base/.claude/skills/linear/templates/feature-request.md +118 -0
  127. package/templates/base/.claude/skills/linear/templates/security-issue.md +162 -0
  128. package/templates/base/.claude/skills/linear/templates/sprint-task.md +175 -0
  129. package/templates/base/.claude/skills/linear/templates/tech-debt.md +137 -0
  130. package/templates/base/.claude/skills/linear/tsconfig.json +17 -0
  131. package/templates/base/.claude/skills/linear/workflows/issue-lifecycle.md +317 -0
  132. package/templates/base/.claude/skills/playwright-e2e-testing/SKILL.md +113 -0
  133. package/templates/base/.claude/skills/playwright-e2e-testing/assets/global-setup.template.js +97 -0
  134. package/templates/base/.claude/skills/playwright-e2e-testing/assets/playwright.config.template.js +171 -0
  135. package/templates/base/.claude/skills/playwright-e2e-testing/assets/test-template.spec.js +163 -0
  136. package/templates/base/.claude/skills/playwright-e2e-testing/examples/README.md +26 -0
  137. package/templates/base/.claude/skills/playwright-e2e-testing/examples/ads.email-deeplink.spec.ts +12 -0
  138. package/templates/base/.claude/skills/playwright-e2e-testing/examples/mobile.realism.spec.ts +16 -0
  139. package/templates/base/.claude/skills/playwright-e2e-testing/examples/smoke.home.spec.ts +6 -0
  140. package/templates/base/.claude/skills/playwright-e2e-testing/references/architecture.md +578 -0
  141. package/templates/base/.claude/skills/playwright-e2e-testing/references/best-practices.md +260 -0
  142. package/templates/base/.claude/skills/playwright-e2e-testing/references/ci-reporting.md +86 -0
  143. package/templates/base/.claude/skills/playwright-e2e-testing/references/debugging.md +629 -0
  144. package/templates/base/.claude/skills/playwright-e2e-testing/references/mobile-realism.md +50 -0
  145. package/templates/base/.claude/skills/playwright-e2e-testing/references/optimization.md +488 -0
  146. package/templates/base/.claude/skills/playwright-e2e-testing/references/patterns.md +513 -0
  147. package/templates/base/.claude/skills/playwright-e2e-testing/references/resources.md +44 -0
  148. package/templates/base/.claude/skills/playwright-e2e-testing/references/visual-a11y.md +66 -0
  149. package/templates/base/.claude/skills/playwright-e2e-testing/scripts/auth-setup.js +202 -0
  150. package/templates/base/.claude/skills/playwright-e2e-testing/scripts/performance-analyzer.js +240 -0
  151. package/templates/base/.claude/skills/playwright-e2e-testing/scripts/trace-url.js +132 -0
  152. package/templates/base/.claude/skills/playwright-e2e-testing/templates/ci/github-actions.playwright.yml +78 -0
  153. package/templates/base/.claude/skills/playwright-e2e-testing/templates/fixtures.ts +44 -0
  154. package/templates/base/.claude/skills/playwright-e2e-testing/templates/global-setup.template.js +97 -0
  155. package/templates/base/.claude/skills/playwright-e2e-testing/templates/global.setup.ts +35 -0
  156. package/templates/base/.claude/skills/playwright-e2e-testing/templates/helpers/ad-gpt-observer.ts +80 -0
  157. package/templates/base/.claude/skills/playwright-e2e-testing/templates/helpers/chromium-mobile-profile.ts +93 -0
  158. package/templates/base/.claude/skills/playwright-e2e-testing/templates/playwright.config.template.js +171 -0
  159. package/templates/base/.claude/skills/playwright-e2e-testing/templates/playwright.config.ts +126 -0
  160. package/templates/base/.claude/skills/playwright-e2e-testing/templates/test-template.spec.js +163 -0
  161. package/templates/base/.claude/skills/playwright-e2e-testing/templates/tests/email-deeplink.ads.spec.ts +44 -0
  162. package/templates/base/.claude/skills/skill-rules.json +184 -0
  163. package/templates/base/.claude/skills/wrangler/SKILL.md +209 -0
  164. package/templates/base/.claude/skills/wrangler/resources/api.md +494 -0
  165. package/templates/base/.claude/skills/wrangler/resources/bundling.md +83 -0
  166. package/templates/base/.claude/skills/wrangler/resources/commands/cert.md +64 -0
  167. package/templates/base/.claude/skills/wrangler/resources/commands/check.md +66 -0
  168. package/templates/base/.claude/skills/wrangler/resources/commands/containers.md +157 -0
  169. package/templates/base/.claude/skills/wrangler/resources/commands/d1.md +843 -0
  170. package/templates/base/.claude/skills/wrangler/resources/commands/delete.md +27 -0
  171. package/templates/base/.claude/skills/wrangler/resources/commands/deploy.md +139 -0
  172. package/templates/base/.claude/skills/wrangler/resources/commands/deployments.md +56 -0
  173. package/templates/base/.claude/skills/wrangler/resources/commands/dev.md +157 -0
  174. package/templates/base/.claude/skills/wrangler/resources/commands/dispatch-namespace.md +69 -0
  175. package/templates/base/.claude/skills/wrangler/resources/commands/docs.md +61 -0
  176. package/templates/base/.claude/skills/wrangler/resources/commands/how-to-run.md +62 -0
  177. package/templates/base/.claude/skills/wrangler/resources/commands/hyperdrive.md +425 -0
  178. package/templates/base/.claude/skills/wrangler/resources/commands/init.md +31 -0
  179. package/templates/base/.claude/skills/wrangler/resources/commands/kv-bulk.md +265 -0
  180. package/templates/base/.claude/skills/wrangler/resources/commands/kv-key.md +353 -0
  181. package/templates/base/.claude/skills/wrangler/resources/commands/kv-namespace.md +265 -0
  182. package/templates/base/.claude/skills/wrangler/resources/commands/login.md +23 -0
  183. package/templates/base/.claude/skills/wrangler/resources/commands/logout.md +19 -0
  184. package/templates/base/.claude/skills/wrangler/resources/commands/mtls-certificate.md +69 -0
  185. package/templates/base/.claude/skills/wrangler/resources/commands/pages.md +175 -0
  186. package/templates/base/.claude/skills/wrangler/resources/commands/pipelines.md +76 -0
  187. package/templates/base/.claude/skills/wrangler/resources/commands/queues.md +132 -0
  188. package/templates/base/.claude/skills/wrangler/resources/commands/r2-bucket.md +342 -0
  189. package/templates/base/.claude/skills/wrangler/resources/commands/r2-object.md +267 -0
  190. package/templates/base/.claude/skills/wrangler/resources/commands/r2-sql.md +65 -0
  191. package/templates/base/.claude/skills/wrangler/resources/commands/rollback.md +40 -0
  192. package/templates/base/.claude/skills/wrangler/resources/commands/secret.md +308 -0
  193. package/templates/base/.claude/skills/wrangler/resources/commands/secrets-store-secret.md +100 -0
  194. package/templates/base/.claude/skills/wrangler/resources/commands/secrets-store-store.md +60 -0
  195. package/templates/base/.claude/skills/wrangler/resources/commands/setup.md +67 -0
  196. package/templates/base/.claude/skills/wrangler/resources/commands/tail.md +37 -0
  197. package/templates/base/.claude/skills/wrangler/resources/commands/telemetry.md +64 -0
  198. package/templates/base/.claude/skills/wrangler/resources/commands/triggers.md +39 -0
  199. package/templates/base/.claude/skills/wrangler/resources/commands/types.md +73 -0
  200. package/templates/base/.claude/skills/wrangler/resources/commands/vectorize.md +941 -0
  201. package/templates/base/.claude/skills/wrangler/resources/commands/versions.md +95 -0
  202. package/templates/base/.claude/skills/wrangler/resources/commands/whoami.md +49 -0
  203. package/templates/base/.claude/skills/wrangler/resources/commands/workflows.md +117 -0
  204. package/templates/base/.claude/skills/wrangler/resources/commands.md +138 -0
  205. package/templates/base/.claude/skills/wrangler/resources/configuration.md +2176 -0
  206. package/templates/base/.claude/skills/wrangler/resources/custom-builds.md +55 -0
  207. package/templates/base/.claude/skills/wrangler/resources/deprecations.md +279 -0
  208. package/templates/base/.claude/skills/wrangler/resources/enviroments.md +416 -0
  209. package/templates/base/.claude/skills/wrangler/resources/extract_sections.py +119 -0
  210. package/templates/base/.claude/skills/wrangler/resources/process_content.py +94 -0
  211. package/templates/base/.claude/skills/wrangler/resources/system-enviroments-variables.md +120 -0
  212. package/templates/base/.dev.vars.example +15 -0
  213. package/templates/base/.env.example +29 -0
  214. package/templates/base/.github/workflows/test.yml +127 -0
  215. package/templates/base/.nycrc.json +16 -0
  216. package/templates/base/CLAUDE.md +218 -0
  217. package/templates/base/README.md +670 -0
  218. package/templates/base/auth-setup-error.png +0 -0
  219. package/templates/base/config/drizzle.config.ts +10 -0
  220. package/templates/base/config/eslint.config.js +364 -0
  221. package/templates/base/config/wrangler.json +76 -0
  222. package/templates/base/docs/app_spec.txt +879 -0
  223. package/templates/base/docs/app_spec_template.md +681 -0
  224. package/templates/base/docs/architecture/README.md +8 -0
  225. package/templates/base/docs/architecture/data-requirements.md +109 -0
  226. package/templates/base/docs/architecture/erd.md +91 -0
  227. package/templates/base/docs/features/feature_list.json +3128 -0
  228. package/templates/base/docs/hono-boilerplate-plan.md +1774 -0
  229. package/templates/base/docs/test-coverage-gap-analysis.md +242 -0
  230. package/templates/base/docs/testing.md +188 -0
  231. package/templates/base/index.html +16 -0
  232. package/templates/base/package.json +115 -0
  233. package/templates/base/playwright.config.ts +158 -0
  234. package/templates/base/pnpm-lock.yaml +8175 -0
  235. package/templates/base/scripts/capture-prod-session.ts +250 -0
  236. package/templates/base/scripts/generate-openapi.ts +23 -0
  237. package/templates/base/scripts/init.sh +121 -0
  238. package/templates/base/src/client/__tests__/button.test.tsx +30 -0
  239. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-authenticated-should-display-dashboard-stats-cards-1.png +0 -0
  240. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-authenticated-should-display-quick-action-cards-1.png +0 -0
  241. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-authenticated-should-display-recent-activity-section-1.png +0 -0
  242. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-authenticated-should-display-user-first-name-in-welcome-message-1.png +0 -0
  243. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-authenticated-should-display-user-information-in-sidebar-1.png +0 -0
  244. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-authenticated-should-render-dashboard-when-authenticated-1.png +0 -0
  245. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-authenticated-should-show-navigation-sidebar-1.png +0 -0
  246. package/templates/base/src/client/__tests__/routes/__screenshots__/dashboard.test.tsx/Dashboard-Page-when-unauthenticated-should-redirect-to-login-when-not-authenticated-1.png +0 -0
  247. package/templates/base/src/client/__tests__/routes/__screenshots__/login.test.tsx/Login-Page-should-display-Google-OAuth-login-button-1.png +0 -0
  248. package/templates/base/src/client/__tests__/routes/__screenshots__/login.test.tsx/Login-Page-should-have-a-link-back-to-home-page-1.png +0 -0
  249. package/templates/base/src/client/__tests__/routes/__screenshots__/login.test.tsx/Login-Page-should-render-login-content-without-waiting-for-authentication-1.png +0 -0
  250. package/templates/base/src/client/__tests__/routes/__screenshots__/login.test.tsx/Login-Page-should-render-login-page-at--login-route-1.png +0 -0
  251. package/templates/base/src/client/__tests__/routes/__screenshots__/login.test.tsx/Login-Page-should-show-Terms-of-Service-and-Privacy-Policy-links-1.png +0 -0
  252. package/templates/base/src/client/__tests__/routes/__screenshots__/login.test.tsx/Login-Page-should-trigger-OAuth-flow-when-clicking-login-button-1.png +0 -0
  253. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-404-handling-should-display-404-text-on-not-found-page-1.png +0 -0
  254. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-404-handling-should-have-navigation-options-on-404-page-1.png +0 -0
  255. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-404-handling-should-render-404-page-for-unknown-routes-1.png +0 -0
  256. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-authenticated-navigation-should-navigate-from-dashboard-to-account-page-1.png +0 -0
  257. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-authenticated-navigation-should-navigate-from-dashboard-to-integrations-page-1.png +0 -0
  258. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-authenticated-navigation-should-navigate-from-dashboard-to-settings-page-1.png +0 -0
  259. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-authenticated-navigation-should-navigate-from-dashboard-to-team-page-1.png +0 -0
  260. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-home-page-navigation-should-display-navigation-links-on-home-page-1.png +0 -0
  261. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-home-page-navigation-should-have-correct-link-destinations-on-home-page-1.png +0 -0
  262. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-unauthenticated-navigation-should-allow-access-to-home-page-without-authentication-1.png +0 -0
  263. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-unauthenticated-navigation-should-allow-access-to-login-page-without-authentication-1.png +0 -0
  264. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-unauthenticated-navigation-should-redirect-unauthenticated-users-from-dashboard-to-login-1.png +0 -0
  265. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-unauthenticated-navigation-should-redirect-unauthenticated-users-from-settings-to-login-1.png +0 -0
  266. package/templates/base/src/client/__tests__/routes/__screenshots__/navigation.test.tsx/Navigation-unauthenticated-navigation-should-redirect-unauthenticated-users-from-team-to-login-1.png +0 -0
  267. package/templates/base/src/client/__tests__/routes/authenticated-layout.test.tsx +252 -0
  268. package/templates/base/src/client/__tests__/routes/dashboard.test.tsx +136 -0
  269. package/templates/base/src/client/__tests__/routes/error-components.test.tsx +186 -0
  270. package/templates/base/src/client/__tests__/routes/login.test.tsx +112 -0
  271. package/templates/base/src/client/__tests__/routes/navigation.test.tsx +272 -0
  272. package/templates/base/src/client/__tests__/routes/root-layout.test.tsx +65 -0
  273. package/templates/base/src/client/__tests__/setup-browser.ts +15 -0
  274. package/templates/base/src/client/__tests__/setup.ts +32 -0
  275. package/templates/base/src/client/__tests__/test-utils.tsx +135 -0
  276. package/templates/base/src/client/api/.gitkeep +0 -0
  277. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-can-be-collapsed-by-default-1.png +0 -0
  278. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-expands-when-collapsed-and-expand-button-is-clicked-1.png +0 -0
  279. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-handles-logout-button-click-1.png +0 -0
  280. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-handles-navigation-clicks-1.png +0 -0
  281. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-hides-navigation-labels-when-collapsed-1.png +0 -0
  282. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-highlights-active-route-1.png +0 -0
  283. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-renders-sidebar-navigation-items-1.png +0 -0
  284. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-shows-keyboard-shortcut-hint-when-expanded-1.png +0 -0
  285. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-shows-user-info-when-authenticated-1.png +0 -0
  286. package/templates/base/src/client/components/__tests__/__screenshots__/sidebar.test.tsx/Sidebar-shows-user-initials-in-avatar-fallback-1.png +0 -0
  287. package/templates/base/src/client/components/__tests__/error-boundary.test.tsx +97 -0
  288. package/templates/base/src/client/components/__tests__/sidebar.test.tsx +281 -0
  289. package/templates/base/src/client/components/error-boundary.tsx +68 -0
  290. package/templates/base/src/client/components/icons.tsx +106 -0
  291. package/templates/base/src/client/components/layout/.gitkeep +0 -0
  292. package/templates/base/src/client/components/sidebar.tsx +267 -0
  293. package/templates/base/src/client/components/ui/.gitkeep +0 -0
  294. package/templates/base/src/client/components/ui/__tests__/avatar.test.tsx +308 -0
  295. package/templates/base/src/client/components/ui/__tests__/card.test.tsx +214 -0
  296. package/templates/base/src/client/components/ui/__tests__/dialog.test.tsx +297 -0
  297. package/templates/base/src/client/components/ui/__tests__/error-fallback.test.tsx +145 -0
  298. package/templates/base/src/client/components/ui/__tests__/input.test.tsx +98 -0
  299. package/templates/base/src/client/components/ui/__tests__/loading-skeleton.test.tsx +139 -0
  300. package/templates/base/src/client/components/ui/__tests__/skeleton.test.tsx +44 -0
  301. package/templates/base/src/client/components/ui/__tests__/sonner.test.tsx +28 -0
  302. package/templates/base/src/client/components/ui/__tests__/tabs.test.tsx +233 -0
  303. package/templates/base/src/client/components/ui/avatar.tsx +101 -0
  304. package/templates/base/src/client/components/ui/badge.tsx +46 -0
  305. package/templates/base/src/client/components/ui/button.tsx +72 -0
  306. package/templates/base/src/client/components/ui/card.tsx +86 -0
  307. package/templates/base/src/client/components/ui/dialog.tsx +140 -0
  308. package/templates/base/src/client/components/ui/error-fallback.tsx +179 -0
  309. package/templates/base/src/client/components/ui/form.tsx +172 -0
  310. package/templates/base/src/client/components/ui/input.tsx +24 -0
  311. package/templates/base/src/client/components/ui/label.tsx +22 -0
  312. package/templates/base/src/client/components/ui/loading-skeleton.tsx +154 -0
  313. package/templates/base/src/client/components/ui/separator.tsx +33 -0
  314. package/templates/base/src/client/components/ui/skeleton.tsx +16 -0
  315. package/templates/base/src/client/components/ui/sonner.tsx +29 -0
  316. package/templates/base/src/client/components/ui/tabs.tsx +121 -0
  317. package/templates/base/src/client/hooks/.gitkeep +0 -0
  318. package/templates/base/src/client/hooks/__tests__/use-auth.test.tsx +306 -0
  319. package/templates/base/src/client/hooks/__tests__/use-theme.test.tsx +172 -0
  320. package/templates/base/src/client/hooks/use-auth.ts +53 -0
  321. package/templates/base/src/client/hooks/use-theme.tsx +78 -0
  322. package/templates/base/src/client/index.css +881 -0
  323. package/templates/base/src/client/lib/query-client.ts +11 -0
  324. package/templates/base/src/client/lib/utils.ts +7 -0
  325. package/templates/base/src/client/main.tsx +26 -0
  326. package/templates/base/src/client/routeTree.gen.ts +258 -0
  327. package/templates/base/src/client/router.ts +15 -0
  328. package/templates/base/src/client/routes/$.tsx +77 -0
  329. package/templates/base/src/client/routes/.gitkeep +0 -0
  330. package/templates/base/src/client/routes/__root.tsx +34 -0
  331. package/templates/base/src/client/routes/__tests__/__screenshots__/invite.test.tsx/Invite-Token-Page-pending-invitation-state-should-display-accept-invitation-button-1.png +0 -0
  332. package/templates/base/src/client/routes/__tests__/__screenshots__/invite.test.tsx/Invite-Token-Page-pending-invitation-state-should-display-decline-button-linking-to-homepage-1.png +0 -0
  333. package/templates/base/src/client/routes/__tests__/__screenshots__/invite.test.tsx/Invite-Token-Page-pending-invitation-state-should-display-invitation-details--email--workspace--role--1.png +0 -0
  334. package/templates/base/src/client/routes/__tests__/__screenshots__/invite.test.tsx/Invite-Token-Page-pending-invitation-state-should-have-a-logo-link-to-homepage-1.png +0 -0
  335. package/templates/base/src/client/routes/__tests__/__screenshots__/invite.test.tsx/Invite-Token-Page-pending-invitation-state-should-render-invitation-page-with-inviter-name-and-workspace-1.png +0 -0
  336. package/templates/base/src/client/routes/__tests__/__screenshots__/invite.test.tsx/Invite-Token-Page-pending-invitation-state-should-show-terms-of-service-and-privacy-policy-links-1.png +0 -0
  337. package/templates/base/src/client/routes/__tests__/invite.test.tsx +138 -0
  338. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/account.test.tsx/Account-Page-should-render-Active-Sessions-section-with-session-cards-1.png +0 -0
  339. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/account.test.tsx/Account-Page-should-render-page-with-correct-title--Account--1.png +0 -0
  340. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/account.test.tsx/Account-Page-should-show-API-Access-section-1.png +0 -0
  341. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/account.test.tsx/Account-Page-should-show-Connected-Accounts-section-with-Google-connected-1.png +0 -0
  342. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/account.test.tsx/Account-Page-should-show-Danger-Zone-section-with-delete-button-1.png +0 -0
  343. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/account.test.tsx/Account-Page-should-show-Security-section-with-Two-Factor-Authentication-1.png +0 -0
  344. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-API-documentation-section-should-display-API-documentation-link-section-1.png +0 -0
  345. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-Create-Webhook-Dialog-should-have-Add-Webhook-trigger-button-1.png +0 -0
  346. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-category-filters-should-display-all-category-buttons-1.png +0 -0
  347. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-integration-cards-should-display-all-integration-cards-with-names-1.png +0 -0
  348. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-integration-cards-should-display-category-badges-on-cards-1.png +0 -0
  349. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-integration-cards-should-show-Configure-button-for-connected-integrations-1.png +0 -0
  350. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-integration-cards-should-show-Connect-button-for-not-connected-integrations-1.png +0 -0
  351. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-integration-cards-should-show-Connected-badge-for-connected-integrations-1.png +0 -0
  352. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-page-rendering-should-display-Available-Integrations-section-1.png +0 -0
  353. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-page-rendering-should-display-Webhooks-section-1.png +0 -0
  354. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-page-rendering-should-display-connected-count-1.png +0 -0
  355. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-page-rendering-should-display-page-description-1.png +0 -0
  356. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-page-rendering-should-display-search-input-1.png +0 -0
  357. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-page-rendering-should-render-with-correct-title--Integrations--1.png +0 -0
  358. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-search-functionality-should-filter-integrations-based-on-search-query-1.png +0 -0
  359. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-search-functionality-should-search-by-description-1.png +0 -0
  360. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-search-functionality-should-show-no-results-message-when-search-has-no-matches-1.png +0 -0
  361. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-webhooks-section-should-display-existing-webhook-1.png +0 -0
  362. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-webhooks-section-should-display-webhook-events-1.png +0 -0
  363. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-webhooks-section-should-display-webhook-last-delivery-info-1.png +0 -0
  364. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-webhooks-section-should-display-webhook-success-status-1.png +0 -0
  365. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/integrations.test.tsx/Integrations-Page-webhooks-section-should-have-Add-Webhook-button-1.png +0 -0
  366. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Account-tab-should-display-connected-accounts-section-with-Google-provider-1.png +0 -0
  367. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Account-tab-should-display-sessions-and-danger-zone-sections-1.png +0 -0
  368. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Notifications-tab-should-display-all-notification-toggle-options-1.png +0 -0
  369. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Notifications-tab-should-display-email-notifications-section-1.png +0 -0
  370. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Notifications-tab-should-display-toggle-switches-for-notification-options-1.png +0 -0
  371. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Notifications-tab-should-have-toggles-checked-by-default-1.png +0 -0
  372. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Profile-tab-should-display--Save-Changes--button-1.png +0 -0
  373. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Profile-tab-should-display-personal-information-form-1.png +0 -0
  374. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Profile-tab-should-display-profile-picture-section-1.png +0 -0
  375. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Profile-tab-should-display-user-email-in-disabled-email-input-1.png +0 -0
  376. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Profile-tab-should-display-user-initials-in-avatar-1.png +0 -0
  377. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-Profile-tab-should-display-user-name-in-the-name-input-1.png +0 -0
  378. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-page-rendering-should-display-all-three-tabs-1.png +0 -0
  379. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-page-rendering-should-display-page-description-1.png +0 -0
  380. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-page-rendering-should-render-with-correct-title--Settings--1.png +0 -0
  381. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-tab-navigation-should-show-Profile-tab-as-default-active-tab-1.png +0 -0
  382. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-tab-navigation-should-switch-to-Account-tab-when-clicked-1.png +0 -0
  383. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/settings.test.tsx/Settings-Page-tab-navigation-should-switch-to-Notifications-tab-when-clicked-1.png +0 -0
  384. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/team.test.tsx/Team-Page-should-display-Active-Members-section-with-member-count-1.png +0 -0
  385. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/team.test.tsx/Team-Page-should-display-Pending-Invitations-section-with-invitation-details-1.png +0 -0
  386. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/team.test.tsx/Team-Page-should-have-invite-member-button-that-can-be-clicked-1.png +0 -0
  387. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/team.test.tsx/Team-Page-should-render-page-with-correct-title-and-description-1.png +0 -0
  388. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/team.test.tsx/Team-Page-should-render-search-input-that-filters-team-members-1.png +0 -0
  389. package/templates/base/src/client/routes/_authenticated/__tests__/__screenshots__/team.test.tsx/Team-Page-should-show-current-user-with---you---indicator-and-role-badge-1.png +0 -0
  390. package/templates/base/src/client/routes/_authenticated/__tests__/account.test.tsx +324 -0
  391. package/templates/base/src/client/routes/_authenticated/__tests__/integrations.test.tsx +520 -0
  392. package/templates/base/src/client/routes/_authenticated/__tests__/settings.test.tsx +414 -0
  393. package/templates/base/src/client/routes/_authenticated/__tests__/team.test.tsx +374 -0
  394. package/templates/base/src/client/routes/_authenticated/account.tsx +416 -0
  395. package/templates/base/src/client/routes/_authenticated/dashboard.tsx +151 -0
  396. package/templates/base/src/client/routes/_authenticated/integrations.tsx +553 -0
  397. package/templates/base/src/client/routes/_authenticated/settings.tsx +310 -0
  398. package/templates/base/src/client/routes/_authenticated/team.tsx +395 -0
  399. package/templates/base/src/client/routes/_authenticated.tsx +69 -0
  400. package/templates/base/src/client/routes/index.tsx +155 -0
  401. package/templates/base/src/client/routes/invite.$token.tsx +191 -0
  402. package/templates/base/src/client/routes/login.tsx +92 -0
  403. package/templates/base/src/server/__tests__/fixtures.ts +461 -0
  404. package/templates/base/src/server/__tests__/mocks/__tests__/db.test.ts +239 -0
  405. package/templates/base/src/server/__tests__/mocks/__tests__/kv.test.ts +293 -0
  406. package/templates/base/src/server/__tests__/mocks/__tests__/r2.test.ts +363 -0
  407. package/templates/base/src/server/__tests__/mocks/db.ts +186 -0
  408. package/templates/base/src/server/__tests__/mocks/index.ts +33 -0
  409. package/templates/base/src/server/__tests__/mocks/kv.ts +286 -0
  410. package/templates/base/src/server/__tests__/mocks/r2.ts +397 -0
  411. package/templates/base/src/server/__tests__/setup.ts +281 -0
  412. package/templates/base/src/server/auth/__tests__/guards.test.ts +162 -0
  413. package/templates/base/src/server/auth/guards.ts +92 -0
  414. package/templates/base/src/server/auth/permissions.test.ts +45 -0
  415. package/templates/base/src/server/auth/permissions.ts +139 -0
  416. package/templates/base/src/server/auth/roles.test.ts +169 -0
  417. package/templates/base/src/server/auth/roles.ts +141 -0
  418. package/templates/base/src/server/db/client.ts +12 -0
  419. package/templates/base/src/server/db/schema/accounts.ts +20 -0
  420. package/templates/base/src/server/db/schema/audit-logs.ts +26 -0
  421. package/templates/base/src/server/db/schema/index.ts +7 -0
  422. package/templates/base/src/server/db/schema/invitations.ts +30 -0
  423. package/templates/base/src/server/db/schema/refresh-tokens.ts +22 -0
  424. package/templates/base/src/server/db/schema/user-accounts.ts +25 -0
  425. package/templates/base/src/server/db/schema/users.ts +33 -0
  426. package/templates/base/src/server/db/seed.ts +267 -0
  427. package/templates/base/src/server/env.test.ts +84 -0
  428. package/templates/base/src/server/env.ts +78 -0
  429. package/templates/base/src/server/index.ts +82 -0
  430. package/templates/base/src/server/lib/audit.ts +73 -0
  431. package/templates/base/src/server/lib/audited-db.test.ts +107 -0
  432. package/templates/base/src/server/lib/audited-db.ts +154 -0
  433. package/templates/base/src/server/lib/email.test.ts +116 -0
  434. package/templates/base/src/server/lib/email.ts +82 -0
  435. package/templates/base/src/server/lib/errors.test.ts +49 -0
  436. package/templates/base/src/server/lib/errors.ts +64 -0
  437. package/templates/base/src/server/lib/oauth.test.ts +238 -0
  438. package/templates/base/src/server/lib/oauth.ts +113 -0
  439. package/templates/base/src/server/lib/pagination.test.ts +52 -0
  440. package/templates/base/src/server/lib/pagination.ts +32 -0
  441. package/templates/base/src/server/lib/password.test.ts +151 -0
  442. package/templates/base/src/server/lib/password.ts +151 -0
  443. package/templates/base/src/server/lib/providers.test.ts +105 -0
  444. package/templates/base/src/server/lib/providers.ts +62 -0
  445. package/templates/base/src/server/lib/r2-storage.test.ts +202 -0
  446. package/templates/base/src/server/lib/r2-storage.ts +107 -0
  447. package/templates/base/src/server/lib/schema-helpers.ts +16 -0
  448. package/templates/base/src/server/lib/session.test.ts +758 -0
  449. package/templates/base/src/server/lib/session.ts +267 -0
  450. package/templates/base/src/server/lib/tokens.test.ts +208 -0
  451. package/templates/base/src/server/lib/tokens.ts +65 -0
  452. package/templates/base/src/server/lib/transaction.test.ts +45 -0
  453. package/templates/base/src/server/lib/transaction.ts +24 -0
  454. package/templates/base/src/server/middleware/account.test.ts +201 -0
  455. package/templates/base/src/server/middleware/account.ts +66 -0
  456. package/templates/base/src/server/middleware/auth.test.ts +345 -0
  457. package/templates/base/src/server/middleware/auth.ts +146 -0
  458. package/templates/base/src/server/middleware/cors.test.ts +88 -0
  459. package/templates/base/src/server/middleware/cors.ts +26 -0
  460. package/templates/base/src/server/middleware/error-handler.test.ts +69 -0
  461. package/templates/base/src/server/middleware/error-handler.ts +43 -0
  462. package/templates/base/src/server/middleware/index.ts +8 -0
  463. package/templates/base/src/server/middleware/rate-limit.test.ts +472 -0
  464. package/templates/base/src/server/middleware/rate-limit.ts +294 -0
  465. package/templates/base/src/server/middleware/request-context.test.ts +175 -0
  466. package/templates/base/src/server/middleware/request-context.ts +28 -0
  467. package/templates/base/src/server/middleware/request-logger.test.ts +92 -0
  468. package/templates/base/src/server/middleware/request-logger.ts +50 -0
  469. package/templates/base/src/server/routes/accounts/__tests__/handlers.test.ts +460 -0
  470. package/templates/base/src/server/routes/accounts/handlers.ts +179 -0
  471. package/templates/base/src/server/routes/accounts/index.ts +55 -0
  472. package/templates/base/src/server/routes/accounts/routes.ts +205 -0
  473. package/templates/base/src/server/routes/accounts/schemas.ts +31 -0
  474. package/templates/base/src/server/routes/api.ts +37 -0
  475. package/templates/base/src/server/routes/audits/__tests__/handlers.test.ts +349 -0
  476. package/templates/base/src/server/routes/audits/handlers.ts +37 -0
  477. package/templates/base/src/server/routes/audits/index.ts +14 -0
  478. package/templates/base/src/server/routes/audits/routes.ts +29 -0
  479. package/templates/base/src/server/routes/audits/schemas.ts +43 -0
  480. package/templates/base/src/server/routes/auth/__tests__/handlers.test.ts +381 -0
  481. package/templates/base/src/server/routes/auth/handlers.ts +222 -0
  482. package/templates/base/src/server/routes/auth/index.ts +37 -0
  483. package/templates/base/src/server/routes/auth/routes.ts +136 -0
  484. package/templates/base/src/server/routes/auth/schemas.ts +48 -0
  485. package/templates/base/src/server/routes/auth/test-login.ts +156 -0
  486. package/templates/base/src/server/routes/health/__tests__/handlers.test.ts +237 -0
  487. package/templates/base/src/server/routes/health/handlers.ts +83 -0
  488. package/templates/base/src/server/routes/health/index.ts +13 -0
  489. package/templates/base/src/server/routes/health/routes.ts +90 -0
  490. package/templates/base/src/server/routes/index.ts +53 -0
  491. package/templates/base/src/server/routes/invitations/__tests__/handlers.test.ts +473 -0
  492. package/templates/base/src/server/routes/invitations/handlers.ts +71 -0
  493. package/templates/base/src/server/routes/invitations/index.ts +25 -0
  494. package/templates/base/src/server/routes/invitations/routes.ts +69 -0
  495. package/templates/base/src/server/routes/invitations/schemas.ts +39 -0
  496. package/templates/base/src/server/routes/openapi.ts +14 -0
  497. package/templates/base/src/server/routes/schemas.ts +53 -0
  498. package/templates/base/src/server/routes/storage/__tests__/handlers.test.ts +408 -0
  499. package/templates/base/src/server/routes/storage/handlers.ts +100 -0
  500. package/templates/base/src/server/routes/storage/index.ts +42 -0
  501. package/templates/base/src/server/routes/storage/routes.ts +91 -0
  502. package/templates/base/src/server/routes/storage/schemas.ts +56 -0
  503. package/templates/base/src/server/routes/users/__tests__/handlers.test.ts +526 -0
  504. package/templates/base/src/server/routes/users/handlers.ts +228 -0
  505. package/templates/base/src/server/routes/users/index.ts +67 -0
  506. package/templates/base/src/server/routes/users/routes.ts +265 -0
  507. package/templates/base/src/server/routes/users/schemas.ts +67 -0
  508. package/templates/base/src/server/services/__tests__/accounts.test.ts +764 -0
  509. package/templates/base/src/server/services/__tests__/audits.test.ts +235 -0
  510. package/templates/base/src/server/services/__tests__/auth.test.ts +765 -0
  511. package/templates/base/src/server/services/__tests__/invitations.test.ts +704 -0
  512. package/templates/base/src/server/services/__tests__/users.test.ts +755 -0
  513. package/templates/base/src/server/services/accounts.ts +269 -0
  514. package/templates/base/src/server/services/audits.ts +82 -0
  515. package/templates/base/src/server/services/auth.ts +225 -0
  516. package/templates/base/src/server/services/index.ts +6 -0
  517. package/templates/base/src/server/services/invitations.ts +306 -0
  518. package/templates/base/src/server/services/users.ts +350 -0
  519. package/templates/base/src/server/types/auth.ts +36 -0
  520. package/templates/base/src/server/types/index.ts +117 -0
  521. package/templates/base/src/shared/schemas/.gitkeep +0 -0
  522. package/templates/base/src/shared/schemas/__tests__/schemas.test.ts +547 -0
  523. package/templates/base/src/shared/schemas/account.ts +15 -0
  524. package/templates/base/src/shared/schemas/index.ts +6 -0
  525. package/templates/base/src/shared/schemas/invitation.ts +9 -0
  526. package/templates/base/src/shared/schemas/profile.ts +10 -0
  527. package/templates/base/src/shared/schemas/user.ts +16 -0
  528. package/templates/base/src/shared/schemas/webhook.ts +12 -0
  529. package/templates/base/src/shared/types/.gitkeep +0 -0
  530. package/templates/base/src/shared/types/account.ts +12 -0
  531. package/templates/base/src/shared/types/api.ts +1399 -0
  532. package/templates/base/src/shared/types/auth.ts +24 -0
  533. package/templates/base/src/shared/types/index.ts +5 -0
  534. package/templates/base/src/shared/types/user.ts +31 -0
  535. package/templates/base/src/test/vitest-zod-matcher.ts +37 -0
  536. package/templates/base/src/test/vitest.d.ts +19 -0
  537. package/templates/base/tests/e2e/README.md +141 -0
  538. package/templates/base/tests/e2e/a11y/accessibility.spec.ts +925 -0
  539. package/templates/base/tests/e2e/a11y/keyboard-navigation.spec.ts +610 -0
  540. package/templates/base/tests/e2e/api/accounts.spec.ts +148 -0
  541. package/templates/base/tests/e2e/api/audit-logs.spec.ts +130 -0
  542. package/templates/base/tests/e2e/api/authenticated-api.spec.ts +311 -0
  543. package/templates/base/tests/e2e/api/storage.spec.ts +109 -0
  544. package/templates/base/tests/e2e/auth-flows.unauth.spec.ts +117 -0
  545. package/templates/base/tests/e2e/auth-logout.unauth.spec.ts +103 -0
  546. package/templates/base/tests/e2e/auth.setup.ts +115 -0
  547. package/templates/base/tests/e2e/auth.spec.ts +146 -0
  548. package/templates/base/tests/e2e/compatibility/cross-browser.spec.ts +152 -0
  549. package/templates/base/tests/e2e/compatibility/cross-browser.spec.ts-snapshots/login-chromium-chromium-darwin.png +0 -0
  550. package/templates/base/tests/e2e/crud/account.spec.ts +356 -0
  551. package/templates/base/tests/e2e/crud/integrations.spec.ts +419 -0
  552. package/templates/base/tests/e2e/crud/team.spec.ts +287 -0
  553. package/templates/base/tests/e2e/crud/users.spec.ts +239 -0
  554. package/templates/base/tests/e2e/errors/error-boundary.spec.ts +428 -0
  555. package/templates/base/tests/e2e/errors/error-handling.spec.ts +47 -0
  556. package/templates/base/tests/e2e/errors/error-handling.unauth.spec.ts +205 -0
  557. package/templates/base/tests/e2e/fixtures.ts +266 -0
  558. package/templates/base/tests/e2e/forms/validation.spec.ts +569 -0
  559. package/templates/base/tests/e2e/invitations/invite-flow.unauth.spec.ts +204 -0
  560. package/templates/base/tests/e2e/journeys/account-lifecycle.spec.ts +314 -0
  561. package/templates/base/tests/e2e/journeys/audit-investigation.spec.ts +299 -0
  562. package/templates/base/tests/e2e/journeys/auth-onboarding.spec.ts +232 -0
  563. package/templates/base/tests/e2e/journeys/critical-flows.spec.ts +281 -0
  564. package/templates/base/tests/e2e/journeys/error-recovery.spec.ts +354 -0
  565. package/templates/base/tests/e2e/journeys/file-management.spec.ts +307 -0
  566. package/templates/base/tests/e2e/journeys/integrations.spec.ts +372 -0
  567. package/templates/base/tests/e2e/journeys/multi-account.spec.ts +317 -0
  568. package/templates/base/tests/e2e/journeys/rbac-enforcement.spec.ts +389 -0
  569. package/templates/base/tests/e2e/journeys/settings-profile.spec.ts +400 -0
  570. package/templates/base/tests/e2e/journeys/team-collaboration.spec.ts +410 -0
  571. package/templates/base/tests/e2e/mobile/responsive.spec.ts +178 -0
  572. package/templates/base/tests/e2e/navigation/routing.spec.ts +371 -0
  573. package/templates/base/tests/e2e/navigation/sidebar.spec.ts +425 -0
  574. package/templates/base/tests/e2e/pages/ui-features.spec.ts +393 -0
  575. package/templates/base/tests/e2e/performance/baselines.spec.ts +162 -0
  576. package/templates/base/tests/e2e/performance/benchmarks.spec.ts +371 -0
  577. package/templates/base/tests/e2e/smoke.unauth.spec.ts +196 -0
  578. package/templates/base/tests/e2e/visual/components.spec.ts +650 -0
  579. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/confirmation-dialog-visual-darwin.png +0 -0
  580. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/dark-mode-background-visual-darwin.png +0 -0
  581. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/dark-mode-card-visual-darwin.png +0 -0
  582. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/dark-mode-sidebar-visual-darwin.png +0 -0
  583. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/dashboard-card-single-visual-darwin.png +0 -0
  584. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/dialog-content-visual-darwin.png +0 -0
  585. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/dialog-with-backdrop-visual-darwin.png +0 -0
  586. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/empty-search-results-visual-darwin.png +0 -0
  587. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/input-default-visual-darwin.png +0 -0
  588. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/input-focus-ring-visual-darwin.png +0 -0
  589. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/primary-button-focus-visual-darwin.png +0 -0
  590. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/primary-button-hover-visual-darwin.png +0 -0
  591. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/primary-button-visual-darwin.png +0 -0
  592. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/sidebar-active-nav-item-visual-darwin.png +0 -0
  593. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/sidebar-collapsed-visual-darwin.png +0 -0
  594. package/templates/base/tests/e2e/visual/components.spec.ts-snapshots/sidebar-navigation-visual-darwin.png +0 -0
  595. package/templates/base/tests/e2e/visual/core-components.spec.ts +192 -0
  596. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/account-page-visual-darwin.png +0 -0
  597. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/create-webhook-dialog-visual-darwin.png +0 -0
  598. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/dashboard-page-visual-darwin.png +0 -0
  599. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/delete-account-dialog-visual-darwin.png +0 -0
  600. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/integrations-page-visual-darwin.png +0 -0
  601. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/invite-member-dialog-visual-darwin.png +0 -0
  602. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/login-page-visual-darwin.png +0 -0
  603. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/settings-account-tab-visual-darwin.png +0 -0
  604. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/settings-profile-tab-visual-darwin.png +0 -0
  605. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/sidebar-navigation-visual-darwin.png +0 -0
  606. package/templates/base/tests/e2e/visual/core-components.spec.ts-snapshots/team-page-visual-darwin.png +0 -0
  607. package/templates/base/tests/e2e/visual/screenshots.spec.ts +230 -0
  608. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/404-page-chromium-darwin.png +0 -0
  609. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/404-page-visual-darwin.png +0 -0
  610. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/account-page-visual-darwin.png +0 -0
  611. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/login-page-chromium-darwin.png +0 -0
  612. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/login-page-visual-darwin.png +0 -0
  613. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/settings-page-visual-darwin.png +0 -0
  614. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/team-invite-dialog-visual-darwin.png +0 -0
  615. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/team-page-visual-darwin.png +0 -0
  616. package/templates/base/tests/e2e/visual/screenshots.spec.ts-snapshots/webhook-create-dialog-visual-darwin.png +0 -0
  617. package/templates/base/tests/e2e/visual/theme-colors.spec.ts +293 -0
  618. package/templates/base/tests/e2e/visual/ui-components.spec.ts +502 -0
  619. package/templates/base/tests/integration/accounts/crud.test.ts +1402 -0
  620. package/templates/base/tests/integration/audits/list.test.ts +1133 -0
  621. package/templates/base/tests/integration/auth/auth-service.test.ts +415 -0
  622. package/templates/base/tests/integration/auth/invitation-token.test.ts +529 -0
  623. package/templates/base/tests/integration/auth/logout.test.ts +524 -0
  624. package/templates/base/tests/integration/auth/oauth.test.ts +768 -0
  625. package/templates/base/tests/integration/auth/refresh-token.test.ts +364 -0
  626. package/templates/base/tests/integration/auth/session-expiry.test.ts +569 -0
  627. package/templates/base/tests/integration/auth/session.test.ts +520 -0
  628. package/templates/base/tests/integration/auth/super-admin.test.ts +451 -0
  629. package/templates/base/tests/integration/authorization/analytics-role.test.ts +1026 -0
  630. package/templates/base/tests/integration/authorization/billing-role.test.ts +776 -0
  631. package/templates/base/tests/integration/authorization/guards-roles.test.ts +539 -0
  632. package/templates/base/tests/integration/authorization/multi-tenancy.test.ts +1112 -0
  633. package/templates/base/tests/integration/authorization/role-hierarchy.test.ts +931 -0
  634. package/templates/base/tests/integration/authorization/roles.test.ts +1347 -0
  635. package/templates/base/tests/integration/config/production-behavior.test.ts +536 -0
  636. package/templates/base/tests/integration/db/constraints.test.ts +695 -0
  637. package/templates/base/tests/integration/fixtures/accounts.ts +136 -0
  638. package/templates/base/tests/integration/fixtures/index.ts +232 -0
  639. package/templates/base/tests/integration/fixtures/invitations.ts +195 -0
  640. package/templates/base/tests/integration/fixtures/users.ts +144 -0
  641. package/templates/base/tests/integration/health/health.test.ts +351 -0
  642. package/templates/base/tests/integration/invitations/crud.test.ts +1457 -0
  643. package/templates/base/tests/integration/invitations/email.test.ts +506 -0
  644. package/templates/base/tests/integration/lib/email.test.ts +174 -0
  645. package/templates/base/tests/integration/lib/oauth.test.ts +368 -0
  646. package/templates/base/tests/integration/lib/password.test.ts +192 -0
  647. package/templates/base/tests/integration/lib/schema-helpers.test.ts +129 -0
  648. package/templates/base/tests/integration/lib/tokens.test.ts +304 -0
  649. package/templates/base/tests/integration/middleware/auth.test.ts +499 -0
  650. package/templates/base/tests/integration/middleware/cors.test.ts +334 -0
  651. package/templates/base/tests/integration/middleware/request-context.test.ts +156 -0
  652. package/templates/base/tests/integration/middleware/request-logger.test.ts +313 -0
  653. package/templates/base/tests/integration/performance/response-times.test.ts +509 -0
  654. package/templates/base/tests/integration/security/cookie-security.test.ts +567 -0
  655. package/templates/base/tests/integration/security/csrf-protection.test.ts +542 -0
  656. package/templates/base/tests/integration/security/jwt-validation.test.ts +209 -0
  657. package/templates/base/tests/integration/security/log-sanitization.test.ts +658 -0
  658. package/templates/base/tests/integration/security/rate-limiting.test.ts +1251 -0
  659. package/templates/base/tests/integration/security/sql-injection.test.ts +663 -0
  660. package/templates/base/tests/integration/security/token-hashing.test.ts +371 -0
  661. package/templates/base/tests/integration/security/xss-prevention.test.ts +541 -0
  662. package/templates/base/tests/integration/setup.ts +834 -0
  663. package/templates/base/tests/integration/smoke.test.ts +288 -0
  664. package/templates/base/tests/integration/storage/upload.test.ts +1162 -0
  665. package/templates/base/tests/integration/storage/validation.test.ts +746 -0
  666. package/templates/base/tests/integration/users/crud.test.ts +1297 -0
  667. package/templates/base/tests/integration/users/list.test.ts +698 -0
  668. package/templates/base/tests/integration/vitest.config.ts +80 -0
  669. package/templates/base/tsconfig.app.json +18 -0
  670. package/templates/base/tsconfig.json +39 -0
  671. package/templates/base/tsconfig.node.json +16 -0
  672. package/templates/base/tsconfig.server.json +26 -0
  673. package/templates/base/tsconfig.tsbuildinfo +1 -0
  674. package/templates/base/vite.config.ts +46 -0
  675. package/templates/base/vitest.config.browser.ts +47 -0
  676. package/templates/base/vitest.config.frontend.ts +47 -0
  677. package/templates/base/vitest.config.ts +82 -0
  678. package/templates/base/vitest.workspace.ts +22 -0
  679. package/templates/modules/audit-logs/.gitkeep +0 -0
  680. package/templates/modules/billing/.gitkeep +0 -0
  681. package/templates/modules/invitations/.gitkeep +0 -0
  682. package/templates/modules/storage/.gitkeep +0 -0
  683. package/templates/modules/webhooks/.gitkeep +0 -0
  684. package/templates/providers/auth-email/.gitkeep +0 -0
  685. package/templates/providers/auth-github/.gitkeep +0 -0
  686. package/templates/providers/auth-google/.gitkeep +0 -0
  687. package/templates/providers/email-resend/.gitkeep +0 -0
  688. package/templates/providers/email-sendgrid/.gitkeep +0 -0
@@ -0,0 +1,877 @@
1
+ /**
2
+ * Browser-injectable snapshot script.
3
+ *
4
+ * This module provides the snapshot functionality as a string that can be
5
+ * injected into the browser via page.addScriptTag() or page.evaluate().
6
+ *
7
+ * The approach is to read the compiled JavaScript at runtime and bundle it
8
+ * into a single script that exposes window.__devBrowser_getAISnapshot() and
9
+ * window.__devBrowser_selectSnapshotRef().
10
+ */
11
+
12
+ import * as fs from "fs";
13
+ import * as path from "path";
14
+
15
+ // Cache the bundled script
16
+ let cachedScript: string | null = null;
17
+
18
+ /**
19
+ * Get the snapshot script that can be injected into the browser.
20
+ * Returns a self-contained JavaScript string that:
21
+ * 1. Defines all necessary functions (domUtils, roleUtils, yaml, ariaSnapshot)
22
+ * 2. Exposes window.__devBrowser_getAISnapshot()
23
+ * 3. Exposes window.__devBrowser_selectSnapshotRef()
24
+ */
25
+ export function getSnapshotScript(): string {
26
+ if (cachedScript) return cachedScript;
27
+
28
+ // Read the compiled JavaScript files
29
+ const snapshotDir = path.dirname(new URL(import.meta.url).pathname);
30
+
31
+ // For now, we'll inline the functions directly
32
+ // In production, we could use a bundler like esbuild to create a single file
33
+ cachedScript = `
34
+ (function() {
35
+ // Skip if already injected
36
+ if (window.__devBrowser_getAISnapshot) return;
37
+
38
+ ${getDomUtilsCode()}
39
+ ${getYamlCode()}
40
+ ${getRoleUtilsCode()}
41
+ ${getAriaSnapshotCode()}
42
+
43
+ // Expose main functions
44
+ window.__devBrowser_getAISnapshot = getAISnapshot;
45
+ window.__devBrowser_selectSnapshotRef = selectSnapshotRef;
46
+ })();
47
+ `;
48
+
49
+ return cachedScript;
50
+ }
51
+
52
+ function getDomUtilsCode(): string {
53
+ return `
54
+ // === domUtils ===
55
+ let cacheStyle;
56
+ let cachesCounter = 0;
57
+
58
+ function beginDOMCaches() {
59
+ ++cachesCounter;
60
+ cacheStyle = cacheStyle || new Map();
61
+ }
62
+
63
+ function endDOMCaches() {
64
+ if (!--cachesCounter) {
65
+ cacheStyle = undefined;
66
+ }
67
+ }
68
+
69
+ function getElementComputedStyle(element, pseudo) {
70
+ const cache = cacheStyle;
71
+ const cacheKey = pseudo ? undefined : element;
72
+ if (cache && cacheKey && cache.has(cacheKey)) return cache.get(cacheKey);
73
+ const style = element.ownerDocument && element.ownerDocument.defaultView
74
+ ? element.ownerDocument.defaultView.getComputedStyle(element, pseudo)
75
+ : undefined;
76
+ if (cache && cacheKey) cache.set(cacheKey, style);
77
+ return style;
78
+ }
79
+
80
+ function parentElementOrShadowHost(element) {
81
+ if (element.parentElement) return element.parentElement;
82
+ if (!element.parentNode) return;
83
+ if (element.parentNode.nodeType === 11 && element.parentNode.host)
84
+ return element.parentNode.host;
85
+ }
86
+
87
+ function enclosingShadowRootOrDocument(element) {
88
+ let node = element;
89
+ while (node.parentNode) node = node.parentNode;
90
+ if (node.nodeType === 11 || node.nodeType === 9)
91
+ return node;
92
+ }
93
+
94
+ function closestCrossShadow(element, css, scope) {
95
+ while (element) {
96
+ const closest = element.closest(css);
97
+ if (scope && closest !== scope && closest?.contains(scope)) return;
98
+ if (closest) return closest;
99
+ element = enclosingShadowHost(element);
100
+ }
101
+ }
102
+
103
+ function enclosingShadowHost(element) {
104
+ while (element.parentElement) element = element.parentElement;
105
+ return parentElementOrShadowHost(element);
106
+ }
107
+
108
+ function isElementStyleVisibilityVisible(element, style) {
109
+ style = style || getElementComputedStyle(element);
110
+ if (!style) return true;
111
+ if (style.visibility !== "visible") return false;
112
+ const detailsOrSummary = element.closest("details,summary");
113
+ if (detailsOrSummary !== element && detailsOrSummary?.nodeName === "DETAILS" && !detailsOrSummary.open)
114
+ return false;
115
+ return true;
116
+ }
117
+
118
+ function computeBox(element) {
119
+ const style = getElementComputedStyle(element);
120
+ if (!style) return { visible: true, inline: false };
121
+ const cursor = style.cursor;
122
+ if (style.display === "contents") {
123
+ for (let child = element.firstChild; child; child = child.nextSibling) {
124
+ if (child.nodeType === 1 && isElementVisible(child))
125
+ return { visible: true, inline: false, cursor };
126
+ if (child.nodeType === 3 && isVisibleTextNode(child))
127
+ return { visible: true, inline: true, cursor };
128
+ }
129
+ return { visible: false, inline: false, cursor };
130
+ }
131
+ if (!isElementStyleVisibilityVisible(element, style))
132
+ return { cursor, visible: false, inline: false };
133
+ const rect = element.getBoundingClientRect();
134
+ return { rect, cursor, visible: rect.width > 0 && rect.height > 0, inline: style.display === "inline" };
135
+ }
136
+
137
+ function isElementVisible(element) {
138
+ return computeBox(element).visible;
139
+ }
140
+
141
+ function isVisibleTextNode(node) {
142
+ const range = node.ownerDocument.createRange();
143
+ range.selectNode(node);
144
+ const rect = range.getBoundingClientRect();
145
+ return rect.width > 0 && rect.height > 0;
146
+ }
147
+
148
+ function elementSafeTagName(element) {
149
+ const tagName = element.tagName;
150
+ if (typeof tagName === "string") return tagName.toUpperCase();
151
+ if (element instanceof HTMLFormElement) return "FORM";
152
+ return element.tagName.toUpperCase();
153
+ }
154
+
155
+ function normalizeWhiteSpace(text) {
156
+ return text.split("\\u00A0").map(chunk =>
157
+ chunk.replace(/\\r\\n/g, "\\n").replace(/[\\u200b\\u00ad]/g, "").replace(/\\s\\s*/g, " ")
158
+ ).join("\\u00A0").trim();
159
+ }
160
+ `;
161
+ }
162
+
163
+ function getYamlCode(): string {
164
+ return `
165
+ // === yaml ===
166
+ function yamlEscapeKeyIfNeeded(str) {
167
+ if (!yamlStringNeedsQuotes(str)) return str;
168
+ return "'" + str.replace(/'/g, "''") + "'";
169
+ }
170
+
171
+ function yamlEscapeValueIfNeeded(str) {
172
+ if (!yamlStringNeedsQuotes(str)) return str;
173
+ return '"' + str.replace(/[\\\\"\x00-\\x1f\\x7f-\\x9f]/g, c => {
174
+ switch (c) {
175
+ case "\\\\": return "\\\\\\\\";
176
+ case '"': return '\\\\"';
177
+ case "\\b": return "\\\\b";
178
+ case "\\f": return "\\\\f";
179
+ case "\\n": return "\\\\n";
180
+ case "\\r": return "\\\\r";
181
+ case "\\t": return "\\\\t";
182
+ default:
183
+ const code = c.charCodeAt(0);
184
+ return "\\\\x" + code.toString(16).padStart(2, "0");
185
+ }
186
+ }) + '"';
187
+ }
188
+
189
+ function yamlStringNeedsQuotes(str) {
190
+ if (str.length === 0) return true;
191
+ if (/^\\s|\\s$/.test(str)) return true;
192
+ if (/[\\x00-\\x08\\x0b\\x0c\\x0e-\\x1f\\x7f-\\x9f]/.test(str)) return true;
193
+ if (/^-/.test(str)) return true;
194
+ if (/[\\n:](\\s|$)/.test(str)) return true;
195
+ if (/\\s#/.test(str)) return true;
196
+ if (/[\\n\\r]/.test(str)) return true;
197
+ if (/^[&*\\],?!>|@"'#%]/.test(str)) return true;
198
+ if (/[{}\`]/.test(str)) return true;
199
+ if (/^\\[/.test(str)) return true;
200
+ if (!isNaN(Number(str)) || ["y","n","yes","no","true","false","on","off","null"].includes(str.toLowerCase())) return true;
201
+ return false;
202
+ }
203
+ `;
204
+ }
205
+
206
+ function getRoleUtilsCode(): string {
207
+ return `
208
+ // === roleUtils ===
209
+ const validRoles = ["alert","alertdialog","application","article","banner","blockquote","button","caption","cell","checkbox","code","columnheader","combobox","complementary","contentinfo","definition","deletion","dialog","directory","document","emphasis","feed","figure","form","generic","grid","gridcell","group","heading","img","insertion","link","list","listbox","listitem","log","main","mark","marquee","math","meter","menu","menubar","menuitem","menuitemcheckbox","menuitemradio","navigation","none","note","option","paragraph","presentation","progressbar","radio","radiogroup","region","row","rowgroup","rowheader","scrollbar","search","searchbox","separator","slider","spinbutton","status","strong","subscript","superscript","switch","tab","table","tablist","tabpanel","term","textbox","time","timer","toolbar","tooltip","tree","treegrid","treeitem"];
210
+
211
+ let cacheAccessibleName;
212
+ let cacheIsHidden;
213
+ let cachePointerEvents;
214
+ let ariaCachesCounter = 0;
215
+
216
+ function beginAriaCaches() {
217
+ beginDOMCaches();
218
+ ++ariaCachesCounter;
219
+ cacheAccessibleName = cacheAccessibleName || new Map();
220
+ cacheIsHidden = cacheIsHidden || new Map();
221
+ cachePointerEvents = cachePointerEvents || new Map();
222
+ }
223
+
224
+ function endAriaCaches() {
225
+ if (!--ariaCachesCounter) {
226
+ cacheAccessibleName = undefined;
227
+ cacheIsHidden = undefined;
228
+ cachePointerEvents = undefined;
229
+ }
230
+ endDOMCaches();
231
+ }
232
+
233
+ function hasExplicitAccessibleName(e) {
234
+ return e.hasAttribute("aria-label") || e.hasAttribute("aria-labelledby");
235
+ }
236
+
237
+ const kAncestorPreventingLandmark = "article:not([role]), aside:not([role]), main:not([role]), nav:not([role]), section:not([role]), [role=article], [role=complementary], [role=main], [role=navigation], [role=region]";
238
+
239
+ const kGlobalAriaAttributes = [
240
+ ["aria-atomic", undefined],["aria-busy", undefined],["aria-controls", undefined],["aria-current", undefined],
241
+ ["aria-describedby", undefined],["aria-details", undefined],["aria-dropeffect", undefined],["aria-flowto", undefined],
242
+ ["aria-grabbed", undefined],["aria-hidden", undefined],["aria-keyshortcuts", undefined],
243
+ ["aria-label", ["caption","code","deletion","emphasis","generic","insertion","paragraph","presentation","strong","subscript","superscript"]],
244
+ ["aria-labelledby", ["caption","code","deletion","emphasis","generic","insertion","paragraph","presentation","strong","subscript","superscript"]],
245
+ ["aria-live", undefined],["aria-owns", undefined],["aria-relevant", undefined],["aria-roledescription", ["generic"]]
246
+ ];
247
+
248
+ function hasGlobalAriaAttribute(element, forRole) {
249
+ return kGlobalAriaAttributes.some(([attr, prohibited]) => !prohibited?.includes(forRole || "") && element.hasAttribute(attr));
250
+ }
251
+
252
+ function hasTabIndex(element) {
253
+ return !Number.isNaN(Number(String(element.getAttribute("tabindex"))));
254
+ }
255
+
256
+ function isFocusable(element) {
257
+ return !isNativelyDisabled(element) && (isNativelyFocusable(element) || hasTabIndex(element));
258
+ }
259
+
260
+ function isNativelyFocusable(element) {
261
+ const tagName = elementSafeTagName(element);
262
+ if (["BUTTON","DETAILS","SELECT","TEXTAREA"].includes(tagName)) return true;
263
+ if (tagName === "A" || tagName === "AREA") return element.hasAttribute("href");
264
+ if (tagName === "INPUT") return !element.hidden;
265
+ return false;
266
+ }
267
+
268
+ function isNativelyDisabled(element) {
269
+ const isNativeFormControl = ["BUTTON","INPUT","SELECT","TEXTAREA","OPTION","OPTGROUP"].includes(elementSafeTagName(element));
270
+ return isNativeFormControl && (element.hasAttribute("disabled") || belongsToDisabledFieldSet(element));
271
+ }
272
+
273
+ function belongsToDisabledFieldSet(element) {
274
+ const fieldSetElement = element?.closest("FIELDSET[DISABLED]");
275
+ if (!fieldSetElement) return false;
276
+ const legendElement = fieldSetElement.querySelector(":scope > LEGEND");
277
+ return !legendElement || !legendElement.contains(element);
278
+ }
279
+
280
+ const inputTypeToRole = {button:"button",checkbox:"checkbox",image:"button",number:"spinbutton",radio:"radio",range:"slider",reset:"button",submit:"button"};
281
+
282
+ function getIdRefs(element, ref) {
283
+ if (!ref) return [];
284
+ const root = enclosingShadowRootOrDocument(element);
285
+ if (!root) return [];
286
+ try {
287
+ const ids = ref.split(" ").filter(id => !!id);
288
+ const result = [];
289
+ for (const id of ids) {
290
+ const firstElement = root.querySelector("#" + CSS.escape(id));
291
+ if (firstElement && !result.includes(firstElement)) result.push(firstElement);
292
+ }
293
+ return result;
294
+ } catch { return []; }
295
+ }
296
+
297
+ const kImplicitRoleByTagName = {
298
+ A: e => e.hasAttribute("href") ? "link" : null,
299
+ AREA: e => e.hasAttribute("href") ? "link" : null,
300
+ ARTICLE: () => "article", ASIDE: () => "complementary", BLOCKQUOTE: () => "blockquote", BUTTON: () => "button",
301
+ CAPTION: () => "caption", CODE: () => "code", DATALIST: () => "listbox", DD: () => "definition",
302
+ DEL: () => "deletion", DETAILS: () => "group", DFN: () => "term", DIALOG: () => "dialog", DT: () => "term",
303
+ EM: () => "emphasis", FIELDSET: () => "group", FIGURE: () => "figure",
304
+ FOOTER: e => closestCrossShadow(e, kAncestorPreventingLandmark) ? null : "contentinfo",
305
+ FORM: e => hasExplicitAccessibleName(e) ? "form" : null,
306
+ H1: () => "heading", H2: () => "heading", H3: () => "heading", H4: () => "heading", H5: () => "heading", H6: () => "heading",
307
+ HEADER: e => closestCrossShadow(e, kAncestorPreventingLandmark) ? null : "banner",
308
+ HR: () => "separator", HTML: () => "document",
309
+ IMG: e => e.getAttribute("alt") === "" && !e.getAttribute("title") && !hasGlobalAriaAttribute(e) && !hasTabIndex(e) ? "presentation" : "img",
310
+ INPUT: e => {
311
+ const type = e.type.toLowerCase();
312
+ if (type === "search") return e.hasAttribute("list") ? "combobox" : "searchbox";
313
+ if (["email","tel","text","url",""].includes(type)) {
314
+ const list = getIdRefs(e, e.getAttribute("list"))[0];
315
+ return list && elementSafeTagName(list) === "DATALIST" ? "combobox" : "textbox";
316
+ }
317
+ if (type === "hidden") return null;
318
+ if (type === "file") return "button";
319
+ return inputTypeToRole[type] || "textbox";
320
+ },
321
+ INS: () => "insertion", LI: () => "listitem", MAIN: () => "main", MARK: () => "mark", MATH: () => "math",
322
+ MENU: () => "list", METER: () => "meter", NAV: () => "navigation", OL: () => "list", OPTGROUP: () => "group",
323
+ OPTION: () => "option", OUTPUT: () => "status", P: () => "paragraph", PROGRESS: () => "progressbar",
324
+ SEARCH: () => "search", SECTION: e => hasExplicitAccessibleName(e) ? "region" : null,
325
+ SELECT: e => e.hasAttribute("multiple") || e.size > 1 ? "listbox" : "combobox",
326
+ STRONG: () => "strong", SUB: () => "subscript", SUP: () => "superscript", SVG: () => "img",
327
+ TABLE: () => "table", TBODY: () => "rowgroup",
328
+ TD: e => { const table = closestCrossShadow(e, "table"); const role = table ? getExplicitAriaRole(table) : ""; return role === "grid" || role === "treegrid" ? "gridcell" : "cell"; },
329
+ TEXTAREA: () => "textbox", TFOOT: () => "rowgroup",
330
+ TH: e => { const scope = e.getAttribute("scope"); if (scope === "col" || scope === "colgroup") return "columnheader"; if (scope === "row" || scope === "rowgroup") return "rowheader"; return "columnheader"; },
331
+ THEAD: () => "rowgroup", TIME: () => "time", TR: () => "row", UL: () => "list"
332
+ };
333
+
334
+ function getExplicitAriaRole(element) {
335
+ const roles = (element.getAttribute("role") || "").split(" ").map(role => role.trim());
336
+ return roles.find(role => validRoles.includes(role)) || null;
337
+ }
338
+
339
+ function getImplicitAriaRole(element) {
340
+ const fn = kImplicitRoleByTagName[elementSafeTagName(element)];
341
+ return fn ? fn(element) : null;
342
+ }
343
+
344
+ function hasPresentationConflictResolution(element, role) {
345
+ return hasGlobalAriaAttribute(element, role) || isFocusable(element);
346
+ }
347
+
348
+ function getAriaRole(element) {
349
+ const explicitRole = getExplicitAriaRole(element);
350
+ if (!explicitRole) return getImplicitAriaRole(element);
351
+ if (explicitRole === "none" || explicitRole === "presentation") {
352
+ const implicitRole = getImplicitAriaRole(element);
353
+ if (hasPresentationConflictResolution(element, implicitRole)) return implicitRole;
354
+ }
355
+ return explicitRole;
356
+ }
357
+
358
+ function getAriaBoolean(attr) {
359
+ return attr === null ? undefined : attr.toLowerCase() === "true";
360
+ }
361
+
362
+ function isElementIgnoredForAria(element) {
363
+ return ["STYLE","SCRIPT","NOSCRIPT","TEMPLATE"].includes(elementSafeTagName(element));
364
+ }
365
+
366
+ function isElementHiddenForAria(element) {
367
+ if (isElementIgnoredForAria(element)) return true;
368
+ const style = getElementComputedStyle(element);
369
+ const isSlot = element.nodeName === "SLOT";
370
+ if (style?.display === "contents" && !isSlot) {
371
+ for (let child = element.firstChild; child; child = child.nextSibling) {
372
+ if (child.nodeType === 1 && !isElementHiddenForAria(child)) return false;
373
+ if (child.nodeType === 3 && isVisibleTextNode(child)) return false;
374
+ }
375
+ return true;
376
+ }
377
+ const isOptionInsideSelect = element.nodeName === "OPTION" && !!element.closest("select");
378
+ if (!isOptionInsideSelect && !isSlot && !isElementStyleVisibilityVisible(element, style)) return true;
379
+ return belongsToDisplayNoneOrAriaHiddenOrNonSlotted(element);
380
+ }
381
+
382
+ function belongsToDisplayNoneOrAriaHiddenOrNonSlotted(element) {
383
+ let hidden = cacheIsHidden?.get(element);
384
+ if (hidden === undefined) {
385
+ hidden = false;
386
+ if (element.parentElement && element.parentElement.shadowRoot && !element.assignedSlot) hidden = true;
387
+ if (!hidden) {
388
+ const style = getElementComputedStyle(element);
389
+ hidden = !style || style.display === "none" || getAriaBoolean(element.getAttribute("aria-hidden")) === true;
390
+ }
391
+ if (!hidden) {
392
+ const parent = parentElementOrShadowHost(element);
393
+ if (parent) hidden = belongsToDisplayNoneOrAriaHiddenOrNonSlotted(parent);
394
+ }
395
+ cacheIsHidden?.set(element, hidden);
396
+ }
397
+ return hidden;
398
+ }
399
+
400
+ function getAriaLabelledByElements(element) {
401
+ const ref = element.getAttribute("aria-labelledby");
402
+ if (ref === null) return null;
403
+ const refs = getIdRefs(element, ref);
404
+ return refs.length ? refs : null;
405
+ }
406
+
407
+ function getElementAccessibleName(element, includeHidden) {
408
+ let accessibleName = cacheAccessibleName?.get(element);
409
+ if (accessibleName === undefined) {
410
+ accessibleName = "";
411
+ const elementProhibitsNaming = ["caption","code","definition","deletion","emphasis","generic","insertion","mark","paragraph","presentation","strong","subscript","suggestion","superscript","term","time"].includes(getAriaRole(element) || "");
412
+ if (!elementProhibitsNaming) {
413
+ accessibleName = normalizeWhiteSpace(getTextAlternativeInternal(element, { includeHidden, visitedElements: new Set(), embeddedInTargetElement: "self" }));
414
+ }
415
+ cacheAccessibleName?.set(element, accessibleName);
416
+ }
417
+ return accessibleName;
418
+ }
419
+
420
+ function getTextAlternativeInternal(element, options) {
421
+ if (options.visitedElements.has(element)) return "";
422
+ const childOptions = { ...options, embeddedInTargetElement: options.embeddedInTargetElement === "self" ? "descendant" : options.embeddedInTargetElement };
423
+
424
+ if (!options.includeHidden) {
425
+ const isEmbeddedInHiddenReferenceTraversal = !!options.embeddedInLabelledBy?.hidden || !!options.embeddedInLabel?.hidden;
426
+ if (isElementIgnoredForAria(element) || (!isEmbeddedInHiddenReferenceTraversal && isElementHiddenForAria(element))) {
427
+ options.visitedElements.add(element);
428
+ return "";
429
+ }
430
+ }
431
+
432
+ const labelledBy = getAriaLabelledByElements(element);
433
+ if (!options.embeddedInLabelledBy) {
434
+ const accessibleName = (labelledBy || []).map(ref => getTextAlternativeInternal(ref, { ...options, embeddedInLabelledBy: { element: ref, hidden: isElementHiddenForAria(ref) }, embeddedInTargetElement: undefined, embeddedInLabel: undefined })).join(" ");
435
+ if (accessibleName) return accessibleName;
436
+ }
437
+
438
+ const role = getAriaRole(element) || "";
439
+ const tagName = elementSafeTagName(element);
440
+
441
+ const ariaLabel = element.getAttribute("aria-label") || "";
442
+ if (ariaLabel.trim()) { options.visitedElements.add(element); return ariaLabel; }
443
+
444
+ if (!["presentation","none"].includes(role)) {
445
+ if (tagName === "INPUT" && ["button","submit","reset"].includes(element.type)) {
446
+ options.visitedElements.add(element);
447
+ const value = element.value || "";
448
+ if (value.trim()) return value;
449
+ if (element.type === "submit") return "Submit";
450
+ if (element.type === "reset") return "Reset";
451
+ return element.getAttribute("title") || "";
452
+ }
453
+ if (tagName === "INPUT" && element.type === "image") {
454
+ options.visitedElements.add(element);
455
+ const alt = element.getAttribute("alt") || "";
456
+ if (alt.trim()) return alt;
457
+ const title = element.getAttribute("title") || "";
458
+ if (title.trim()) return title;
459
+ return "Submit";
460
+ }
461
+ if (tagName === "IMG") {
462
+ options.visitedElements.add(element);
463
+ const alt = element.getAttribute("alt") || "";
464
+ if (alt.trim()) return alt;
465
+ return element.getAttribute("title") || "";
466
+ }
467
+ if (!labelledBy && ["BUTTON","INPUT","TEXTAREA","SELECT"].includes(tagName)) {
468
+ const labels = element.labels;
469
+ if (labels?.length) {
470
+ options.visitedElements.add(element);
471
+ return [...labels].map(label => getTextAlternativeInternal(label, { ...options, embeddedInLabel: { element: label, hidden: isElementHiddenForAria(label) }, embeddedInLabelledBy: undefined, embeddedInTargetElement: undefined })).filter(name => !!name).join(" ");
472
+ }
473
+ }
474
+ }
475
+
476
+ const allowsNameFromContent = ["button","cell","checkbox","columnheader","gridcell","heading","link","menuitem","menuitemcheckbox","menuitemradio","option","radio","row","rowheader","switch","tab","tooltip","treeitem"].includes(role);
477
+ if (allowsNameFromContent || !!options.embeddedInLabelledBy || !!options.embeddedInLabel) {
478
+ options.visitedElements.add(element);
479
+ const accessibleName = innerAccumulatedElementText(element, childOptions);
480
+ const maybeTrimmedAccessibleName = options.embeddedInTargetElement === "self" ? accessibleName.trim() : accessibleName;
481
+ if (maybeTrimmedAccessibleName) return accessibleName;
482
+ }
483
+
484
+ if (!["presentation","none"].includes(role) || tagName === "IFRAME") {
485
+ options.visitedElements.add(element);
486
+ const title = element.getAttribute("title") || "";
487
+ if (title.trim()) return title;
488
+ }
489
+
490
+ options.visitedElements.add(element);
491
+ return "";
492
+ }
493
+
494
+ function innerAccumulatedElementText(element, options) {
495
+ const tokens = [];
496
+ const visit = (node, skipSlotted) => {
497
+ if (skipSlotted && node.assignedSlot) return;
498
+ if (node.nodeType === 1) {
499
+ const display = getElementComputedStyle(node)?.display || "inline";
500
+ let token = getTextAlternativeInternal(node, options);
501
+ if (display !== "inline" || node.nodeName === "BR") token = " " + token + " ";
502
+ tokens.push(token);
503
+ } else if (node.nodeType === 3) {
504
+ tokens.push(node.textContent || "");
505
+ }
506
+ };
507
+ const assignedNodes = element.nodeName === "SLOT" ? element.assignedNodes() : [];
508
+ if (assignedNodes.length) {
509
+ for (const child of assignedNodes) visit(child, false);
510
+ } else {
511
+ for (let child = element.firstChild; child; child = child.nextSibling) visit(child, true);
512
+ if (element.shadowRoot) {
513
+ for (let child = element.shadowRoot.firstChild; child; child = child.nextSibling) visit(child, true);
514
+ }
515
+ }
516
+ return tokens.join("");
517
+ }
518
+
519
+ const kAriaCheckedRoles = ["checkbox","menuitemcheckbox","option","radio","switch","menuitemradio","treeitem"];
520
+ function getAriaChecked(element) {
521
+ const tagName = elementSafeTagName(element);
522
+ if (tagName === "INPUT" && element.indeterminate) return "mixed";
523
+ if (tagName === "INPUT" && ["checkbox","radio"].includes(element.type)) return element.checked;
524
+ if (kAriaCheckedRoles.includes(getAriaRole(element) || "")) {
525
+ const checked = element.getAttribute("aria-checked");
526
+ if (checked === "true") return true;
527
+ if (checked === "mixed") return "mixed";
528
+ return false;
529
+ }
530
+ return false;
531
+ }
532
+
533
+ const kAriaDisabledRoles = ["application","button","composite","gridcell","group","input","link","menuitem","scrollbar","separator","tab","checkbox","columnheader","combobox","grid","listbox","menu","menubar","menuitemcheckbox","menuitemradio","option","radio","radiogroup","row","rowheader","searchbox","select","slider","spinbutton","switch","tablist","textbox","toolbar","tree","treegrid","treeitem"];
534
+ function getAriaDisabled(element) {
535
+ return isNativelyDisabled(element) || hasExplicitAriaDisabled(element);
536
+ }
537
+ function hasExplicitAriaDisabled(element, isAncestor) {
538
+ if (!element) return false;
539
+ if (isAncestor || kAriaDisabledRoles.includes(getAriaRole(element) || "")) {
540
+ const attribute = (element.getAttribute("aria-disabled") || "").toLowerCase();
541
+ if (attribute === "true") return true;
542
+ if (attribute === "false") return false;
543
+ return hasExplicitAriaDisabled(parentElementOrShadowHost(element), true);
544
+ }
545
+ return false;
546
+ }
547
+
548
+ const kAriaExpandedRoles = ["application","button","checkbox","combobox","gridcell","link","listbox","menuitem","row","rowheader","tab","treeitem","columnheader","menuitemcheckbox","menuitemradio","switch"];
549
+ function getAriaExpanded(element) {
550
+ if (elementSafeTagName(element) === "DETAILS") return element.open;
551
+ if (kAriaExpandedRoles.includes(getAriaRole(element) || "")) {
552
+ const expanded = element.getAttribute("aria-expanded");
553
+ if (expanded === null) return undefined;
554
+ if (expanded === "true") return true;
555
+ return false;
556
+ }
557
+ return undefined;
558
+ }
559
+
560
+ const kAriaLevelRoles = ["heading","listitem","row","treeitem"];
561
+ function getAriaLevel(element) {
562
+ const native = {H1:1,H2:2,H3:3,H4:4,H5:5,H6:6}[elementSafeTagName(element)];
563
+ if (native) return native;
564
+ if (kAriaLevelRoles.includes(getAriaRole(element) || "")) {
565
+ const attr = element.getAttribute("aria-level");
566
+ const value = attr === null ? Number.NaN : Number(attr);
567
+ if (Number.isInteger(value) && value >= 1) return value;
568
+ }
569
+ return 0;
570
+ }
571
+
572
+ const kAriaPressedRoles = ["button"];
573
+ function getAriaPressed(element) {
574
+ if (kAriaPressedRoles.includes(getAriaRole(element) || "")) {
575
+ const pressed = element.getAttribute("aria-pressed");
576
+ if (pressed === "true") return true;
577
+ if (pressed === "mixed") return "mixed";
578
+ }
579
+ return false;
580
+ }
581
+
582
+ const kAriaSelectedRoles = ["gridcell","option","row","tab","rowheader","columnheader","treeitem"];
583
+ function getAriaSelected(element) {
584
+ if (elementSafeTagName(element) === "OPTION") return element.selected;
585
+ if (kAriaSelectedRoles.includes(getAriaRole(element) || "")) return getAriaBoolean(element.getAttribute("aria-selected")) === true;
586
+ return false;
587
+ }
588
+
589
+ function receivesPointerEvents(element) {
590
+ const cache = cachePointerEvents;
591
+ let e = element;
592
+ let result;
593
+ const parents = [];
594
+ for (; e; e = parentElementOrShadowHost(e)) {
595
+ const cached = cache?.get(e);
596
+ if (cached !== undefined) { result = cached; break; }
597
+ parents.push(e);
598
+ const style = getElementComputedStyle(e);
599
+ if (!style) { result = true; break; }
600
+ const value = style.pointerEvents;
601
+ if (value) { result = value !== "none"; break; }
602
+ }
603
+ if (result === undefined) result = true;
604
+ for (const parent of parents) cache?.set(parent, result);
605
+ return result;
606
+ }
607
+
608
+ function getCSSContent(element, pseudo) {
609
+ const style = getElementComputedStyle(element, pseudo);
610
+ if (!style) return undefined;
611
+ const contentValue = style.content;
612
+ if (!contentValue || contentValue === "none" || contentValue === "normal") return undefined;
613
+ if (style.display === "none" || style.visibility === "hidden") return undefined;
614
+ const match = contentValue.match(/^"(.*)"$/);
615
+ if (match) {
616
+ const content = match[1].replace(/\\\\"/g, '"');
617
+ if (pseudo) {
618
+ const display = style.display || "inline";
619
+ if (display !== "inline") return " " + content + " ";
620
+ }
621
+ return content;
622
+ }
623
+ return undefined;
624
+ }
625
+ `;
626
+ }
627
+
628
+ function getAriaSnapshotCode(): string {
629
+ return `
630
+ // === ariaSnapshot ===
631
+ let lastRef = 0;
632
+
633
+ function generateAriaTree(rootElement) {
634
+ const options = { visibility: "ariaOrVisible", refs: "interactable", refPrefix: "", includeGenericRole: true, renderActive: true, renderCursorPointer: true };
635
+ const visited = new Set();
636
+ const snapshot = {
637
+ root: { role: "fragment", name: "", children: [], element: rootElement, props: {}, box: computeBox(rootElement), receivesPointerEvents: true },
638
+ elements: new Map(),
639
+ refs: new Map(),
640
+ iframeRefs: []
641
+ };
642
+
643
+ const visit = (ariaNode, node, parentElementVisible) => {
644
+ if (visited.has(node)) return;
645
+ visited.add(node);
646
+ if (node.nodeType === Node.TEXT_NODE && node.nodeValue) {
647
+ if (!parentElementVisible) return;
648
+ const text = node.nodeValue;
649
+ if (ariaNode.role !== "textbox" && text) ariaNode.children.push(node.nodeValue || "");
650
+ return;
651
+ }
652
+ if (node.nodeType !== Node.ELEMENT_NODE) return;
653
+ const element = node;
654
+ const isElementVisibleForAria = !isElementHiddenForAria(element);
655
+ let visible = isElementVisibleForAria;
656
+ if (options.visibility === "ariaOrVisible") visible = isElementVisibleForAria || isElementVisible(element);
657
+ if (options.visibility === "ariaAndVisible") visible = isElementVisibleForAria && isElementVisible(element);
658
+ if (options.visibility === "aria" && !visible) return;
659
+ const ariaChildren = [];
660
+ if (element.hasAttribute("aria-owns")) {
661
+ const ids = element.getAttribute("aria-owns").split(/\\s+/);
662
+ for (const id of ids) {
663
+ const ownedElement = rootElement.ownerDocument.getElementById(id);
664
+ if (ownedElement) ariaChildren.push(ownedElement);
665
+ }
666
+ }
667
+ const childAriaNode = visible ? toAriaNode(element, options) : null;
668
+ if (childAriaNode) {
669
+ if (childAriaNode.ref) {
670
+ snapshot.elements.set(childAriaNode.ref, element);
671
+ snapshot.refs.set(element, childAriaNode.ref);
672
+ if (childAriaNode.role === "iframe") snapshot.iframeRefs.push(childAriaNode.ref);
673
+ }
674
+ ariaNode.children.push(childAriaNode);
675
+ }
676
+ processElement(childAriaNode || ariaNode, element, ariaChildren, visible);
677
+ };
678
+
679
+ function processElement(ariaNode, element, ariaChildren, parentElementVisible) {
680
+ const display = getElementComputedStyle(element)?.display || "inline";
681
+ const treatAsBlock = display !== "inline" || element.nodeName === "BR" ? " " : "";
682
+ if (treatAsBlock) ariaNode.children.push(treatAsBlock);
683
+ ariaNode.children.push(getCSSContent(element, "::before") || "");
684
+ const assignedNodes = element.nodeName === "SLOT" ? element.assignedNodes() : [];
685
+ if (assignedNodes.length) {
686
+ for (const child of assignedNodes) visit(ariaNode, child, parentElementVisible);
687
+ } else {
688
+ for (let child = element.firstChild; child; child = child.nextSibling) {
689
+ if (!child.assignedSlot) visit(ariaNode, child, parentElementVisible);
690
+ }
691
+ if (element.shadowRoot) {
692
+ for (let child = element.shadowRoot.firstChild; child; child = child.nextSibling) visit(ariaNode, child, parentElementVisible);
693
+ }
694
+ }
695
+ for (const child of ariaChildren) visit(ariaNode, child, parentElementVisible);
696
+ ariaNode.children.push(getCSSContent(element, "::after") || "");
697
+ if (treatAsBlock) ariaNode.children.push(treatAsBlock);
698
+ if (ariaNode.children.length === 1 && ariaNode.name === ariaNode.children[0]) ariaNode.children = [];
699
+ if (ariaNode.role === "link" && element.hasAttribute("href")) ariaNode.props["url"] = element.getAttribute("href");
700
+ if (ariaNode.role === "textbox" && element.hasAttribute("placeholder") && element.getAttribute("placeholder") !== ariaNode.name) ariaNode.props["placeholder"] = element.getAttribute("placeholder");
701
+ }
702
+
703
+ beginAriaCaches();
704
+ try { visit(snapshot.root, rootElement, true); }
705
+ finally { endAriaCaches(); }
706
+ normalizeStringChildren(snapshot.root);
707
+ normalizeGenericRoles(snapshot.root);
708
+ return snapshot;
709
+ }
710
+
711
+ function computeAriaRef(ariaNode, options) {
712
+ if (options.refs === "none") return;
713
+ if (options.refs === "interactable" && (!ariaNode.box.visible || !ariaNode.receivesPointerEvents)) return;
714
+ let ariaRef = ariaNode.element._ariaRef;
715
+ if (!ariaRef || ariaRef.role !== ariaNode.role || ariaRef.name !== ariaNode.name) {
716
+ ariaRef = { role: ariaNode.role, name: ariaNode.name, ref: (options.refPrefix || "") + "e" + (++lastRef) };
717
+ ariaNode.element._ariaRef = ariaRef;
718
+ }
719
+ ariaNode.ref = ariaRef.ref;
720
+ }
721
+
722
+ function toAriaNode(element, options) {
723
+ const active = element.ownerDocument.activeElement === element;
724
+ if (element.nodeName === "IFRAME") {
725
+ const ariaNode = { role: "iframe", name: "", children: [], props: {}, element, box: computeBox(element), receivesPointerEvents: true, active };
726
+ computeAriaRef(ariaNode, options);
727
+ return ariaNode;
728
+ }
729
+ const defaultRole = options.includeGenericRole ? "generic" : null;
730
+ const role = getAriaRole(element) || defaultRole;
731
+ if (!role || role === "presentation" || role === "none") return null;
732
+ const name = normalizeWhiteSpace(getElementAccessibleName(element, false) || "");
733
+ const receivesPointerEventsValue = receivesPointerEvents(element);
734
+ const box = computeBox(element);
735
+ if (role === "generic" && box.inline && element.childNodes.length === 1 && element.childNodes[0].nodeType === Node.TEXT_NODE) return null;
736
+ const result = { role, name, children: [], props: {}, element, box, receivesPointerEvents: receivesPointerEventsValue, active };
737
+ computeAriaRef(result, options);
738
+ if (kAriaCheckedRoles.includes(role)) result.checked = getAriaChecked(element);
739
+ if (kAriaDisabledRoles.includes(role)) result.disabled = getAriaDisabled(element);
740
+ if (kAriaExpandedRoles.includes(role)) result.expanded = getAriaExpanded(element);
741
+ if (kAriaLevelRoles.includes(role)) result.level = getAriaLevel(element);
742
+ if (kAriaPressedRoles.includes(role)) result.pressed = getAriaPressed(element);
743
+ if (kAriaSelectedRoles.includes(role)) result.selected = getAriaSelected(element);
744
+ if (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement) {
745
+ if (element.type !== "checkbox" && element.type !== "radio" && element.type !== "file") result.children = [element.value];
746
+ }
747
+ return result;
748
+ }
749
+
750
+ function normalizeGenericRoles(node) {
751
+ const normalizeChildren = (node) => {
752
+ const result = [];
753
+ for (const child of node.children || []) {
754
+ if (typeof child === "string") { result.push(child); continue; }
755
+ const normalized = normalizeChildren(child);
756
+ result.push(...normalized);
757
+ }
758
+ const removeSelf = node.role === "generic" && !node.name && result.length <= 1 && result.every(c => typeof c !== "string" && !!c.ref);
759
+ if (removeSelf) return result;
760
+ node.children = result;
761
+ return [node];
762
+ };
763
+ normalizeChildren(node);
764
+ }
765
+
766
+ function normalizeStringChildren(rootA11yNode) {
767
+ const flushChildren = (buffer, normalizedChildren) => {
768
+ if (!buffer.length) return;
769
+ const text = normalizeWhiteSpace(buffer.join(""));
770
+ if (text) normalizedChildren.push(text);
771
+ buffer.length = 0;
772
+ };
773
+ const visit = (ariaNode) => {
774
+ const normalizedChildren = [];
775
+ const buffer = [];
776
+ for (const child of ariaNode.children || []) {
777
+ if (typeof child === "string") { buffer.push(child); }
778
+ else { flushChildren(buffer, normalizedChildren); visit(child); normalizedChildren.push(child); }
779
+ }
780
+ flushChildren(buffer, normalizedChildren);
781
+ ariaNode.children = normalizedChildren.length ? normalizedChildren : [];
782
+ if (ariaNode.children.length === 1 && ariaNode.children[0] === ariaNode.name) ariaNode.children = [];
783
+ };
784
+ visit(rootA11yNode);
785
+ }
786
+
787
+ function hasPointerCursor(ariaNode) { return ariaNode.box.cursor === "pointer"; }
788
+
789
+ function renderAriaTree(ariaSnapshot) {
790
+ const options = { visibility: "ariaOrVisible", refs: "interactable", refPrefix: "", includeGenericRole: true, renderActive: true, renderCursorPointer: true };
791
+ const lines = [];
792
+ let nodesToRender = ariaSnapshot.root.role === "fragment" ? ariaSnapshot.root.children : [ariaSnapshot.root];
793
+
794
+ const visitText = (text, indent) => {
795
+ const escaped = yamlEscapeValueIfNeeded(text);
796
+ if (escaped) lines.push(indent + "- text: " + escaped);
797
+ };
798
+
799
+ const createKey = (ariaNode, renderCursorPointer) => {
800
+ let key = ariaNode.role;
801
+ if (ariaNode.name && ariaNode.name.length <= 900) {
802
+ const name = ariaNode.name;
803
+ if (name) {
804
+ const stringifiedName = name.startsWith("/") && name.endsWith("/") ? name : JSON.stringify(name);
805
+ key += " " + stringifiedName;
806
+ }
807
+ }
808
+ if (ariaNode.checked === "mixed") key += " [checked=mixed]";
809
+ if (ariaNode.checked === true) key += " [checked]";
810
+ if (ariaNode.disabled) key += " [disabled]";
811
+ if (ariaNode.expanded) key += " [expanded]";
812
+ if (ariaNode.active && options.renderActive) key += " [active]";
813
+ if (ariaNode.level) key += " [level=" + ariaNode.level + "]";
814
+ if (ariaNode.pressed === "mixed") key += " [pressed=mixed]";
815
+ if (ariaNode.pressed === true) key += " [pressed]";
816
+ if (ariaNode.selected === true) key += " [selected]";
817
+ if (ariaNode.ref) {
818
+ key += " [ref=" + ariaNode.ref + "]";
819
+ if (renderCursorPointer && hasPointerCursor(ariaNode)) key += " [cursor=pointer]";
820
+ }
821
+ return key;
822
+ };
823
+
824
+ const getSingleInlinedTextChild = (ariaNode) => {
825
+ return ariaNode?.children.length === 1 && typeof ariaNode.children[0] === "string" && !Object.keys(ariaNode.props).length ? ariaNode.children[0] : undefined;
826
+ };
827
+
828
+ const visit = (ariaNode, indent, renderCursorPointer) => {
829
+ const escapedKey = indent + "- " + yamlEscapeKeyIfNeeded(createKey(ariaNode, renderCursorPointer));
830
+ const singleInlinedTextChild = getSingleInlinedTextChild(ariaNode);
831
+ if (!ariaNode.children.length && !Object.keys(ariaNode.props).length) {
832
+ lines.push(escapedKey);
833
+ } else if (singleInlinedTextChild !== undefined) {
834
+ lines.push(escapedKey + ": " + yamlEscapeValueIfNeeded(singleInlinedTextChild));
835
+ } else {
836
+ lines.push(escapedKey + ":");
837
+ for (const [name, value] of Object.entries(ariaNode.props)) lines.push(indent + " - /" + name + ": " + yamlEscapeValueIfNeeded(value));
838
+ const childIndent = indent + " ";
839
+ const inCursorPointer = !!ariaNode.ref && renderCursorPointer && hasPointerCursor(ariaNode);
840
+ for (const child of ariaNode.children) {
841
+ if (typeof child === "string") visitText(child, childIndent);
842
+ else visit(child, childIndent, renderCursorPointer && !inCursorPointer);
843
+ }
844
+ }
845
+ };
846
+
847
+ for (const nodeToRender of nodesToRender) {
848
+ if (typeof nodeToRender === "string") visitText(nodeToRender, "");
849
+ else visit(nodeToRender, "", !!options.renderCursorPointer);
850
+ }
851
+ return lines.join("\\n");
852
+ }
853
+
854
+ function getAISnapshot() {
855
+ const snapshot = generateAriaTree(document.body);
856
+ const refsObject = {};
857
+ for (const [ref, element] of snapshot.elements) refsObject[ref] = element;
858
+ window.__devBrowserRefs = refsObject;
859
+ return renderAriaTree(snapshot);
860
+ }
861
+
862
+ function selectSnapshotRef(ref) {
863
+ const refs = window.__devBrowserRefs;
864
+ if (!refs) throw new Error("No snapshot refs found. Call getAISnapshot first.");
865
+ const element = refs[ref];
866
+ if (!element) throw new Error('Ref "' + ref + '" not found. Available refs: ' + Object.keys(refs).join(", "));
867
+ return element;
868
+ }
869
+ `;
870
+ }
871
+
872
+ /**
873
+ * Clear the cached script (useful for development/testing)
874
+ */
875
+ export function clearSnapshotScriptCache(): void {
876
+ cachedScript = null;
877
+ }