howone 0.1.29 → 0.1.31

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 (243) hide show
  1. package/package.json +1 -1
  2. package/templates/vite/.howone/skills/hallmark/SKILL.md +48 -42
  3. package/templates/vite/.howone/skills/hallmark/references/anti-patterns.md +12 -6
  4. package/templates/vite/.howone/skills/hallmark/references/assets.md +7 -0
  5. package/templates/vite/.howone/skills/hallmark/references/component-cookbook.md +19 -10
  6. package/templates/vite/.howone/skills/hallmark/references/components/f2-sticky-scroll-stack.md +1 -1
  7. package/templates/vite/.howone/skills/hallmark/references/components/ft6-letter-close.md +1 -1
  8. package/templates/vite/.howone/skills/hallmark/references/components/h7-demo-video-clipped-by-viewport-edge.md +1 -1
  9. package/templates/vite/.howone/skills/hallmark/references/components/h9-custom-illustration-centerpiece.md +1 -1
  10. package/templates/vite/.howone/skills/hallmark/references/components/n10-floating-on-scroll-morph.md +1 -1
  11. package/templates/vite/.howone/skills/hallmark/references/components/n11-mega-menu.md +40 -0
  12. package/templates/vite/.howone/skills/hallmark/references/components/n12-banner-retract.md +34 -0
  13. package/templates/vite/.howone/skills/hallmark/references/components/n13-inline-cmdk-pill.md +39 -0
  14. package/templates/vite/.howone/skills/hallmark/references/components/n1b-saas-three-section.md +35 -0
  15. package/templates/vite/.howone/skills/hallmark/references/components/n9-edge-aligned-minimal.md +1 -1
  16. package/templates/vite/.howone/skills/hallmark/references/components/s3-sticky-pinned.md +2 -2
  17. package/templates/vite/.howone/skills/hallmark/references/copy.md +8 -8
  18. package/templates/vite/.howone/skills/hallmark/references/custom-craft.md +2 -2
  19. package/templates/vite/.howone/skills/hallmark/references/custom-theme.md +50 -12
  20. package/templates/vite/.howone/skills/hallmark/references/export-formats.md +1 -1
  21. package/templates/vite/.howone/skills/hallmark/references/genres/atmospheric.md +11 -7
  22. package/templates/vite/.howone/skills/hallmark/references/genres/editorial.md +6 -4
  23. package/templates/vite/.howone/skills/hallmark/references/genres/modern-minimal.md +10 -6
  24. package/templates/vite/.howone/skills/hallmark/references/genres/playful.md +15 -10
  25. package/templates/vite/.howone/skills/hallmark/references/hero-enrichment.md +13 -12
  26. package/templates/vite/.howone/skills/hallmark/references/interaction-and-states.md +2 -1
  27. package/templates/vite/.howone/skills/hallmark/references/layout-and-space.md +4 -3
  28. package/templates/vite/.howone/skills/hallmark/references/macrostructures/04-stat-led.md +3 -1
  29. package/templates/vite/.howone/skills/hallmark/references/macrostructures/12-letter.md +1 -1
  30. package/templates/vite/.howone/skills/hallmark/references/macrostructures.md +1 -1
  31. package/templates/vite/.howone/skills/hallmark/references/microinteractions.md +1 -3
  32. package/templates/vite/.howone/skills/hallmark/references/preview-examples.md +12 -12
  33. package/templates/vite/.howone/skills/hallmark/references/responsive.md +8 -8
  34. package/templates/vite/.howone/skills/hallmark/references/slop-test.md +72 -85
  35. package/templates/vite/.howone/skills/hallmark/references/structure.md +9 -13
  36. package/templates/vite/.howone/skills/hallmark/references/study.md +40 -17
  37. package/templates/vite/.howone/skills/hallmark/references/themes/carnival.md +301 -0
  38. package/templates/vite/.howone/skills/hallmark/references/themes/cobalt.md +146 -0
  39. package/templates/vite/.howone/skills/hallmark/references/themes/hum.md +403 -0
  40. package/templates/vite/.howone/skills/hallmark/references/themes/lumen.md +478 -0
  41. package/templates/vite/.howone/skills/hallmark/references/typography.md +3 -3
  42. package/templates/vite/.howone/skills/hallmark/references/verbs/redesign.md +1 -1
  43. package/templates/vite/.howone/skills/howone/01-architect/01-app-generation.md +132 -176
  44. package/templates/vite/.howone/skills/howone/{02-database → 02-entity-schema}/01-schema-design.md +3 -1
  45. package/templates/vite/.howone/skills/howone/{04-ai → 03-ai-capabilities}/01-ai-capability-architecture.md +6 -5
  46. package/templates/vite/.howone/skills/howone/{04-ai/04-service-capability-catalog.md → 03-ai-capabilities/03-service-capability-catalog.md} +15 -11
  47. package/templates/vite/.howone/skills/howone/{03-sdk → 04-app-sdk}/01-client-setup.md +6 -4
  48. package/templates/vite/.howone/skills/howone/{03-sdk → 04-app-sdk}/07-ai-action-calls.md +3 -3
  49. package/templates/vite/.howone/skills/howone/{04-ai/03-ai-sdk-handoff.md → 04-app-sdk/08-ai-manifest-handoff.md} +2 -2
  50. package/templates/vite/.howone/skills/howone/SKILL.md +98 -131
  51. package/templates/vite/.howone/skills/howone/agents/openai.yaml +3 -3
  52. package/templates/vite/AGENTS.md +2 -2
  53. package/templates/vite/.howone/skills/hallmark/LICENSE +0 -21
  54. package/templates/vite/.howone/skills/hallmark/README.md +0 -147
  55. package/templates/vite/.howone/skills/hallmark/ROADMAP.md +0 -201
  56. package/templates/vite/.howone/skills/hallmark/docs/recipes.md +0 -186
  57. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-anya.jpg +0 -0
  58. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-bananastudio.jpg +0 -0
  59. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-hyperlane.jpg +0 -0
  60. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-najm.jpg +0 -0
  61. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-slow-pour.jpg +0 -0
  62. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-soroe.jpg +0 -0
  63. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-tally.jpg +0 -0
  64. package/templates/vite/.howone/skills/hallmark/docs/screenshots/hero-wayfare.jpg +0 -0
  65. package/templates/vite/.howone/skills/hallmark/docs/study-examples.md +0 -176
  66. package/templates/vite/.howone/skills/hallmark/docs/talk-slides.md +0 -364
  67. package/templates/vite/.howone/skills/hallmark/package.json +0 -36
  68. package/templates/vite/.howone/skills/hallmark/site/OG-hallmark.png +0 -0
  69. package/templates/vite/.howone/skills/hallmark/site/_tests/01-tide-podcast/brief.md +0 -71
  70. package/templates/vite/.howone/skills/hallmark/site/_tests/01-tide-podcast/index.html +0 -64
  71. package/templates/vite/.howone/skills/hallmark/site/_tests/01-tide-podcast/style.css +0 -240
  72. package/templates/vite/.howone/skills/hallmark/site/_tests/02-streampipe-cli/brief.md +0 -65
  73. package/templates/vite/.howone/skills/hallmark/site/_tests/02-streampipe-cli/index.html +0 -105
  74. package/templates/vite/.howone/skills/hallmark/site/_tests/02-streampipe-cli/style.css +0 -250
  75. package/templates/vite/.howone/skills/hallmark/site/_tests/03-maple-bakery/brief.md +0 -64
  76. package/templates/vite/.howone/skills/hallmark/site/_tests/03-maple-bakery/index.html +0 -131
  77. package/templates/vite/.howone/skills/hallmark/site/_tests/03-maple-bakery/style.css +0 -240
  78. package/templates/vite/.howone/skills/hallmark/site/_tests/04-meridian-manifesto/brief.md +0 -67
  79. package/templates/vite/.howone/skills/hallmark/site/_tests/04-meridian-manifesto/index.html +0 -86
  80. package/templates/vite/.howone/skills/hallmark/site/_tests/04-meridian-manifesto/style.css +0 -262
  81. package/templates/vite/.howone/skills/hallmark/site/_tests/05-tracejam-saas/brief.md +0 -63
  82. package/templates/vite/.howone/skills/hallmark/site/_tests/05-tracejam-saas/index.html +0 -167
  83. package/templates/vite/.howone/skills/hallmark/site/_tests/05-tracejam-saas/style.css +0 -457
  84. package/templates/vite/.howone/skills/hallmark/site/_tests/06-anya-portfolio/brief.md +0 -65
  85. package/templates/vite/.howone/skills/hallmark/site/_tests/06-anya-portfolio/index.html +0 -159
  86. package/templates/vite/.howone/skills/hallmark/site/_tests/06-anya-portfolio/style.css +0 -288
  87. package/templates/vite/.howone/skills/hallmark/site/_tests/07-foundry-compliance/brief.md +0 -64
  88. package/templates/vite/.howone/skills/hallmark/site/_tests/07-foundry-compliance/index.html +0 -146
  89. package/templates/vite/.howone/skills/hallmark/site/_tests/07-foundry-compliance/style.css +0 -484
  90. package/templates/vite/.howone/skills/hallmark/site/_tests/08-cohort-courses/brief.md +0 -64
  91. package/templates/vite/.howone/skills/hallmark/site/_tests/08-cohort-courses/index.html +0 -116
  92. package/templates/vite/.howone/skills/hallmark/site/_tests/08-cohort-courses/style.css +0 -354
  93. package/templates/vite/.howone/skills/hallmark/site/_tests/09-slow-pour/index.html +0 -638
  94. package/templates/vite/.howone/skills/hallmark/site/_tests/10-owl-hours/index.html +0 -515
  95. package/templates/vite/.howone/skills/hallmark/site/_tests/11-soroe-ceramics/index.html +0 -515
  96. package/templates/vite/.howone/skills/hallmark/site/_tests/12-loafer/index.html +0 -608
  97. package/templates/vite/.howone/skills/hallmark/site/_tests/13-alma/index.html +0 -587
  98. package/templates/vite/.howone/skills/hallmark/site/_tests/README.md +0 -157
  99. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/BananaStudio-loop.mp4 +0 -0
  100. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/BananaStudio-still.jpg +0 -0
  101. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/Hyperlane-example.mp4 +0 -0
  102. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/Hyperlane-still.jpg +0 -0
  103. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/Najm-loop.mp4 +0 -0
  104. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/Najm-still.jpg +0 -0
  105. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/Podcast-loop.mp4 +0 -0
  106. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/SaaS-loop.mp4 +0 -0
  107. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/SaaS-still.jpg +0 -0
  108. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/Soroe-loop.mp4 +0 -0
  109. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/Soroe-still.jpg +0 -0
  110. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/after-quiet-hour.png +0 -0
  111. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/anya-loop.mp4 +0 -0
  112. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/anya-still.jpg +0 -0
  113. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/audit-example.png +0 -0
  114. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/before-quiet-hour.png +0 -0
  115. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/example-redesign-uractivation.png +0 -0
  116. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/slow-pour-loop.mp4 +0 -0
  117. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/slow-pour-still.jpg +0 -0
  118. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/study-example.png +0 -0
  119. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/uractivation-after-loop.mp4 +0 -0
  120. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/wayfare-loop.mp4 +0 -0
  121. package/templates/vite/.howone/skills/hallmark/site/_tests/_thumbs/wayfare-still.jpg +0 -0
  122. package/templates/vite/.howone/skills/hallmark/site/_tests/custom/01-coffeebox/index.html +0 -77
  123. package/templates/vite/.howone/skills/hallmark/site/_tests/custom/01-coffeebox/style.css +0 -238
  124. package/templates/vite/.howone/skills/hallmark/site/_tests/custom/02-loop/index.html +0 -110
  125. package/templates/vite/.howone/skills/hallmark/site/_tests/custom/02-loop/style.css +0 -326
  126. package/templates/vite/.howone/skills/hallmark/site/_tests/custom/03-mossroot/index.html +0 -134
  127. package/templates/vite/.howone/skills/hallmark/site/_tests/custom/03-mossroot/style.css +0 -262
  128. package/templates/vite/.howone/skills/hallmark/site/_tests/custom/README.md +0 -30
  129. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/README.md +0 -17
  130. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/audit/audit-report.md +0 -56
  131. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/audit/input.html +0 -160
  132. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/audit/notes.md +0 -29
  133. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/redesign/input.html +0 -63
  134. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/redesign/notes.md +0 -72
  135. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/redesign/output.html +0 -374
  136. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/study/diagnosis.md +0 -52
  137. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/study/input-description.md +0 -29
  138. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/study/notes.md +0 -61
  139. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/study/output.css +0 -193
  140. package/templates/vite/.howone/skills/hallmark/site/_tests/verbs/study/output.html +0 -66
  141. package/templates/vite/.howone/skills/hallmark/site/css/base.css +0 -194
  142. package/templates/vite/.howone/skills/hallmark/site/css/components.css +0 -4886
  143. package/templates/vite/.howone/skills/hallmark/site/css/sections.css +0 -2072
  144. package/templates/vite/.howone/skills/hallmark/site/css/tokens.css +0 -1129
  145. package/templates/vite/.howone/skills/hallmark/site/examples/bananastudio/index.html +0 -475
  146. package/templates/vite/.howone/skills/hallmark/site/examples/bananastudio/styles.css +0 -1584
  147. package/templates/vite/.howone/skills/hallmark/site/examples/bananastudio/tokens.css +0 -96
  148. package/templates/vite/.howone/skills/hallmark/site/examples/hyperlane/index.html +0 -344
  149. package/templates/vite/.howone/skills/hallmark/site/examples/hyperlane/script.js +0 -103
  150. package/templates/vite/.howone/skills/hallmark/site/examples/hyperlane/styles.css +0 -1103
  151. package/templates/vite/.howone/skills/hallmark/site/examples/hyperlane/tokens.css +0 -83
  152. package/templates/vite/.howone/skills/hallmark/site/examples/najm/index.html +0 -368
  153. package/templates/vite/.howone/skills/hallmark/site/examples/najm/script.js +0 -133
  154. package/templates/vite/.howone/skills/hallmark/site/examples/najm/styles.css +0 -1062
  155. package/templates/vite/.howone/skills/hallmark/site/examples/najm/tokens.css +0 -97
  156. package/templates/vite/.howone/skills/hallmark/site/examples/tally/app.js +0 -84
  157. package/templates/vite/.howone/skills/hallmark/site/examples/tally/index.html +0 -446
  158. package/templates/vite/.howone/skills/hallmark/site/examples/tally/styles.css +0 -1087
  159. package/templates/vite/.howone/skills/hallmark/site/examples/tally/tokens.css +0 -101
  160. package/templates/vite/.howone/skills/hallmark/site/examples/wayfare/index.html +0 -359
  161. package/templates/vite/.howone/skills/hallmark/site/examples/wayfare/style.css +0 -1168
  162. package/templates/vite/.howone/skills/hallmark/site/examples/wayfare/tokens.css +0 -81
  163. package/templates/vite/.howone/skills/hallmark/site/favicon-dark.svg +0 -5
  164. package/templates/vite/.howone/skills/hallmark/site/favicon-light.svg +0 -5
  165. package/templates/vite/.howone/skills/hallmark/site/index.html +0 -1043
  166. package/templates/vite/.howone/skills/hallmark/site/js/main.js +0 -1175
  167. package/templates/vite/.howone/skills/hallmark/vercel.json +0 -6
  168. package/templates/vite/.howone/skills/impeccable/SKILL.md +0 -168
  169. package/templates/vite/.howone/skills/impeccable/agents/impeccable-asset-producer.md +0 -101
  170. package/templates/vite/.howone/skills/impeccable/reference/adapt.md +0 -190
  171. package/templates/vite/.howone/skills/impeccable/reference/animate.md +0 -175
  172. package/templates/vite/.howone/skills/impeccable/reference/audit.md +0 -133
  173. package/templates/vite/.howone/skills/impeccable/reference/bolder.md +0 -113
  174. package/templates/vite/.howone/skills/impeccable/reference/brand.md +0 -118
  175. package/templates/vite/.howone/skills/impeccable/reference/clarify.md +0 -174
  176. package/templates/vite/.howone/skills/impeccable/reference/codex.md +0 -105
  177. package/templates/vite/.howone/skills/impeccable/reference/cognitive-load.md +0 -106
  178. package/templates/vite/.howone/skills/impeccable/reference/color-and-contrast.md +0 -105
  179. package/templates/vite/.howone/skills/impeccable/reference/colorize.md +0 -154
  180. package/templates/vite/.howone/skills/impeccable/reference/craft.md +0 -123
  181. package/templates/vite/.howone/skills/impeccable/reference/critique.md +0 -273
  182. package/templates/vite/.howone/skills/impeccable/reference/delight.md +0 -302
  183. package/templates/vite/.howone/skills/impeccable/reference/distill.md +0 -111
  184. package/templates/vite/.howone/skills/impeccable/reference/document.md +0 -427
  185. package/templates/vite/.howone/skills/impeccable/reference/extract.md +0 -69
  186. package/templates/vite/.howone/skills/impeccable/reference/harden.md +0 -347
  187. package/templates/vite/.howone/skills/impeccable/reference/heuristics-scoring.md +0 -234
  188. package/templates/vite/.howone/skills/impeccable/reference/interaction-design.md +0 -195
  189. package/templates/vite/.howone/skills/impeccable/reference/layout.md +0 -141
  190. package/templates/vite/.howone/skills/impeccable/reference/live.md +0 -622
  191. package/templates/vite/.howone/skills/impeccable/reference/motion-design.md +0 -109
  192. package/templates/vite/.howone/skills/impeccable/reference/onboard.md +0 -234
  193. package/templates/vite/.howone/skills/impeccable/reference/optimize.md +0 -258
  194. package/templates/vite/.howone/skills/impeccable/reference/overdrive.md +0 -130
  195. package/templates/vite/.howone/skills/impeccable/reference/personas.md +0 -179
  196. package/templates/vite/.howone/skills/impeccable/reference/polish.md +0 -242
  197. package/templates/vite/.howone/skills/impeccable/reference/product.md +0 -62
  198. package/templates/vite/.howone/skills/impeccable/reference/quieter.md +0 -99
  199. package/templates/vite/.howone/skills/impeccable/reference/responsive-design.md +0 -114
  200. package/templates/vite/.howone/skills/impeccable/reference/shape.md +0 -165
  201. package/templates/vite/.howone/skills/impeccable/reference/spatial-design.md +0 -100
  202. package/templates/vite/.howone/skills/impeccable/reference/teach.md +0 -156
  203. package/templates/vite/.howone/skills/impeccable/reference/typeset.md +0 -124
  204. package/templates/vite/.howone/skills/impeccable/reference/typography.md +0 -159
  205. package/templates/vite/.howone/skills/impeccable/reference/ux-writing.md +0 -107
  206. package/templates/vite/.howone/skills/impeccable/scripts/cleanup-deprecated.mjs +0 -284
  207. package/templates/vite/.howone/skills/impeccable/scripts/command-metadata.json +0 -94
  208. package/templates/vite/.howone/skills/impeccable/scripts/critique-storage.mjs +0 -242
  209. package/templates/vite/.howone/skills/impeccable/scripts/design-parser.mjs +0 -820
  210. package/templates/vite/.howone/skills/impeccable/scripts/detect-csp.mjs +0 -198
  211. package/templates/vite/.howone/skills/impeccable/scripts/detect.mjs +0 -21
  212. package/templates/vite/.howone/skills/impeccable/scripts/impeccable-paths.mjs +0 -110
  213. package/templates/vite/.howone/skills/impeccable/scripts/is-generated.mjs +0 -69
  214. package/templates/vite/.howone/skills/impeccable/scripts/live-accept.mjs +0 -595
  215. package/templates/vite/.howone/skills/impeccable/scripts/live-browser-session.js +0 -123
  216. package/templates/vite/.howone/skills/impeccable/scripts/live-browser.js +0 -4860
  217. package/templates/vite/.howone/skills/impeccable/scripts/live-complete.mjs +0 -75
  218. package/templates/vite/.howone/skills/impeccable/scripts/live-completion.mjs +0 -18
  219. package/templates/vite/.howone/skills/impeccable/scripts/live-inject.mjs +0 -446
  220. package/templates/vite/.howone/skills/impeccable/scripts/live-poll.mjs +0 -200
  221. package/templates/vite/.howone/skills/impeccable/scripts/live-resume.mjs +0 -48
  222. package/templates/vite/.howone/skills/impeccable/scripts/live-server.mjs +0 -838
  223. package/templates/vite/.howone/skills/impeccable/scripts/live-session-store.mjs +0 -254
  224. package/templates/vite/.howone/skills/impeccable/scripts/live-status.mjs +0 -47
  225. package/templates/vite/.howone/skills/impeccable/scripts/live-wrap.mjs +0 -632
  226. package/templates/vite/.howone/skills/impeccable/scripts/live.mjs +0 -247
  227. package/templates/vite/.howone/skills/impeccable/scripts/load-context.mjs +0 -141
  228. package/templates/vite/.howone/skills/impeccable/scripts/modern-screenshot.umd.js +0 -14
  229. package/templates/vite/.howone/skills/impeccable/scripts/pin.mjs +0 -214
  230. /package/templates/vite/.howone/skills/howone/{02-database → 02-entity-schema}/02-schema-operations.md +0 -0
  231. /package/templates/vite/.howone/skills/howone/{02-database → 02-entity-schema}/03-data-access-patterns.md +0 -0
  232. /package/templates/vite/.howone/skills/howone/{02-database → 02-entity-schema}/04-query-dsl-and-responses.md +0 -0
  233. /package/templates/vite/.howone/skills/howone/{02-database → 02-entity-schema}/05-ai-persistence-patterns.md +0 -0
  234. /package/templates/vite/.howone/skills/howone/{04-ai → 03-ai-capabilities}/02-workflow-contract-rules.md +0 -0
  235. /package/templates/vite/.howone/skills/howone/{04-ai/05-workflow-operations.md → 03-ai-capabilities/04-workflow-operations.md} +0 -0
  236. /package/templates/vite/.howone/skills/howone/{04-ai/06-ai-feature-playbooks.md → 03-ai-capabilities/05-ai-feature-playbooks.md} +0 -0
  237. /package/templates/vite/.howone/skills/howone/{03-sdk → 04-app-sdk}/02-entity-operations.md +0 -0
  238. /package/templates/vite/.howone/skills/howone/{03-sdk → 04-app-sdk}/03-auth.md +0 -0
  239. /package/templates/vite/.howone/skills/howone/{03-sdk → 04-app-sdk}/04-react-integration.md +0 -0
  240. /package/templates/vite/.howone/skills/howone/{03-sdk → 04-app-sdk}/05-file-upload.md +0 -0
  241. /package/templates/vite/.howone/skills/howone/{03-sdk → 04-app-sdk}/06-raw-http.md +0 -0
  242. /package/templates/vite/.howone/skills/howone/{03-sdk/08-extension-boundaries.md → 04-app-sdk/09-extension-boundaries.md} +0 -0
  243. /package/templates/vite/.howone/skills/howone/{03-sdk/09-workflow-execute-sse.md → 04-app-sdk/10-workflow-execute-sse.md} +0 -0
