@invisibleloop/pulse 0.1.21

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 (344) hide show
  1. package/.claude/commands/build-page.md +59 -0
  2. package/.claude/commands/new-doc-page.md +45 -0
  3. package/.claude/commands/verify.md +52 -0
  4. package/.claude/pulse-checklist.md +111 -0
  5. package/.claude/settings.local.json +102 -0
  6. package/.github/workflows/ci.yml +22 -0
  7. package/.github/workflows/publish.yml +41 -0
  8. package/.pulse/load-reports/home/1773432711417.json +22 -0
  9. package/CLAUDE.md +383 -0
  10. package/README.md +95 -0
  11. package/docs/.claude/pulse-checklist.md +111 -0
  12. package/docs/public/.pulse-ui-version +1 -0
  13. package/docs/public/dist/accessibility.boot-5DVTARJU.js +115 -0
  14. package/docs/public/dist/actions.boot-P66HKQEM.js +164 -0
  15. package/docs/public/dist/auth.boot-IMAJAUPH.js +140 -0
  16. package/docs/public/dist/caching.boot-DVR6KDE7.js +53 -0
  17. package/docs/public/dist/components--accordion.boot-3HVKMNWC.js +11 -0
  18. package/docs/public/dist/components--alert.boot-GCEXOZAC.js +6 -0
  19. package/docs/public/dist/components--app-badge.boot-DVT3GCHJ.js +6 -0
  20. package/docs/public/dist/components--avatar.boot-PSW24EVA.js +5 -0
  21. package/docs/public/dist/components--badge.boot-TYDY2RMK.js +7 -0
  22. package/docs/public/dist/components--banner.boot-EI5PZSZK.js +7 -0
  23. package/docs/public/dist/components--breadcrumbs.boot-SMA2E2GO.js +34 -0
  24. package/docs/public/dist/components--button.boot-J54BQM2E.js +23 -0
  25. package/docs/public/dist/components--card.boot-PZGNDIB6.js +138 -0
  26. package/docs/public/dist/components--carousel.boot-TP6LPFZZ.js +12 -0
  27. package/docs/public/dist/components--charts.boot-2EOYQWKL.js +108 -0
  28. package/docs/public/dist/components--checkbox.boot-DS5BSL6T.js +54 -0
  29. package/docs/public/dist/components--cluster.boot-HHVIBBJG.js +9 -0
  30. package/docs/public/dist/components--code-window.boot-2GR2DV33.js +20 -0
  31. package/docs/public/dist/components--container.boot-7LOOGK2K.js +5 -0
  32. package/docs/public/dist/components--cta.boot-FSNZ5YRT.js +11 -0
  33. package/docs/public/dist/components--divider.boot-3NI2C3QG.js +6 -0
  34. package/docs/public/dist/components--empty.boot-YX2UR3PV.js +7 -0
  35. package/docs/public/dist/components--feature.boot-MUD7NSUO.js +13 -0
  36. package/docs/public/dist/components--fieldset.boot-J7BYHMKF.js +19 -0
  37. package/docs/public/dist/components--fileupload.boot-NIKVTTPD.js +52 -0
  38. package/docs/public/dist/components--footer.boot-EYUK5FRG.js +14 -0
  39. package/docs/public/dist/components--grid.boot-URDQVDDR.js +59 -0
  40. package/docs/public/dist/components--heading.boot-BPQKU43E.js +44 -0
  41. package/docs/public/dist/components--hero.boot-4RAPRGAB.js +17 -0
  42. package/docs/public/dist/components--icons.boot-ZITNU5JP.js +68 -0
  43. package/docs/public/dist/components--image.boot-XEEGHQZF.js +19 -0
  44. package/docs/public/dist/components--input.boot-SGASZG5K.js +7 -0
  45. package/docs/public/dist/components--list.boot-W3XC5MHD.js +55 -0
  46. package/docs/public/dist/components--media.boot-5VFIETZO.js +13 -0
  47. package/docs/public/dist/components--modal.boot-RZUYXBN2.js +47 -0
  48. package/docs/public/dist/components--nav.boot-ODBOHU7O.js +33 -0
  49. package/docs/public/dist/components--pricing.boot-4AQ4ZVBY.js +21 -0
  50. package/docs/public/dist/components--progress.boot-GHAGYZOK.js +30 -0
  51. package/docs/public/dist/components--prose.boot-QANJL6JI.js +67 -0
  52. package/docs/public/dist/components--pullquote.boot-Q2WMNAZU.js +22 -0
  53. package/docs/public/dist/components--radio.boot-TJRDQ2OL.js +75 -0
  54. package/docs/public/dist/components--rating.boot-QBAN6DEL.js +38 -0
  55. package/docs/public/dist/components--search.boot-PXH5O5AG.js +17 -0
  56. package/docs/public/dist/components--section.boot-AQGIYHWW.js +12 -0
  57. package/docs/public/dist/components--segmented.boot-BEVTKEJO.js +33 -0
  58. package/docs/public/dist/components--select.boot-47X5RHOC.js +10 -0
  59. package/docs/public/dist/components--slider.boot-PSRRX7XL.js +47 -0
  60. package/docs/public/dist/components--spinner.boot-MZ5MO2OH.js +22 -0
  61. package/docs/public/dist/components--stack.boot-DI4NJXBF.js +9 -0
  62. package/docs/public/dist/components--stat.boot-QMFUWBQT.js +9 -0
  63. package/docs/public/dist/components--stepper.boot-34PP2NEV.js +22 -0
  64. package/docs/public/dist/components--table.boot-FCQGSFIQ.js +11 -0
  65. package/docs/public/dist/components--testimonial.boot-DWQPDKYG.js +11 -0
  66. package/docs/public/dist/components--textarea.boot-QVXLBOJ5.js +4 -0
  67. package/docs/public/dist/components--timeline.boot-26LN52P2.js +95 -0
  68. package/docs/public/dist/components--toggle.boot-IQQEI76S.js +29 -0
  69. package/docs/public/dist/components--tooltip.boot-LGHCO6NN.js +9 -0
  70. package/docs/public/dist/components.boot-SE6PQ4P7.js +103 -0
  71. package/docs/public/dist/config.boot-DTRRWUE6.js +126 -0
  72. package/docs/public/dist/constraints.boot-DUHDZBMC.js +71 -0
  73. package/docs/public/dist/deploy.boot-SLAD3NI2.js +163 -0
  74. package/docs/public/dist/docs-8e3d4b5c.css +1 -0
  75. package/docs/public/dist/extending.boot-UA3CN243.js +159 -0
  76. package/docs/public/dist/faq.boot-6EQAWLQR.js +43 -0
  77. package/docs/public/dist/getting-started.boot-TDKIFL5U.js +86 -0
  78. package/docs/public/dist/guard.boot-AUHAWTG4.js +80 -0
  79. package/docs/public/dist/home.boot-BVQXRH32.js +383 -0
  80. package/docs/public/dist/how-it-works.boot-LTWAKWKW.js +104 -0
  81. package/docs/public/dist/hydration.boot-JRM6IPJL.js +78 -0
  82. package/docs/public/dist/images.boot-M6ZVKTZS.js +80 -0
  83. package/docs/public/dist/manifest.json +94 -0
  84. package/docs/public/dist/meta.boot-7NXGPHR4.js +79 -0
  85. package/docs/public/dist/mutations.boot-F6F43UDX.js +79 -0
  86. package/docs/public/dist/navigation.boot-AOXWS3ZF.js +57 -0
  87. package/docs/public/dist/performance.boot-C3UPCOBK.js +98 -0
  88. package/docs/public/dist/persist.boot-WT32PQOQ.js +61 -0
  89. package/docs/public/dist/project-structure.boot-FB3LRVJ4.js +63 -0
  90. package/docs/public/dist/prompt-examples.boot-YKR4VDK4.js +31 -0
  91. package/docs/public/dist/pulse-ui-81a85c03.css +1 -0
  92. package/docs/public/dist/raw-responses.boot-M4KA5YXL.js +104 -0
  93. package/docs/public/dist/routing.boot-FNX5FDGH.js +70 -0
  94. package/docs/public/dist/runtime-B73WLANC.js +1 -0
  95. package/docs/public/dist/runtime-KO4BHUQ3.js +49 -0
  96. package/docs/public/dist/runtime-L2HNXIHW.js +59 -0
  97. package/docs/public/dist/runtime-QFURDKA2.js +5 -0
  98. package/docs/public/dist/runtime-UVPXO4IR.js +375 -0
  99. package/docs/public/dist/runtime-VMJA3Z4N.js +10 -0
  100. package/docs/public/dist/runtime-ZJ4FXT5O.js +11 -0
  101. package/docs/public/dist/server-api.boot-K7X3LCFB.js +219 -0
  102. package/docs/public/dist/server-data.boot-Y7HQYC4R.js +157 -0
  103. package/docs/public/dist/slash-commands.boot-V2UV7OW2.js +26 -0
  104. package/docs/public/dist/spec.boot-2WU7ZHCV.js +159 -0
  105. package/docs/public/dist/state.boot-B24GUE3R.js +73 -0
  106. package/docs/public/dist/store.boot-TLIB4XHH.js +150 -0
  107. package/docs/public/dist/streaming.boot-W2DZSMW4.js +80 -0
  108. package/docs/public/dist/stripe.boot-QN3C2GEL.js +164 -0
  109. package/docs/public/dist/supabase.boot-BG4XXLZE.js +303 -0
  110. package/docs/public/dist/testing.boot-6U4WKMTE.js +130 -0
  111. package/docs/public/dist/validation.boot-PQHYGW5B.js +100 -0
  112. package/docs/public/docs.css +2020 -0
  113. package/docs/public/menu.js +83 -0
  114. package/docs/public/pulse-ui.css +2739 -0
  115. package/docs/public/pulse-ui.js +236 -0
  116. package/docs/server.js +192 -0
  117. package/docs/src/lib/component-page.js +47 -0
  118. package/docs/src/lib/highlight.js +255 -0
  119. package/docs/src/lib/layout.js +131 -0
  120. package/docs/src/lib/metrics-store.js +6 -0
  121. package/docs/src/lib/nav.js +159 -0
  122. package/docs/src/lib/stats.js +81 -0
  123. package/docs/src/pages/accessibility.js +157 -0
  124. package/docs/src/pages/actions.js +191 -0
  125. package/docs/src/pages/auth.js +177 -0
  126. package/docs/src/pages/caching.js +95 -0
  127. package/docs/src/pages/components/accordion.js +48 -0
  128. package/docs/src/pages/components/alert.js +35 -0
  129. package/docs/src/pages/components/app-badge.js +41 -0
  130. package/docs/src/pages/components/avatar.js +35 -0
  131. package/docs/src/pages/components/badge.js +36 -0
  132. package/docs/src/pages/components/banner.js +45 -0
  133. package/docs/src/pages/components/breadcrumbs.js +94 -0
  134. package/docs/src/pages/components/button.js +84 -0
  135. package/docs/src/pages/components/card.js +225 -0
  136. package/docs/src/pages/components/carousel.js +72 -0
  137. package/docs/src/pages/components/charts.js +278 -0
  138. package/docs/src/pages/components/checkbox.js +129 -0
  139. package/docs/src/pages/components/cluster.js +47 -0
  140. package/docs/src/pages/components/code-window.js +57 -0
  141. package/docs/src/pages/components/container.js +40 -0
  142. package/docs/src/pages/components/cta.js +53 -0
  143. package/docs/src/pages/components/divider.js +37 -0
  144. package/docs/src/pages/components/empty.js +36 -0
  145. package/docs/src/pages/components/feature.js +60 -0
  146. package/docs/src/pages/components/fieldset.js +65 -0
  147. package/docs/src/pages/components/fileupload.js +127 -0
  148. package/docs/src/pages/components/footer.js +58 -0
  149. package/docs/src/pages/components/grid.js +165 -0
  150. package/docs/src/pages/components/heading.js +107 -0
  151. package/docs/src/pages/components/hero.js +65 -0
  152. package/docs/src/pages/components/icons.js +285 -0
  153. package/docs/src/pages/components/image.js +71 -0
  154. package/docs/src/pages/components/input.js +51 -0
  155. package/docs/src/pages/components/list.js +112 -0
  156. package/docs/src/pages/components/media.js +51 -0
  157. package/docs/src/pages/components/modal.js +111 -0
  158. package/docs/src/pages/components/nav.js +86 -0
  159. package/docs/src/pages/components/pricing.js +68 -0
  160. package/docs/src/pages/components/progress.js +102 -0
  161. package/docs/src/pages/components/prose.js +111 -0
  162. package/docs/src/pages/components/pullquote.js +71 -0
  163. package/docs/src/pages/components/radio.js +194 -0
  164. package/docs/src/pages/components/rating.js +106 -0
  165. package/docs/src/pages/components/search.js +61 -0
  166. package/docs/src/pages/components/section.js +59 -0
  167. package/docs/src/pages/components/segmented.js +121 -0
  168. package/docs/src/pages/components/select.js +45 -0
  169. package/docs/src/pages/components/slider.js +114 -0
  170. package/docs/src/pages/components/spinner.js +73 -0
  171. package/docs/src/pages/components/stack.js +48 -0
  172. package/docs/src/pages/components/stat.js +55 -0
  173. package/docs/src/pages/components/stepper.js +66 -0
  174. package/docs/src/pages/components/table.js +45 -0
  175. package/docs/src/pages/components/testimonial.js +49 -0
  176. package/docs/src/pages/components/textarea.js +31 -0
  177. package/docs/src/pages/components/timeline.js +227 -0
  178. package/docs/src/pages/components/toggle.js +84 -0
  179. package/docs/src/pages/components/tooltip.js +48 -0
  180. package/docs/src/pages/components.js +204 -0
  181. package/docs/src/pages/config.js +193 -0
  182. package/docs/src/pages/constraints.js +99 -0
  183. package/docs/src/pages/deploy.js +233 -0
  184. package/docs/src/pages/extending.js +198 -0
  185. package/docs/src/pages/faq.js +96 -0
  186. package/docs/src/pages/getting-started.js +106 -0
  187. package/docs/src/pages/guard.js +121 -0
  188. package/docs/src/pages/home.js +401 -0
  189. package/docs/src/pages/how-it-works.js +183 -0
  190. package/docs/src/pages/hydration.js +98 -0
  191. package/docs/src/pages/images.js +121 -0
  192. package/docs/src/pages/meta.js +120 -0
  193. package/docs/src/pages/mutations.js +106 -0
  194. package/docs/src/pages/navigation.js +85 -0
  195. package/docs/src/pages/performance.js +157 -0
  196. package/docs/src/pages/persist.js +88 -0
  197. package/docs/src/pages/project-structure.js +90 -0
  198. package/docs/src/pages/prompt-examples.js +186 -0
  199. package/docs/src/pages/raw-responses.js +124 -0
  200. package/docs/src/pages/routing.js +99 -0
  201. package/docs/src/pages/server-api.js +281 -0
  202. package/docs/src/pages/server-data.js +185 -0
  203. package/docs/src/pages/slash-commands.js +55 -0
  204. package/docs/src/pages/spec.js +207 -0
  205. package/docs/src/pages/state.js +101 -0
  206. package/docs/src/pages/store.js +181 -0
  207. package/docs/src/pages/streaming.js +108 -0
  208. package/docs/src/pages/stripe.js +193 -0
  209. package/docs/src/pages/supabase.js +323 -0
  210. package/docs/src/pages/testing.js +198 -0
  211. package/docs/src/pages/validation.js +138 -0
  212. package/examples/contact.js +166 -0
  213. package/examples/counter.js +94 -0
  214. package/examples/dev.server.js +91 -0
  215. package/examples/examples.test.js +394 -0
  216. package/examples/pricing.js +244 -0
  217. package/examples/products.js +191 -0
  218. package/examples/quiz.js +208 -0
  219. package/examples/shared.js +78 -0
  220. package/examples/todos.js +162 -0
  221. package/package.json +75 -0
  222. package/public/.pulse-ui-version +1 -0
  223. package/public/chippy-bird.css +246 -0
  224. package/public/examples/contact.css +119 -0
  225. package/public/examples/counter.css +79 -0
  226. package/public/examples/pricing.css +132 -0
  227. package/public/examples/products.css +100 -0
  228. package/public/examples/quiz.css +200 -0
  229. package/public/examples/todos.css +137 -0
  230. package/public/favicon.ico +0 -0
  231. package/public/log-dashboard.css +383 -0
  232. package/public/pulse-ui.css +2740 -0
  233. package/public/pulse-ui.js +236 -0
  234. package/public/pulse.css +149 -0
  235. package/scripts/build.js +411 -0
  236. package/src/agent/checklist.md +111 -0
  237. package/src/agent/coverage-check.js +66 -0
  238. package/src/agent/guide-components.md +274 -0
  239. package/src/agent/guide-examples.md +54 -0
  240. package/src/agent/guide-routing.md +36 -0
  241. package/src/agent/guide-server.md +258 -0
  242. package/src/agent/guide-spec.md +103 -0
  243. package/src/agent/guide-styles.md +191 -0
  244. package/src/agent/guide.md +979 -0
  245. package/src/agent/identity.md +106 -0
  246. package/src/agent/workflow.md +108 -0
  247. package/src/cli/cli.test.js +82 -0
  248. package/src/cli/dev.js +195 -0
  249. package/src/cli/discover.js +113 -0
  250. package/src/cli/index.js +361 -0
  251. package/src/cli/load-report.js +91 -0
  252. package/src/cli/load-runner.js +121 -0
  253. package/src/cli/report-server.js +723 -0
  254. package/src/cli/report.js +116 -0
  255. package/src/cli/scaffold.archive.js +1371 -0
  256. package/src/cli/scaffold.js +349 -0
  257. package/src/cli/start.js +74 -0
  258. package/src/html.js +19 -0
  259. package/src/mcp/server.js +884 -0
  260. package/src/mcp/validate-worker.js +110 -0
  261. package/src/runtime/image.js +74 -0
  262. package/src/runtime/image.test.js +111 -0
  263. package/src/runtime/index.js +621 -0
  264. package/src/runtime/navigate.js +146 -0
  265. package/src/runtime/runtime.test.js +773 -0
  266. package/src/runtime/ssr.js +464 -0
  267. package/src/runtime/ssr.test.js +421 -0
  268. package/src/runtime/store.js +92 -0
  269. package/src/runtime/toast.js +163 -0
  270. package/src/server/index.js +1386 -0
  271. package/src/server/server.test.js +1248 -0
  272. package/src/spec/schema.js +428 -0
  273. package/src/spec/schema.test.js +291 -0
  274. package/src/store/index.js +102 -0
  275. package/src/store/store.test.js +210 -0
  276. package/src/testing/html.js +283 -0
  277. package/src/testing/index.js +249 -0
  278. package/src/testing/testing.test.js +450 -0
  279. package/src/ui/accordion.js +28 -0
  280. package/src/ui/alert.js +43 -0
  281. package/src/ui/app-badge.js +48 -0
  282. package/src/ui/avatar.js +47 -0
  283. package/src/ui/badge.js +24 -0
  284. package/src/ui/banner.js +26 -0
  285. package/src/ui/breadcrumbs.js +38 -0
  286. package/src/ui/button.js +66 -0
  287. package/src/ui/card.js +34 -0
  288. package/src/ui/carousel.js +59 -0
  289. package/src/ui/charts.js +321 -0
  290. package/src/ui/checkbox.js +65 -0
  291. package/src/ui/cluster.js +44 -0
  292. package/src/ui/code-window.js +39 -0
  293. package/src/ui/container.js +24 -0
  294. package/src/ui/cta.js +37 -0
  295. package/src/ui/divider.js +29 -0
  296. package/src/ui/empty.js +33 -0
  297. package/src/ui/feature.js +33 -0
  298. package/src/ui/fieldset.js +37 -0
  299. package/src/ui/fileupload.js +89 -0
  300. package/src/ui/footer.js +38 -0
  301. package/src/ui/grid.js +36 -0
  302. package/src/ui/heading.js +45 -0
  303. package/src/ui/hero.js +37 -0
  304. package/src/ui/icons.js +161 -0
  305. package/src/ui/index.js +89 -0
  306. package/src/ui/input.js +74 -0
  307. package/src/ui/list.js +36 -0
  308. package/src/ui/media.js +44 -0
  309. package/src/ui/modal.js +80 -0
  310. package/src/ui/nav.js +61 -0
  311. package/src/ui/pricing.js +56 -0
  312. package/src/ui/progress.js +62 -0
  313. package/src/ui/prose.js +29 -0
  314. package/src/ui/pullquote.js +34 -0
  315. package/src/ui/radio.js +102 -0
  316. package/src/ui/rating.js +93 -0
  317. package/src/ui/search.js +77 -0
  318. package/src/ui/section.js +69 -0
  319. package/src/ui/segmented.js +50 -0
  320. package/src/ui/select.js +77 -0
  321. package/src/ui/slider.js +84 -0
  322. package/src/ui/spinner.js +34 -0
  323. package/src/ui/stack.js +36 -0
  324. package/src/ui/stat.js +52 -0
  325. package/src/ui/stepper.js +46 -0
  326. package/src/ui/switch.js +57 -0
  327. package/src/ui/table.js +45 -0
  328. package/src/ui/testimonial.js +48 -0
  329. package/src/ui/textarea.js +72 -0
  330. package/src/ui/timeline.js +72 -0
  331. package/src/ui/tooltip.js +28 -0
  332. package/src/ui/ui.test.js +1241 -0
  333. package/src/ui/uiimage.js +65 -0
  334. package/tsconfig.json +13 -0
  335. package/types/html.d.ts +17 -0
  336. package/types/image.d.ts +70 -0
  337. package/types/index.d.ts +7 -0
  338. package/types/navigate.d.ts +38 -0
  339. package/types/runtime.d.ts +63 -0
  340. package/types/schema.d.ts +243 -0
  341. package/types/server.d.ts +145 -0
  342. package/types/ssr.d.ts +110 -0
  343. package/types/testing.d.ts +154 -0
  344. package/types/ui.d.ts +704 -0