@@ -1,820 +0,0 @@
1
- // Parse a DESIGN.md (Stitch-spec format) into a structured JSON model that
2
- // the live-mode design-system panel can render. Deterministic, dependency-free.
3
- //
4
- // Two-layer: YAML frontmatter (machine-readable tokens) + markdown body
5
- // (prose with six canonical H2 sections). When frontmatter is present, it's
6
- // exposed on `model.frontmatter` alongside the prose-scraped sections;
7
- // consumers can prefer frontmatter values and fall back to prose.
8
-
9
- const CANONICAL_SECTIONS = [
10
- 'Overview',
11
- 'Colors',
12
- 'Typography',
13
- 'Elevation',
14
- 'Components',
15
- "Do's and Don'ts",
16
- ];
17
-
18
- // ---------- Frontmatter (Stitch YAML subset) ----------
19
-
20
- function parseFrontmatter(md) {
21
- const lines = md.split(/\r?\n/);
22
- if (lines[0]?.trim() !== '---') return { frontmatter: null, body: md };
23
-
24
- let end = -1;
25
- for (let i = 1; i < lines.length; i++) {
26
- if (lines[i].trim() === '---') { end = i; break; }
27
- }
28
- if (end === -1) return { frontmatter: null, body: md };
29
-
30
- const yaml = lines.slice(1, end).join('\n');
31
- const body = lines.slice(end + 1).join('\n');
32
- try {
33
- return { frontmatter: parseYamlSubset(yaml), body };
34
- } catch {
35
- return { frontmatter: null, body: md };
36
- }
37
- }
38
-
39
- // Minimal YAML reader for the Stitch frontmatter subset: scalar maps with
40
- // one level of nested objects (typography roles, components). Indent-based,
41
- // 2-space convention. No arrays, no anchors, no multi-line scalars — Stitch's
42
- // schema doesn't need them and accepting them would require a real YAML
43
- // dependency we don't want to vendor.
44
- function parseYamlSubset(yaml) {
45
- const lines = yaml.split(/\r?\n/);
46
- const root = {};
47
- const stack = [{ indent: -1, obj: root }];
48
-
49
- for (const raw of lines) {
50
- // Skip blanks and line-only comments. Don't strip inline comments:
51
- // unquoted hex values start with `#` and can't be safely distinguished
52
- // from a comment after whitespace.
53
- if (!raw.trim() || /^\s*#/.test(raw)) continue;
54
-
55
- const indent = raw.match(/^\s*/)[0].length;
56
- const content = raw.slice(indent);
57
-
58
- const colonIdx = findTopLevelColon(content);
59
- if (colonIdx === -1) continue;
60
-
61
- while (stack.length > 1 && stack[stack.length - 1].indent >= indent) {
62
- stack.pop();
63
- }
64
-
65
- const key = content.slice(0, colonIdx).trim();
66
- const rest = content.slice(colonIdx + 1).trim();
67
- const parent = stack[stack.length - 1].obj;
68
-
69
- if (rest === '') {
70
- const obj = {};
71
- parent[key] = obj;
72
- stack.push({ indent, obj });
73
- } else {
74
- parent[key] = parseScalar(rest);
75
- }
76
- }
77
-
78
- return root;
79
- }
80
-
81
- function findTopLevelColon(s) {
82
- let inQuote = null;
83
- for (let i = 0; i < s.length; i++) {
84
- const ch = s[i];
85
- if (inQuote) {
86
- if (ch === inQuote && s[i - 1] !== '\\') inQuote = null;
87
- } else if (ch === '"' || ch === "'") {
88
- inQuote = ch;
89
- } else if (ch === ':') {
90
- return i;
91
- }
92
- }
93
- return -1;
94
- }
95
-
96
- function parseScalar(raw) {
97
- const s = raw.trim();
98
- if ((s.startsWith('"') && s.endsWith('"')) || (s.startsWith("'") && s.endsWith("'"))) {
99
- return s.slice(1, -1);
100
- }
101
- if (s === 'true') return true;
102
- if (s === 'false') return false;
103
- if (s === 'null' || s === '~') return null;
104
- if (/^-?\d+$/.test(s)) return Number(s);
105
- if (/^-?\d*\.\d+$/.test(s)) return Number(s);
106
- return s;
107
- }
108
-
109
- const HEX_RE = /#[0-9a-fA-F]{3,8}\b/g;
110
- const OKLCH_RE = /oklch\([^)]+\)/gi;
111
- const RGBA_RE = /rgba?\([^)]+\)/gi;
112
- const BOX_SHADOW_RE = /(?:box-shadow:\s*)?((?:-?\d[\w\d\s\-.,/()#%]*)+)/;
113
- const NAMED_RULE_RE = /\*\*(The [^*]+?Rule)\.\*\*\s*(.+)/;
114
-
115
- // ---------- Section splitting ----------
116
-
117
- function splitSections(md) {
118
- const lines = md.split(/\r?\n/);
119
- let title = null;
120
- const sections = {};
121
- let current = null;
122
-
123
- for (const raw of lines) {
124
- const line = raw.trimEnd();
125
-
126
- if (!title && line.startsWith('# ') && !line.startsWith('## ')) {
127
- title = line.replace(/^#\s+/, '').trim();
128
- continue;
129
- }
130
-
131
- const h2 = line.match(/^##\s+(?:\d+\.\s*)?([^:\n]+?)(?::\s*(.+))?$/);
132
- if (h2) {
133
- const rawName = normalizeApostrophes(h2[1].trim());
134
- const subtitle = h2[2] ? h2[2].trim() : null;
135
- const canonical = matchCanonicalSection(rawName);
136
- if (canonical) {
137
- current = { name: canonical, subtitle, lines: [] };
138
- sections[canonical] = current;
139
- continue;
140
- }
141
- // non-canonical H2 — ignore but stop feeding into current
142
- current = null;
143
- continue;
144
- }
145
-
146
- if (current) current.lines.push(raw);
147
- }
148
-
149
- return { title, sections };
150
- }
151
-
152
- function normalizeApostrophes(s) {
153
- return s.replace(/[\u2018\u2019]/g, "'");
154
- }
155
-
156
- function matchCanonicalSection(name) {
157
- const normalized = normalizeApostrophes(name).toLowerCase();
158
- // Exact match first
159
- for (const c of CANONICAL_SECTIONS) {
160
- if (normalizeApostrophes(c).toLowerCase() === normalized) return c;
161
- }
162
- // Keyword-contained match: "Overview & Creative North Star" -> "Overview",
163
- // "Elevation & Depth" -> "Elevation", etc.
164
- for (const c of CANONICAL_SECTIONS) {
165
- const key = normalizeApostrophes(c).toLowerCase();
166
- const pattern = new RegExp(`\\b${key.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}\\b`);
167
- if (pattern.test(normalized)) return c;
168
- }
169
- return null;
170
- }
171
-
172
- // ---------- Subsection splitting (inside a canonical section) ----------
173
-
174
- function splitSubsections(lines) {
175
- const subs = [];
176
- let current = { name: null, lines: [] };
177
- subs.push(current);
178
-
179
- for (const raw of lines) {
180
- const h3 = raw.match(/^###\s+(.+?)\s*$/);
181
- if (h3) {
182
- current = { name: h3[1].trim(), lines: [] };
183
- subs.push(current);
184
- continue;
185
- }
186
- current.lines.push(raw);
187
- }
188
-
189
- return subs;
190
- }
191
-
192
- // ---------- Generic helpers ----------
193
-
194
- function collectParagraphs(lines) {
195
- const paragraphs = [];
196
- let buf = [];
197
- const flush = () => {
198
- if (buf.length) {
199
- paragraphs.push(buf.join(' ').trim());
200
- buf = [];
201
- }
202
- };
203
- for (const raw of lines) {
204
- const trimmed = raw.trim();
205
- if (trimmed === '') { flush(); continue; }
206
- // Horizontal rules (---, ***) and headings/bullets end a paragraph.
207
- if (/^(?:-{3,}|\*{3,}|_{3,})$/.test(trimmed)) { flush(); continue; }
208
- if (raw.startsWith('#') || raw.match(/^[-*]\s/)) { flush(); continue; }
209
- buf.push(trimmed);
210
- }
211
- flush();
212
- return paragraphs.filter(Boolean);
213
- }
214
-
215
- function collectBullets(lines) {
216
- const bullets = [];
217
- let current = null;
218
- for (const raw of lines) {
219
- const m = raw.match(/^\s*[-*]\s+(.+)$/);
220
- if (m) {
221
- if (current) bullets.push(current);
222
- current = m[1];
223
- continue;
224
- }
225
- // continuation of a bullet (indented line)
226
- if (current && raw.match(/^\s{2,}\S/)) {
227
- current += ' ' + raw.trim();
228
- continue;
229
- }
230
- // blank line ends a bullet
231
- if (raw.trim() === '' && current) {
232
- bullets.push(current);
233
- current = null;
234
- }
235
- }
236
- if (current) bullets.push(current);
237
- return bullets;
238
- }
239
-
240
- function stripBold(s) {
241
- return s.replace(/\*\*(.+?)\*\*/g, '$1');
242
- }
243
-
244
- function extractNamedRules(lines) {
245
- const rules = [];
246
- const seen = new Set();
247
-
248
- // Style A (Impeccable): "**The X Rule.** body body body" — can span lines.
249
- const joined = lines.join('\n');
250
- const inlineStart = /\*\*(The [^*]+?Rule)\.\*\*/g;
251
- const inlineMatches = [];
252
- let m;
253
- while ((m = inlineStart.exec(joined)) !== null) {
254
- inlineMatches.push({ name: m[1], start: m.index, end: inlineStart.lastIndex });
255
- }
256
- for (let i = 0; i < inlineMatches.length; i++) {
257
- const mm = inlineMatches[i];
258
- const bodyEnd = i + 1 < inlineMatches.length ? inlineMatches[i + 1].start : joined.length;
259
- const body = joined
260
- .slice(mm.end, bodyEnd)
261
- .replace(/\n##[^\n]*$/s, '')
262
- .replace(/\n###[^\n]*$/s, '')
263
- .trim();
264
- const name = stripBold(mm.name).trim();
265
- seen.add(name.toLowerCase());
266
- rules.push({ name, body: stripBold(body) });
267
- }
268
-
269
- // Style B (Stitch): `### The "X" Rule` or `### The X Fallback`, body is the
270
- // bullets/paragraphs until the next heading. Accept Rule / Fallback / Principle.
271
- for (let i = 0; i < lines.length; i++) {
272
- const h3 = lines[i].match(/^###\s+(.+?)\s*$/);
273
- if (!h3) continue;
274
- const headerName = stripBold(h3[1]).replace(/["“”]/g, '').trim();
275
- if (!/^The\b.*\b(Rule|Fallback|Principle)\b/i.test(headerName)) continue;
276
- if (seen.has(headerName.toLowerCase())) continue;
277
-
278
- const bodyLines = [];
279
- for (let j = i + 1; j < lines.length; j++) {
280
- if (/^##\s|^###\s/.test(lines[j])) break;
281
- bodyLines.push(lines[j]);
282
- }
283
- const body = stripBold(bodyLines.join('\n').replace(/\n+/g, ' ')).trim();
284
- if (body) {
285
- seen.add(headerName.toLowerCase());
286
- rules.push({ name: headerName, body });
287
- }
288
- }
289
-
290
- // Style C (Stitch bullet form): "* **The Layering Principle:** body"
291
- // Colon/period lives inside the bold, so match "**...**" then inspect.
292
- for (const b of collectBullets(lines)) {
293
- const mm = b.match(/^\*\*([^*]+?)\*\*\s*(.+)$/);
294
- if (!mm) continue;
295
- const nameRaw = mm[1].replace(/[.:]\s*$/, '').replace(/["“”]/g, '').trim();
296
- if (!/^The\b.+\b(Rule|Fallback|Principle)$/i.test(nameRaw)) continue;
297
- if (seen.has(nameRaw.toLowerCase())) continue;
298
- seen.add(nameRaw.toLowerCase());
299
- rules.push({ name: nameRaw, body: stripBold(mm[2]).trim() });
300
- }
301
-
302
- return rules;
303
- }
304
-
305
- // ---------- Per-section extractors ----------
306
-
307
- function extractOverview(section) {
308
- if (!section) return null;
309
- const text = section.lines.join('\n');
310
- const northStar = text.match(/\*\*Creative North Star:\s*"([^"]+)"\*\*/);
311
- const keyChars = [];
312
- const keyCharMatch = text.match(/\*\*Key Characteristics:\*\*\s*\n([\s\S]+?)(?:\n##|\n###|$)/);
313
- if (keyCharMatch) {
314
- for (const line of keyCharMatch[1].split('\n')) {
315
- const m = line.match(/^\s*[-*]\s+(.+)$/);
316
- if (m) keyChars.push(stripBold(m[1].trim()));
317
- }
318
- }
319
-
320
- // Philosophy paragraphs: everything that isn't a rule header or key-char block
321
- const paragraphs = collectParagraphs(section.lines).filter(
322
- (p) =>
323
- !p.startsWith('**Creative North Star') &&
324
- !p.startsWith('**Key Characteristics')
325
- );
326
-
327
- return {
328
- subtitle: section.subtitle,
329
- creativeNorthStar: northStar ? northStar[1] : null,
330
- philosophy: paragraphs,
331
- keyCharacteristics: keyChars,
332
- };
333
- }
334
-
335
- function extractColors(section) {
336
- if (!section) return null;
337
- const subs = splitSubsections(section.lines);
338
-
339
- const description = collectParagraphs(subs[0].lines).join(' ');
340
- const groups = [];
341
- const ROLE_KEYWORDS = /^(primary|secondary|tertiary|neutral|accent)\b/i;
342
-
343
- for (const sub of subs.slice(1)) {
344
- if (!sub.name || /Named Rules?/i.test(sub.name) || /^The\s/i.test(sub.name)) continue;
345
-
346
- const bullets = collectBullets(sub.lines);
347
- const parsed = bullets.map((b) => parseColorBullet(b)).filter(Boolean);
348
- if (parsed.length === 0) continue;
349
-
350
- // If every bullet starts with a role keyword (Primary/Secondary/...), promote
351
- // each bullet to its own group. Otherwise keep the subsection as the group.
352
- const allRoleBullets =
353
- parsed.length > 0 && parsed.every((p) => p.name && ROLE_KEYWORDS.test(p.name));
354
-
355
- if (allRoleBullets) {
356
- for (const p of parsed) {
357
- groups.push({ role: p.name, colors: [p] });
358
- }
359
- } else {
360
- groups.push({ role: sub.name, colors: parsed });
361
- }
362
- }
363
-
364
- // If the Colors section has no subsections at all (unlikely), fall back to
365
- // scanning the whole section as a flat bullet list.
366
- if (groups.length === 0) {
367
- const flat = collectBullets(section.lines)
368
- .map((b) => parseColorBullet(b))
369
- .filter(Boolean);
370
- if (flat.length) {
371
- for (const p of flat) {
372
- if (p.name && ROLE_KEYWORDS.test(p.name)) {
373
- groups.push({ role: p.name, colors: [p] });
374
- } else {
375
- const fallback = groups.find((g) => g.role === 'Palette');
376
- if (fallback) fallback.colors.push(p);
377
- else groups.push({ role: 'Palette', colors: [p] });
378
- }
379
- }
380
- }
381
- }
382
-
383
- return {
384
- subtitle: section.subtitle,
385
- description: description || null,
386
- groups,
387
- rules: extractNamedRules(section.lines),
388
- };
389
- }
390
-
391
- function parseColorBullet(bullet) {
392
- const text = bullet.trim();
393
-
394
- // Case 1 (Impeccable): **Name** (value-with-maybe-nested-parens): description
395
- const bold = text.match(/^\*\*(.+?)\*\*\s*(.*)$/);
396
- if (bold && bold[2].startsWith('(')) {
397
- const value = extractParenGroup(bold[2]);
398
- if (value !== null) {
399
- const after = bold[2].slice(value.length + 2).trimStart();
400
- if (after.startsWith(':')) {
401
- return buildColor(bold[1], value, after.slice(1).trim());
402
- }
403
- }
404
- }
405
-
406
- // Case 2 (Stitch): **Name (values):** description — value embedded in bold.
407
- const stitch = text.match(/^\*\*([^*]+?)\s*\(([^)]+)\):\*\*\s*(.*)$/);
408
- if (stitch) {
409
- return buildColor(stitch[1].trim(), stitch[2], stitch[3]);
410
- }
411
-
412
- // Case 3: bullet without bold, just hex/oklch inside.
413
- const values = collectColorValues(text);
414
- if (values.length) {
415
- return buildColor(null, values.join(' to '), text);
416
- }
417
- return null;
418
- }
419
-
420
- function extractParenGroup(s) {
421
- if (s[0] !== '(') return null;
422
- let depth = 0;
423
- for (let i = 0; i < s.length; i++) {
424
- if (s[i] === '(') depth++;
425
- else if (s[i] === ')') {
426
- depth--;
427
- if (depth === 0) return s.slice(1, i);
428
- }
429
- }
430
- return null;
431
- }
432
-
433
- function buildColor(name, rawValue, description) {
434
- const values = collectColorValues(rawValue);
435
- const primary = values[0] ?? rawValue.trim();
436
- return {
437
- name: name ? stripBold(name).trim() : null,
438
- value: primary,
439
- valueRange: values.length > 1 ? values : null,
440
- format: detectFormat(primary),
441
- description: stripBold(description || '').trim() || null,
442
- };
443
- }
444
-
445
- function collectColorValues(s) {
446
- const out = [];
447
- s.replace(HEX_RE, (v) => {
448
- out.push(v);
449
- return v;
450
- });
451
- s.replace(OKLCH_RE, (v) => {
452
- out.push(v);
453
- return v;
454
- });
455
- return out;
456
- }
457
-
458
- function detectFormat(v) {
459
- if (!v) return 'unknown';
460
- if (v.startsWith('#')) return 'hex';
461
- if (/^oklch/i.test(v)) return 'oklch';
462
- if (/^rgb/i.test(v)) return 'rgb';
463
- return 'unknown';
464
- }
465
-
466
- function scanInlineColors(lines) {
467
- const out = [];
468
- for (const line of lines) {
469
- if (!/^\s*[-*]\s/.test(line)) continue;
470
- const trimmed = line.replace(/^\s*[-*]\s+/, '');
471
- const color = parseColorBullet(trimmed);
472
- if (color) out.push(color);
473
- }
474
- return out;
475
- }
476
-
477
- function parseStitchInlineGroups(lines) {
478
- // Stitch writes: `* **Primary (`#00478d` to `#005eb8`):** Use for "..."`
479
- // Each bullet IS its own role. Group them under the spoken role name.
480
- const out = [];
481
- for (const line of lines) {
482
- if (!/^\s*[-*]\s/.test(line)) continue;
483
- const trimmed = line.replace(/^\s*[-*]\s+/, '').trim();
484
- const m = trimmed.match(
485
- /^\*\*([A-Z][a-zA-Z]+)\s*\(([^)]+)\):\*\*\s*(.*)$/
486
- );
487
- if (m) {
488
- const role = m[1];
489
- const color = buildColor(role, m[2], m[3]);
490
- out.push({ role, colors: [color] });
491
- }
492
- }
493
- return out;
494
- }
495
-
496
- function extractTypography(section) {
497
- if (!section) return null;
498
- const text = section.lines.join('\n');
499
-
500
- const fonts = {};
501
- // Pattern A: **Display Font:** Family (with fallback)
502
- const fontLineRe = /\*\*([\w\s/]+?)Font:\*\*\s*([^\n(]+?)(?:\s*\(with\s+([^)]+)\))?\s*$/gm;
503
- let fm;
504
- while ((fm = fontLineRe.exec(text)) !== null) {
505
- const rawRole = fm[1].trim().toLowerCase().replace(/\s+/g, '-');
506
- const role = normalizeFontRole(rawRole) || 'display';
507
- fonts[role] = {
508
- family: fm[2].trim(),
509
- fallback: fm[3] ? fm[3].trim() : null,
510
- };
511
- }
512
-
513
- // Pattern B (Stitch): * **Display & Headlines (Noto Serif):** description
514
- if (Object.keys(fonts).length === 0) {
515
- const stitchRe = /\*\*([\w\s&/]+?)\s*\(([^)]+)\):\*\*\s*(.+)/g;
516
- let sm;
517
- while ((sm = stitchRe.exec(text)) !== null) {
518
- const rawRole = sm[1]
519
- .trim()
520
- .toLowerCase()
521
- .replace(/\s*&\s*/g, '-')
522
- .replace(/\s+/g, '-');
523
- const role = normalizeFontRole(rawRole) || rawRole;
524
- fonts[role] = { family: sm[2].trim(), fallback: null, purpose: sm[3].trim() };
525
- }
526
- }
527
-
528
- // Character paragraph — either a **Character:** label, or fall back to the
529
- // first free paragraph under the section header (Stitch style).
530
- const characterMatch = text.match(/\*\*Character:\*\*\s*([^\n]+(?:\n[^\n]+)*?)(?=\n\n|\n###|\n##|$)/);
531
- let character = characterMatch ? characterMatch[1].replace(/\n/g, ' ').trim() : null;
532
- if (!character) {
533
- const paragraphs = collectParagraphs(section.lines).filter(
534
- (p) => !/^\*\*[\w\s/&]+Font/i.test(p) && !/^\*\*[\w\s/&]+\([^)]+\)/.test(p)
535
- );
536
- if (paragraphs.length) character = paragraphs[0];
537
- }
538
-
539
- // Hierarchy bullets under ### Hierarchy
540
- const subs = splitSubsections(section.lines);
541
- let hierarchy = [];
542
- const hierSub = subs.find((s) => s.name && /hierarch/i.test(s.name));
543
- if (hierSub) {
544
- const bullets = collectBullets(hierSub.lines);
545
- hierarchy = bullets.map(parseTypeBullet).filter(Boolean);
546
- }
547
-
548
- return {
549
- subtitle: section.subtitle,
550
- fonts,
551
- character,
552
- hierarchy,
553
- rules: extractNamedRules(section.lines),
554
- };
555
- }
556
-
557
- function normalizeFontRole(raw) {
558
- // Canonical roles the panel cares about: display, body, label, mono.
559
- // Stitch often writes compound roles like "display-&-headlines" or "ui-&-body"
560
- // — collapse them to the first canonical role present.
561
- const tokens = raw.split(/[-/&\s]+/).filter(Boolean);
562
- const priority = ['display', 'headline', 'body', 'ui', 'label', 'mono'];
563
- const canonical = { headline: 'display', ui: 'body' };
564
- for (const p of priority) {
565
- if (tokens.includes(p)) return canonical[p] || p;
566
- }
567
- return null;
568
- }
569
-
570
- function parseTypeBullet(bullet) {
571
- // - **Display** (family, weight 300, italic, clamp(...), line-height 1): purpose
572
- const m = bullet.match(/^\*\*(.+?)\*\*\s*\(([^)]+)\):\s*(.*)$/);
573
- if (!m) return null;
574
- const name = m[1].trim();
575
- const specs = m[2].split(',').map((s) => s.trim());
576
- return {
577
- name,
578
- specs,
579
- purpose: stripBold(m[3] || '').trim() || null,
580
- };
581
- }
582
-
583
- function extractElevation(section) {
584
- if (!section) return null;
585
- const subs = splitSubsections(section.lines);
586
-
587
- const description = collectParagraphs(subs[0].lines).join(' ') || null;
588
-
589
- const shadows = [];
590
- const seen = new Set();
591
- const dedupe = (entry) => {
592
- const key = (entry.name || '') + '::' + entry.value;
593
- if (seen.has(key)) return;
594
- seen.add(key);
595
- shadows.push(entry);
596
- };
597
-
598
- for (const b of collectBullets(section.lines)) {
599
- const parsed = parseShadowBullet(b);
600
- if (parsed) dedupe(parsed);
601
- }
602
-
603
- // Fallback: extract shadows written inline in prose. Stitch style is
604
- // "...use an extra-diffused shadow: `box-shadow: 0 12px 40px rgba(...)`."
605
- for (const p of collectParagraphs(section.lines)) {
606
- for (const inline of extractInlineShadows(p)) dedupe(inline);
607
- }
608
- for (const b of collectBullets(section.lines)) {
609
- for (const inline of extractInlineShadows(b)) dedupe(inline);
610
- }
611
-
612
- return {
613
- subtitle: section.subtitle,
614
- description,
615
- shadows,
616
- rules: extractNamedRules(section.lines),
617
- };
618
- }
619
-
620
- function extractInlineShadows(text) {
621
- // Find `box-shadow: ...` anywhere in prose and capture the value. Work on the
622
- // raw string so it handles both backtick-fenced and unfenced variants.
623
- const out = [];
624
- const re = /box-shadow\s*:\s*([^`;\n]+)/gi;
625
- let m;
626
- while ((m = re.exec(text)) !== null) {
627
- const value = m[1].replace(/[`.)]+$/, '').trim();
628
- if (!value) continue;
629
- // Name heuristic: the noun immediately before the shadow phrase.
630
- // e.g. "an extra-diffused shadow: ..." -> "extra-diffused shadow"
631
- const before = text.slice(0, m.index);
632
- const nameMatch = before.match(/\b([A-Za-z][A-Za-z\- ]{2,40})\s+shadow\b[^A-Za-z0-9]*$/i);
633
- let name = null;
634
- if (nameMatch) {
635
- const stripped = nameMatch[1]
636
- .replace(/^(?:use|using|apply|applying|is|are|looks? like)\s+/i, '')
637
- .replace(/^(?:a|an|the)\s+/i, '')
638
- .trim();
639
- if (stripped) {
640
- name =
641
- stripped.charAt(0).toUpperCase() + stripped.slice(1) + ' shadow';
642
- }
643
- }
644
- out.push({
645
- name,
646
- value,
647
- purpose: null,
648
- });
649
- }
650
- return out;
651
- }
652
-
653
- function parseShadowBullet(bullet) {
654
- // - **Name** (`box-shadow: value`): purpose
655
- // - **Name** (`value`): purpose
656
- // Only accept if the paren content looks like a shadow value (contains px,
657
- // rem, rgba, or box-shadow). This filters out `**Rule Name:**` bullets.
658
- const m = bullet.match(/^\*\*(.+?)\*\*\s*\(`?([^`]+?)`?\):\s*(.*)$/);
659
- if (!m) return null;
660
- const rawValue = m[2].replace(/^box-shadow:\s*/i, '').trim();
661
- const looksLikeShadow =
662
- /box-shadow|rgba?\(|\bpx\b|\brem\b|^-?\d+\s/i.test(rawValue) &&
663
- /\d/.test(rawValue);
664
- if (!looksLikeShadow) return null;
665
- const name = stripBold(m[1]).trim();
666
- return {
667
- name,
668
- value: rawValue,
669
- purpose: stripBold(m[3] || '').trim() || null,
670
- };
671
- }
672
-
673
- function extractComponents(section) {
674
- if (!section) return null;
675
- const subs = splitSubsections(section.lines);
676
- const components = [];
677
-
678
- for (const sub of subs.slice(1)) {
679
- if (!sub.name) continue;
680
-
681
- const bullets = collectBullets(sub.lines);
682
- const paragraphs = collectParagraphs(sub.lines);
683
-
684
- const variants = [];
685
- const properties = {};
686
-
687
- for (const b of bullets) {
688
- // - **Key:** value
689
- const m = b.match(/^\*\*(.+?):?\*\*:?\s*(.+)$/);
690
- if (m) {
691
- const key = stripBold(m[1]).trim();
692
- const value = stripBold(m[2]).trim();
693
- // Heuristic: "Primary", "Secondary", "Hover", "Focus" etc are variants;
694
- // "Shape", "Background", "Padding" are properties.
695
- if (/^(primary|secondary|tertiary|ghost|hover|focus|active|disabled|default|error|selected|unselected|state)$/i.test(key.split(/[\s/]/)[0])) {
696
- variants.push({ name: key, description: value });
697
- } else {
698
- properties[key.toLowerCase()] = value;
699
- }
700
- }
701
- }
702
-
703
- components.push({
704
- name: sub.name,
705
- description: paragraphs.join(' ') || null,
706
- properties,
707
- variants,
708
- });
709
- }
710
-
711
- return {
712
- subtitle: section.subtitle,
713
- components,
714
- };
715
- }
716
-
717
- function extractDosDonts(section) {
718
- if (!section) return null;
719
- const subs = splitSubsections(section.lines);
720
- const dos = [];
721
- const donts = [];
722
-
723
- for (const sub of subs.slice(1)) {
724
- if (!sub.name) continue;
725
- const subName = normalizeApostrophes(sub.name);
726
- const bullets = collectBullets(sub.lines).map((b) => stripBold(b).trim());
727
- if (/^do'?t?:?$/i.test(subName) || /^do:?$/i.test(subName)) {
728
- dos.push(...bullets);
729
- } else if (/^don'?t:?$/i.test(subName)) {
730
- donts.push(...bullets);
731
- }
732
- }
733
-
734
- // Classify by bullet prefix as a backup (catches loose bullets outside H3 wrappers)
735
- for (const b of collectBullets(section.lines)) {
736
- const stripped = normalizeApostrophes(stripBold(b).trim());
737
- if (/^don'?t\b/i.test(stripped)) {
738
- if (!donts.some((d) => normalizeApostrophes(d) === stripped)) donts.push(stripped);
739
- } else if (/^do\b/i.test(stripped)) {
740
- if (!dos.some((d) => normalizeApostrophes(d) === stripped)) dos.push(stripped);
741
- }
742
- }
743
-
744
- return { dos, donts };
745
- }
746
-
747
- // ---------- Coverage assessment ----------
748
-
749
- function assessCoverage(model) {
750
- const report = {};
751
-
752
- report.overview = model.overview
753
- ? {
754
- northStar: Boolean(model.overview.creativeNorthStar),
755
- philosophy: model.overview.philosophy.length > 0,
756
- keyCharacteristics: model.overview.keyCharacteristics.length,
757
- }
758
- : 'missing';
759
-
760
- report.colors = model.colors
761
- ? {
762
- groups: model.colors.groups.length,
763
- totalColors: model.colors.groups.reduce((n, g) => n + g.colors.length, 0),
764
- rules: model.colors.rules.length,
765
- }
766
- : 'missing';
767
-
768
- report.typography = model.typography
769
- ? {
770
- fonts: Object.keys(model.typography.fonts).length,
771
- hierarchyEntries: model.typography.hierarchy.length,
772
- character: Boolean(model.typography.character),
773
- rules: model.typography.rules.length,
774
- }
775
- : 'missing';
776
-
777
- report.elevation = model.elevation
778
- ? {
779
- shadows: model.elevation.shadows.length,
780
- rules: model.elevation.rules.length,
781
- description: Boolean(model.elevation.description),
782
- }
783
- : 'missing';
784
-
785
- report.components = model.components
786
- ? {
787
- count: model.components.components.length,
788
- variantTotal: model.components.components.reduce((n, c) => n + c.variants.length, 0),
789
- }
790
- : 'missing';
791
-
792
- report.dosDonts = model.dosDonts
793
- ? {
794
- dos: model.dosDonts.dos.length,
795
- donts: model.dosDonts.donts.length,
796
- }
797
- : 'missing';
798
-
799
- return report;
800
- }
801
-
802
- // ---------- Main ----------
803
-
804
- export function parseDesignMd(md) {
805
- const { frontmatter, body } = parseFrontmatter(md);
806
- const { title, sections } = splitSections(body);
807
- return {
808
- schemaVersion: 2,
809
- title,
810
- frontmatter,
811
- overview: extractOverview(sections['Overview']),
812
- colors: extractColors(sections['Colors']),
813
- typography: extractTypography(sections['Typography']),
814
- elevation: extractElevation(sections['Elevation']),
815
- components: extractComponents(sections['Components']),
816
- dosDonts: extractDosDonts(sections["Do's and Don'ts"]),
817
- };
818
- }
819
-
820
- export { assessCoverage };