@@ -0,0 +1,114 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { table, callout } from '../../lib/layout.js'
4
+ import { slider } from '../../../../src/ui/index.js'
5
+
6
+ const { prev, next } = prevNext('/components/slider')
7
+
8
+ export default {
9
+ route: '/components/slider',
10
+ meta: {
11
+ title: 'Slider — Pulse Docs',
12
+ description: 'Styled range input slider component for Pulse UI.',
13
+ styles: ['/pulse-ui.css', '/docs.css'],
14
+ },
15
+ state: { brightness: 100 },
16
+ mutations: {
17
+ setBrightness: (state, e) => ({ brightness: Number(e.target.value) }),
18
+ },
19
+ view: (state) => renderComponentPage({
20
+ currentHref: '/components/slider',
21
+ prev,
22
+ next,
23
+ name: 'slider',
24
+ description: 'Styled range input with label, hint, and a CSS-driven fill gradient that tracks the thumb live as you drag.',
25
+ content: `
26
+
27
+ <h2 class="doc-h2" id="basic">Basic</h2>
28
+ ${demo(
29
+ slider({ name: 'volume', label: 'Volume', showValue: true }),
30
+ `slider({ name: 'volume', label: 'Volume', showValue: true })`,
31
+ { col: true }
32
+ )}
33
+
34
+ <h2 class="doc-h2" id="min-max-step">Min, max, step, and hint</h2>
35
+ ${demo(
36
+ slider({
37
+ name: 'brightness',
38
+ label: 'Brightness',
39
+ min: 0,
40
+ max: 200,
41
+ step: 10,
42
+ value: 100,
43
+ showValue: true,
44
+ hint: 'Adjust display brightness (0–200)',
45
+ }),
46
+ `slider({
47
+ name: 'brightness',
48
+ label: 'Brightness',
49
+ min: 0,
50
+ max: 200,
51
+ step: 10,
52
+ value: 100,
53
+ showValue: true,
54
+ hint: 'Adjust display brightness (0–200)',
55
+ })`,
56
+ { col: true }
57
+ )}
58
+
59
+ <h2 class="doc-h2" id="live-value">Live value in state</h2>
60
+ <p class="u-mb-4 u-text-sm u-color-muted">Use <code>data-event="change:mutationName"</code> to capture the value when the user releases the handle. The fill gradient always updates live — no mutation needed for that.</p>
61
+ ${demo(
62
+ `<div>
63
+ ${slider({ name: 'brightness', label: 'Brightness', value: state.brightness, min: 0, max: 200, step: 10, showValue: true, event: 'change:setBrightness' })}
64
+ </div>`,
65
+ `// state
66
+ { brightness: 100 }
67
+
68
+ // mutation
69
+ setBrightness: (state, e) => ({ brightness: Number(e.target.value) })
70
+
71
+ // view
72
+ slider({
73
+ name: 'brightness',
74
+ label: 'Brightness',
75
+ min: 0,
76
+ max: 200,
77
+ step: 10,
78
+ value: state.brightness,
79
+ showValue: true,
80
+ event: 'change:setBrightness',
81
+ })`,
82
+ { col: true }
83
+ )}
84
+
85
+ <h2 class="doc-h2" id="disabled">Disabled</h2>
86
+ ${demo(
87
+ slider({ name: 'opacity', label: 'Opacity', value: 30, disabled: true }),
88
+ `slider({ name: 'opacity', label: 'Opacity', value: 30, disabled: true })`,
89
+ { col: true }
90
+ )}
91
+
92
+ ${callout('note', 'The fill gradient updates live during drag automatically. To read the final value, either use <code>data-event="change:mutationName"</code> on the slider or submit it as part of a form — the value is available in FormData as a number string under <code>name</code>.')}
93
+ ${callout('warning', 'Do not use <code>data-event="input:mutationName"</code> on a slider. Pulse replaces <code>innerHTML</code> on every mutation, which interrupts the drag mid-gesture. Use <code>change</code> instead — it fires once when the user releases the handle.')}
94
+
95
+ ${table(
96
+ ['Prop', 'Type', 'Default', ''],
97
+ [
98
+ ['<code>name</code>', 'string', '—', 'Field name — submitted in FormData'],
99
+ ['<code>label</code>', 'string', '—', 'Visible label text'],
100
+ ['<code>min</code>', 'number', '0', 'Minimum value'],
101
+ ['<code>max</code>', 'number', '100', 'Maximum value'],
102
+ ['<code>step</code>', 'number', '1', 'Step increment'],
103
+ ['<code>value</code>', 'number', '50', 'Current value'],
104
+ ['<code>showValue</code>', 'boolean', 'false', 'Show current value live beside the label as you drag'],
105
+ ['<code>disabled</code>', 'boolean', 'false', ''],
106
+ ['<code>hint</code>', 'string', '—', 'Helper text below the slider'],
107
+ ['<code>event</code>', 'string', '—', '<code>data-event</code> binding, e.g. <code>change:mutationName</code>'],
108
+ ['<code>id</code>', 'string', '—', 'Override the generated <code>id</code>'],
109
+ ['<code>class</code>', 'string', '—', ''],
110
+ ]
111
+ )}
112
+ `,
113
+ }),
114
+ }
@@ -0,0 +1,73 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { table } from '../../lib/layout.js'
4
+ import { spinner, button, cluster } from '../../../../src/ui/index.js'
5
+
6
+ const { prev, next } = prevNext('/components/spinner')
7
+
8
+ export default {
9
+ route: '/components/spinner',
10
+ meta: {
11
+ title: 'Spinner — Pulse Docs',
12
+ description: 'Loading spinner component for Pulse UI.',
13
+ styles: ['/pulse-ui.css', '/docs.css'],
14
+ },
15
+ state: {},
16
+ view: () => renderComponentPage({
17
+ currentHref: '/components/spinner',
18
+ prev,
19
+ next,
20
+ name: 'spinner',
21
+ description: 'CSS-animated loading ring. No JavaScript required. Use to indicate background activity — inside buttons, next to labels, or as a full-area overlay.',
22
+ content: `
23
+
24
+ <h2 class="doc-h2" id="sizes">Sizes</h2>
25
+ ${demo(
26
+ cluster({ gap: 'lg', align: 'center', content:
27
+ spinner({ size: 'sm' }) +
28
+ spinner({ size: 'md' }) +
29
+ spinner({ size: 'lg' }),
30
+ }),
31
+ `spinner({ size: 'sm' })
32
+ spinner({ size: 'md' })
33
+ spinner({ size: 'lg' })`
34
+ )}
35
+
36
+ <h2 class="doc-h2" id="colors">Colours</h2>
37
+ ${demo(
38
+ cluster({ gap: 'lg', align: 'center', content:
39
+ spinner({ color: 'accent' }) +
40
+ spinner({ color: 'muted' }) +
41
+ `<span style="background:var(--ui-accent);padding:.75rem;border-radius:var(--ui-radius);display:inline-flex">` +
42
+ spinner({ color: 'white' }) +
43
+ `</span>`,
44
+ }),
45
+ `spinner({ color: 'accent' })
46
+ spinner({ color: 'muted' })
47
+ spinner({ color: 'white' })`
48
+ )}
49
+
50
+ <h2 class="doc-h2" id="in-button">Inside a button</h2>
51
+ <p>Pair with a <code>button()</code> to show loading state. Pass the spinner as the button's <code>icon</code> prop.</p>
52
+ ${demo(
53
+ button({ label: 'Saving', icon: spinner({ size: 'sm', color: 'white' }), disabled: true }) + ' ' +
54
+ button({ label: 'Loading', icon: spinner({ size: 'sm' }), variant: 'secondary', disabled: true }),
55
+ `button({
56
+ label: 'Saving',
57
+ icon: spinner({ size: 'sm', color: 'white' }),
58
+ disabled: state.loading,
59
+ })`
60
+ )}
61
+
62
+ ${table(
63
+ ['Prop', 'Type', 'Default', ''],
64
+ [
65
+ ['<code>size</code>', '<code>sm | md | lg</code>', '<code>md</code>', '1rem / 1.5rem / 2.5rem'],
66
+ ['<code>color</code>', '<code>accent | muted | white</code>', '<code>accent</code>', ''],
67
+ ['<code>label</code>', 'string', '<code>Loading…</code>', 'Sets <code>aria-label</code> on the <code>role="status"</code> element'],
68
+ ['<code>class</code>', 'string', '—', ''],
69
+ ]
70
+ )}
71
+ `,
72
+ }),
73
+ }
@@ -0,0 +1,48 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { table } from '../../lib/layout.js'
4
+ import { button, stack } from '../../../../src/ui/index.js'
5
+
6
+ const { prev, next } = prevNext('/components/stack')
7
+
8
+ export default {
9
+ route: '/components/stack',
10
+ meta: {
11
+ title: 'Stack — Pulse Docs',
12
+ description: 'Stack component for Pulse UI.',
13
+ styles: ['/pulse-ui.css', '/docs.css'],
14
+ },
15
+ state: {},
16
+ view: () => renderComponentPage({
17
+ currentHref: '/components/stack',
18
+ prev,
19
+ next,
20
+ name: 'stack',
21
+ description: 'Flex column with consistent vertical gap. Replaces the common pattern of adding <code>margin-bottom</code> to every child element.',
22
+ content: `
23
+ ${demo(
24
+ stack({
25
+ gap: 'sm',
26
+ content:
27
+ button({ label: 'Primary action' }) +
28
+ button({ label: 'Secondary action', variant: 'secondary' }) +
29
+ button({ label: 'Ghost action', variant: 'ghost' }),
30
+ }),
31
+ `stack({ gap: 'sm', align: 'start', content:
32
+ button({ label: 'Primary action' }) +
33
+ button({ label: 'Secondary action', variant: 'secondary' }) +
34
+ button({ label: 'Ghost', variant: 'ghost' })
35
+ })`
36
+ )}
37
+
38
+ ${table(
39
+ ['Prop', 'Type', 'Default', ''],
40
+ [
41
+ ['<code>content</code>', 'string (HTML)', '—', 'Raw HTML slot'],
42
+ ['<code>gap</code>', 'string', "'md'", "'xs' · 'sm' · 'md' · 'lg' · 'xl'"],
43
+ ['<code>align</code>', 'string', "'stretch'", "'stretch' · 'start' · 'center' · 'end'"],
44
+ ]
45
+ )}
46
+ `,
47
+ }),
48
+ }
@@ -0,0 +1,55 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { table } from '../../lib/layout.js'
4
+ import { stat } from '../../../../src/ui/index.js'
5
+
6
+ const { prev, next } = prevNext('/components/stat')
7
+
8
+ export default {
9
+ route: '/components/stat',
10
+ meta: {
11
+ title: 'Stat — Pulse Docs',
12
+ description: 'Stat component for Pulse UI.',
13
+ styles: ['/pulse-ui.css', '/docs.css'],
14
+ },
15
+ state: {},
16
+ view: () => renderComponentPage({
17
+ currentHref: '/components/stat',
18
+ prev,
19
+ next,
20
+ name: 'stat',
21
+ description: 'Numeric metric display with optional trend indicator. Use inside a <code>card()</code> or a flex/grid container for dashboard layouts.',
22
+ content: `
23
+ ${demo(
24
+ `<div style="display:flex;gap:2rem;flex-wrap:wrap">` +
25
+ stat({ label: 'Total users', value: '12,483', change: '+8.2%', trend: 'up' }) +
26
+ stat({ label: 'Bounce rate', value: '24%', change: '−3.1%', trend: 'down' }) +
27
+ stat({ label: 'Avg. session', value: '4m 12s' }) +
28
+ `</div>`,
29
+ `stat({ label: 'Total users', value: '12,483', change: '+8.2%', trend: 'up' })
30
+ stat({ label: 'Bounce rate', value: '24%', change: '−3.1%', trend: 'down' })
31
+ stat({ label: 'Avg. session', value: '4m 12s' })`
32
+ )}
33
+ <p>The <code>center</code> prop centre-aligns the value, label, and change — useful in grids or dashboard cards.</p>
34
+ ${demo(
35
+ `<div style="display:flex;gap:2rem;flex-wrap:wrap">` +
36
+ stat({ label: 'Total users', value: '12,483', change: '+8.2%', trend: 'up', center: true }) +
37
+ stat({ label: 'Bounce rate', value: '24%', change: '−3.1%', trend: 'down', center: true }) +
38
+ stat({ label: 'Avg. session', value: '4m 12s', center: true }) +
39
+ `</div>`,
40
+ `stat({ label: 'Total users', value: '12,483', change: '+8.2%', trend: 'up', center: true })`
41
+ )}
42
+
43
+ ${table(
44
+ ['Prop', 'Type', 'Default', ''],
45
+ [
46
+ ['<code>label</code>', 'string', '—', ''],
47
+ ['<code>value</code>', 'string', '—', 'Formatted value string — e.g. "2.4k", "98%"'],
48
+ ['<code>change</code>', 'string', '—', 'Change label — e.g. "+12%", "−3"'],
49
+ ['<code>trend</code>', 'string', 'neutral', 'up · down · neutral'],
50
+ ['<code>center</code>', 'boolean', 'false', 'Centre-aligns value, label, and change'],
51
+ ]
52
+ )}
53
+ `,
54
+ }),
55
+ }
@@ -0,0 +1,66 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { table } from '../../lib/layout.js'
4
+ import { stepper } from '../../../../src/ui/index.js'
5
+
6
+ const { prev, next } = prevNext('/components/stepper')
7
+
8
+ const STEPS = ['Account', 'Details', 'Payment', 'Review']
9
+
10
+ export default {
11
+ route: '/components/stepper',
12
+ meta: {
13
+ title: 'Stepper — Pulse Docs',
14
+ description: 'Horizontal step progress indicator component for Pulse UI.',
15
+ styles: ['/pulse-ui.css', '/docs.css'],
16
+ },
17
+ state: {},
18
+ view: () => renderComponentPage({
19
+ currentHref: '/components/stepper',
20
+ prev,
21
+ next,
22
+ name: 'stepper',
23
+ description: 'Horizontal step progress indicator. Completed steps show a check icon; the active step has an accent border; upcoming steps are muted. A connector line between steps fills with accent colour as progress advances.',
24
+ content: `
25
+
26
+ <h2 class="doc-h2" id="step1">Step 1 active</h2>
27
+ ${demo(
28
+ stepper({ steps: STEPS, current: 0 }),
29
+ `stepper({
30
+ steps: ['Account', 'Details', 'Payment', 'Review'],
31
+ current: 0,
32
+ })`,
33
+ { col: true }
34
+ )}
35
+
36
+ <h2 class="doc-h2" id="step2">Step 2 active</h2>
37
+ ${demo(
38
+ stepper({ steps: STEPS, current: 1 }),
39
+ `stepper({
40
+ steps: ['Account', 'Details', 'Payment', 'Review'],
41
+ current: 1,
42
+ })`,
43
+ { col: true }
44
+ )}
45
+
46
+ <h2 class="doc-h2" id="complete">All complete</h2>
47
+ ${demo(
48
+ stepper({ steps: STEPS, current: STEPS.length }),
49
+ `stepper({
50
+ steps: ['Account', 'Details', 'Payment', 'Review'],
51
+ current: steps.length, // past the last index = all complete
52
+ })`,
53
+ { col: true }
54
+ )}
55
+
56
+ ${table(
57
+ ['Prop', 'Type', 'Default', ''],
58
+ [
59
+ ['<code>steps</code>', 'string[]', '[]', 'Array of step label strings'],
60
+ ['<code>current</code>', 'number', '0', '0-based index of the active step. Pass <code>steps.length</code> to mark all steps complete.'],
61
+ ['<code>class</code>', 'string', '—', ''],
62
+ ]
63
+ )}
64
+ `,
65
+ }),
66
+ }
@@ -0,0 +1,45 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { badge, table as uiTable } from '../../../../src/ui/index.js'
4
+ import { escHtml } from '../../../../src/html.js'
5
+
6
+ const { prev, next } = prevNext('/components/table')
7
+
8
+ export default {
9
+ route: '/components/table',
10
+ meta: {
11
+ title: 'Table — Pulse Docs',
12
+ description: 'Table component for Pulse UI.',
13
+ styles: ['/pulse-ui.css', '/docs.css'],
14
+ },
15
+ state: {},
16
+ view: () => renderComponentPage({
17
+ currentHref: '/components/table',
18
+ prev,
19
+ next,
20
+ name: 'table',
21
+ description: 'The scroll wrapper has <code>role="region"</code> and <code>tabindex="0"</code> so keyboard users can scroll horizontally on narrow screens. Row cells are raw HTML slots — wrap user-supplied values in <code>escHtml()</code> when building them.',
22
+ content: `
23
+ ${demo(
24
+ uiTable({
25
+ caption: 'Team members',
26
+ headers: ['Name', 'Role', 'Status'],
27
+ rows: [
28
+ [escHtml('Alice Smith'), badge({ label: 'Admin', variant: 'info' }), badge({ label: 'Active', variant: 'success' })],
29
+ [escHtml('Bob Jones'), badge({ label: 'Editor', variant: 'default' }), badge({ label: 'Active', variant: 'success' })],
30
+ [escHtml('Carol White'), badge({ label: 'Viewer', variant: 'default' }), badge({ label: 'Inactive', variant: 'default' })],
31
+ ],
32
+ }),
33
+ `table({
34
+ caption: 'Team members',
35
+ headers: ['Name', 'Role', 'Status'],
36
+ rows: server.users.map(u => [
37
+ escHtml(u.name),
38
+ badge({ label: u.role, variant: u.role === 'admin' ? 'info' : 'default' }),
39
+ badge({ label: u.status, variant: u.active ? 'success' : 'default' }),
40
+ ]),
41
+ })`
42
+ )}
43
+ `,
44
+ }),
45
+ }
@@ -0,0 +1,49 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { table } from '../../lib/layout.js'
4
+ import { testimonial } from '../../../../src/ui/index.js'
5
+
6
+ const { prev, next } = prevNext('/components/testimonial')
7
+
8
+ export default {
9
+ route: '/components/testimonial',
10
+ meta: {
11
+ title: 'Testimonial — Pulse Docs',
12
+ description: 'Testimonial component for Pulse UI.',
13
+ styles: ['/pulse-ui.css', '/docs.css'],
14
+ },
15
+ state: {},
16
+ view: () => renderComponentPage({
17
+ currentHref: '/components/testimonial',
18
+ prev,
19
+ next,
20
+ name: 'testimonial',
21
+ description: 'Customer quote card with star rating, avatar, and attribution. When <code>src</code> is omitted the avatar shows initials derived from <code>name</code>.',
22
+ content: `
23
+ ${demo(
24
+ `<div style="display:grid;grid-template-columns:1fr 1fr;gap:1.5rem">` +
25
+ testimonial({ quote: 'This is the fastest app I have ever used. It replaced three tools I was paying for.', name: 'Alice Marsh', role: 'Head of Product, Stride', rating: 5 }) +
26
+ testimonial({ quote: 'Switched from the competition six months ago and have not looked back once.', name: 'Ben Carter', role: 'Founder, Loopback', rating: 5 }) +
27
+ `</div>`,
28
+ `testimonial({
29
+ quote: 'This is the fastest app I have ever used.',
30
+ name: 'Alice Marsh',
31
+ role: 'Head of Product, Stride',
32
+ rating: 5,
33
+ src: user.avatarUrl, // optional — falls back to initials
34
+ })`
35
+ )}
36
+
37
+ ${table(
38
+ ['Prop', 'Type', 'Default', ''],
39
+ [
40
+ ['<code>quote</code>', 'string', '—', ''],
41
+ ['<code>name</code>', 'string', '—', 'Author name — also used for avatar initials'],
42
+ ['<code>role</code>', 'string', '—', ''],
43
+ ['<code>src</code>', 'string', '—', 'Avatar image URL; omit to show initials'],
44
+ ['<code>rating</code>', 'number', '0', '1–5 stars; omit or set 0 to hide'],
45
+ ]
46
+ )}
47
+ `,
48
+ }),
49
+ }
@@ -0,0 +1,31 @@
1
+ import { renderComponentPage, demo } from '../../lib/component-page.js'
2
+ import { prevNext } from '../../lib/nav.js'
3
+ import { textarea } from '../../../../src/ui/index.js'
4
+
5
+ const { prev, next } = prevNext('/components/textarea')
6
+
7
+ export default {
8
+ route: '/components/textarea',
9
+ meta: {
10
+ title: 'Textarea — Pulse Docs',
11
+ description: 'Textarea component for Pulse UI.',
12
+ styles: ['/pulse-ui.css', '/docs.css'],
13
+ },
14
+ state: {},
15
+ view: () => renderComponentPage({
16
+ currentHref: '/components/textarea',
17
+ prev,
18
+ next,
19
+ name: 'textarea',
20
+ description: 'Multi-line text input with label, hint, and error support. Wired up identically to <code>input</code> — <code>for</code>/<code>id</code> and <code>aria-describedby</code> are derived from <code>name</code>.',
21
+ content: `
22
+ ${demo(
23
+ textarea({ name: 'bio', label: 'Bio', placeholder: 'Tell us about yourself…', hint: 'Max 500 characters', rows: 3 }) +
24
+ textarea({ name: 'note', label: 'Note', value: 'hello world', error: 'Note is too long', rows: 3 }),
25
+ `textarea({ name: 'bio', label: 'Bio', placeholder: 'Tell us about yourself…', hint: 'Max 500 characters' })
26
+ textarea({ name: 'note', label: 'Note', value: state.note, error: server.errors.note, rows: 6 })`,
27
+ { col: true }
28
+ )}
29
+ `,
30
+ }),
31
+ }