@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.
- package/.claude/commands/build-page.md +59 -0
- package/.claude/commands/new-doc-page.md +45 -0
- package/.claude/commands/verify.md +52 -0
- package/.claude/pulse-checklist.md +111 -0
- package/.claude/settings.local.json +102 -0
- package/.github/workflows/ci.yml +22 -0
- package/.github/workflows/publish.yml +41 -0
- package/.pulse/load-reports/home/1773432711417.json +22 -0
- package/CLAUDE.md +383 -0
- package/README.md +95 -0
- package/docs/.claude/pulse-checklist.md +111 -0
- package/docs/public/.pulse-ui-version +1 -0
- package/docs/public/dist/accessibility.boot-5DVTARJU.js +115 -0
- package/docs/public/dist/actions.boot-P66HKQEM.js +164 -0
- package/docs/public/dist/auth.boot-IMAJAUPH.js +140 -0
- package/docs/public/dist/caching.boot-DVR6KDE7.js +53 -0
- package/docs/public/dist/components--accordion.boot-3HVKMNWC.js +11 -0
- package/docs/public/dist/components--alert.boot-GCEXOZAC.js +6 -0
- package/docs/public/dist/components--app-badge.boot-DVT3GCHJ.js +6 -0
- package/docs/public/dist/components--avatar.boot-PSW24EVA.js +5 -0
- package/docs/public/dist/components--badge.boot-TYDY2RMK.js +7 -0
- package/docs/public/dist/components--banner.boot-EI5PZSZK.js +7 -0
- package/docs/public/dist/components--breadcrumbs.boot-SMA2E2GO.js +34 -0
- package/docs/public/dist/components--button.boot-J54BQM2E.js +23 -0
- package/docs/public/dist/components--card.boot-PZGNDIB6.js +138 -0
- package/docs/public/dist/components--carousel.boot-TP6LPFZZ.js +12 -0
- package/docs/public/dist/components--charts.boot-2EOYQWKL.js +108 -0
- package/docs/public/dist/components--checkbox.boot-DS5BSL6T.js +54 -0
- package/docs/public/dist/components--cluster.boot-HHVIBBJG.js +9 -0
- package/docs/public/dist/components--code-window.boot-2GR2DV33.js +20 -0
- package/docs/public/dist/components--container.boot-7LOOGK2K.js +5 -0
- package/docs/public/dist/components--cta.boot-FSNZ5YRT.js +11 -0
- package/docs/public/dist/components--divider.boot-3NI2C3QG.js +6 -0
- package/docs/public/dist/components--empty.boot-YX2UR3PV.js +7 -0
- package/docs/public/dist/components--feature.boot-MUD7NSUO.js +13 -0
- package/docs/public/dist/components--fieldset.boot-J7BYHMKF.js +19 -0
- package/docs/public/dist/components--fileupload.boot-NIKVTTPD.js +52 -0
- package/docs/public/dist/components--footer.boot-EYUK5FRG.js +14 -0
- package/docs/public/dist/components--grid.boot-URDQVDDR.js +59 -0
- package/docs/public/dist/components--heading.boot-BPQKU43E.js +44 -0
- package/docs/public/dist/components--hero.boot-4RAPRGAB.js +17 -0
- package/docs/public/dist/components--icons.boot-ZITNU5JP.js +68 -0
- package/docs/public/dist/components--image.boot-XEEGHQZF.js +19 -0
- package/docs/public/dist/components--input.boot-SGASZG5K.js +7 -0
- package/docs/public/dist/components--list.boot-W3XC5MHD.js +55 -0
- package/docs/public/dist/components--media.boot-5VFIETZO.js +13 -0
- package/docs/public/dist/components--modal.boot-RZUYXBN2.js +47 -0
- package/docs/public/dist/components--nav.boot-ODBOHU7O.js +33 -0
- package/docs/public/dist/components--pricing.boot-4AQ4ZVBY.js +21 -0
- package/docs/public/dist/components--progress.boot-GHAGYZOK.js +30 -0
- package/docs/public/dist/components--prose.boot-QANJL6JI.js +67 -0
- package/docs/public/dist/components--pullquote.boot-Q2WMNAZU.js +22 -0
- package/docs/public/dist/components--radio.boot-TJRDQ2OL.js +75 -0
- package/docs/public/dist/components--rating.boot-QBAN6DEL.js +38 -0
- package/docs/public/dist/components--search.boot-PXH5O5AG.js +17 -0
- package/docs/public/dist/components--section.boot-AQGIYHWW.js +12 -0
- package/docs/public/dist/components--segmented.boot-BEVTKEJO.js +33 -0
- package/docs/public/dist/components--select.boot-47X5RHOC.js +10 -0
- package/docs/public/dist/components--slider.boot-PSRRX7XL.js +47 -0
- package/docs/public/dist/components--spinner.boot-MZ5MO2OH.js +22 -0
- package/docs/public/dist/components--stack.boot-DI4NJXBF.js +9 -0
- package/docs/public/dist/components--stat.boot-QMFUWBQT.js +9 -0
- package/docs/public/dist/components--stepper.boot-34PP2NEV.js +22 -0
- package/docs/public/dist/components--table.boot-FCQGSFIQ.js +11 -0
- package/docs/public/dist/components--testimonial.boot-DWQPDKYG.js +11 -0
- package/docs/public/dist/components--textarea.boot-QVXLBOJ5.js +4 -0
- package/docs/public/dist/components--timeline.boot-26LN52P2.js +95 -0
- package/docs/public/dist/components--toggle.boot-IQQEI76S.js +29 -0
- package/docs/public/dist/components--tooltip.boot-LGHCO6NN.js +9 -0
- package/docs/public/dist/components.boot-SE6PQ4P7.js +103 -0
- package/docs/public/dist/config.boot-DTRRWUE6.js +126 -0
- package/docs/public/dist/constraints.boot-DUHDZBMC.js +71 -0
- package/docs/public/dist/deploy.boot-SLAD3NI2.js +163 -0
- package/docs/public/dist/docs-8e3d4b5c.css +1 -0
- package/docs/public/dist/extending.boot-UA3CN243.js +159 -0
- package/docs/public/dist/faq.boot-6EQAWLQR.js +43 -0
- package/docs/public/dist/getting-started.boot-TDKIFL5U.js +86 -0
- package/docs/public/dist/guard.boot-AUHAWTG4.js +80 -0
- package/docs/public/dist/home.boot-BVQXRH32.js +383 -0
- package/docs/public/dist/how-it-works.boot-LTWAKWKW.js +104 -0
- package/docs/public/dist/hydration.boot-JRM6IPJL.js +78 -0
- package/docs/public/dist/images.boot-M6ZVKTZS.js +80 -0
- package/docs/public/dist/manifest.json +94 -0
- package/docs/public/dist/meta.boot-7NXGPHR4.js +79 -0
- package/docs/public/dist/mutations.boot-F6F43UDX.js +79 -0
- package/docs/public/dist/navigation.boot-AOXWS3ZF.js +57 -0
- package/docs/public/dist/performance.boot-C3UPCOBK.js +98 -0
- package/docs/public/dist/persist.boot-WT32PQOQ.js +61 -0
- package/docs/public/dist/project-structure.boot-FB3LRVJ4.js +63 -0
- package/docs/public/dist/prompt-examples.boot-YKR4VDK4.js +31 -0
- package/docs/public/dist/pulse-ui-81a85c03.css +1 -0
- package/docs/public/dist/raw-responses.boot-M4KA5YXL.js +104 -0
- package/docs/public/dist/routing.boot-FNX5FDGH.js +70 -0
- package/docs/public/dist/runtime-B73WLANC.js +1 -0
- package/docs/public/dist/runtime-KO4BHUQ3.js +49 -0
- package/docs/public/dist/runtime-L2HNXIHW.js +59 -0
- package/docs/public/dist/runtime-QFURDKA2.js +5 -0
- package/docs/public/dist/runtime-UVPXO4IR.js +375 -0
- package/docs/public/dist/runtime-VMJA3Z4N.js +10 -0
- package/docs/public/dist/runtime-ZJ4FXT5O.js +11 -0
- package/docs/public/dist/server-api.boot-K7X3LCFB.js +219 -0
- package/docs/public/dist/server-data.boot-Y7HQYC4R.js +157 -0
- package/docs/public/dist/slash-commands.boot-V2UV7OW2.js +26 -0
- package/docs/public/dist/spec.boot-2WU7ZHCV.js +159 -0
- package/docs/public/dist/state.boot-B24GUE3R.js +73 -0
- package/docs/public/dist/store.boot-TLIB4XHH.js +150 -0
- package/docs/public/dist/streaming.boot-W2DZSMW4.js +80 -0
- package/docs/public/dist/stripe.boot-QN3C2GEL.js +164 -0
- package/docs/public/dist/supabase.boot-BG4XXLZE.js +303 -0
- package/docs/public/dist/testing.boot-6U4WKMTE.js +130 -0
- package/docs/public/dist/validation.boot-PQHYGW5B.js +100 -0
- package/docs/public/docs.css +2020 -0
- package/docs/public/menu.js +83 -0
- package/docs/public/pulse-ui.css +2739 -0
- package/docs/public/pulse-ui.js +236 -0
- package/docs/server.js +192 -0
- package/docs/src/lib/component-page.js +47 -0
- package/docs/src/lib/highlight.js +255 -0
- package/docs/src/lib/layout.js +131 -0
- package/docs/src/lib/metrics-store.js +6 -0
- package/docs/src/lib/nav.js +159 -0
- package/docs/src/lib/stats.js +81 -0
- package/docs/src/pages/accessibility.js +157 -0
- package/docs/src/pages/actions.js +191 -0
- package/docs/src/pages/auth.js +177 -0
- package/docs/src/pages/caching.js +95 -0
- package/docs/src/pages/components/accordion.js +48 -0
- package/docs/src/pages/components/alert.js +35 -0
- package/docs/src/pages/components/app-badge.js +41 -0
- package/docs/src/pages/components/avatar.js +35 -0
- package/docs/src/pages/components/badge.js +36 -0
- package/docs/src/pages/components/banner.js +45 -0
- package/docs/src/pages/components/breadcrumbs.js +94 -0
- package/docs/src/pages/components/button.js +84 -0
- package/docs/src/pages/components/card.js +225 -0
- package/docs/src/pages/components/carousel.js +72 -0
- package/docs/src/pages/components/charts.js +278 -0
- package/docs/src/pages/components/checkbox.js +129 -0
- package/docs/src/pages/components/cluster.js +47 -0
- package/docs/src/pages/components/code-window.js +57 -0
- package/docs/src/pages/components/container.js +40 -0
- package/docs/src/pages/components/cta.js +53 -0
- package/docs/src/pages/components/divider.js +37 -0
- package/docs/src/pages/components/empty.js +36 -0
- package/docs/src/pages/components/feature.js +60 -0
- package/docs/src/pages/components/fieldset.js +65 -0
- package/docs/src/pages/components/fileupload.js +127 -0
- package/docs/src/pages/components/footer.js +58 -0
- package/docs/src/pages/components/grid.js +165 -0
- package/docs/src/pages/components/heading.js +107 -0
- package/docs/src/pages/components/hero.js +65 -0
- package/docs/src/pages/components/icons.js +285 -0
- package/docs/src/pages/components/image.js +71 -0
- package/docs/src/pages/components/input.js +51 -0
- package/docs/src/pages/components/list.js +112 -0
- package/docs/src/pages/components/media.js +51 -0
- package/docs/src/pages/components/modal.js +111 -0
- package/docs/src/pages/components/nav.js +86 -0
- package/docs/src/pages/components/pricing.js +68 -0
- package/docs/src/pages/components/progress.js +102 -0
- package/docs/src/pages/components/prose.js +111 -0
- package/docs/src/pages/components/pullquote.js +71 -0
- package/docs/src/pages/components/radio.js +194 -0
- package/docs/src/pages/components/rating.js +106 -0
- package/docs/src/pages/components/search.js +61 -0
- package/docs/src/pages/components/section.js +59 -0
- package/docs/src/pages/components/segmented.js +121 -0
- package/docs/src/pages/components/select.js +45 -0
- package/docs/src/pages/components/slider.js +114 -0
- package/docs/src/pages/components/spinner.js +73 -0
- package/docs/src/pages/components/stack.js +48 -0
- package/docs/src/pages/components/stat.js +55 -0
- package/docs/src/pages/components/stepper.js +66 -0
- package/docs/src/pages/components/table.js +45 -0
- package/docs/src/pages/components/testimonial.js +49 -0
- package/docs/src/pages/components/textarea.js +31 -0
- package/docs/src/pages/components/timeline.js +227 -0
- package/docs/src/pages/components/toggle.js +84 -0
- package/docs/src/pages/components/tooltip.js +48 -0
- package/docs/src/pages/components.js +204 -0
- package/docs/src/pages/config.js +193 -0
- package/docs/src/pages/constraints.js +99 -0
- package/docs/src/pages/deploy.js +233 -0
- package/docs/src/pages/extending.js +198 -0
- package/docs/src/pages/faq.js +96 -0
- package/docs/src/pages/getting-started.js +106 -0
- package/docs/src/pages/guard.js +121 -0
- package/docs/src/pages/home.js +401 -0
- package/docs/src/pages/how-it-works.js +183 -0
- package/docs/src/pages/hydration.js +98 -0
- package/docs/src/pages/images.js +121 -0
- package/docs/src/pages/meta.js +120 -0
- package/docs/src/pages/mutations.js +106 -0
- package/docs/src/pages/navigation.js +85 -0
- package/docs/src/pages/performance.js +157 -0
- package/docs/src/pages/persist.js +88 -0
- package/docs/src/pages/project-structure.js +90 -0
- package/docs/src/pages/prompt-examples.js +186 -0
- package/docs/src/pages/raw-responses.js +124 -0
- package/docs/src/pages/routing.js +99 -0
- package/docs/src/pages/server-api.js +281 -0
- package/docs/src/pages/server-data.js +185 -0
- package/docs/src/pages/slash-commands.js +55 -0
- package/docs/src/pages/spec.js +207 -0
- package/docs/src/pages/state.js +101 -0
- package/docs/src/pages/store.js +181 -0
- package/docs/src/pages/streaming.js +108 -0
- package/docs/src/pages/stripe.js +193 -0
- package/docs/src/pages/supabase.js +323 -0
- package/docs/src/pages/testing.js +198 -0
- package/docs/src/pages/validation.js +138 -0
- package/examples/contact.js +166 -0
- package/examples/counter.js +94 -0
- package/examples/dev.server.js +91 -0
- package/examples/examples.test.js +394 -0
- package/examples/pricing.js +244 -0
- package/examples/products.js +191 -0
- package/examples/quiz.js +208 -0
- package/examples/shared.js +78 -0
- package/examples/todos.js +162 -0
- package/package.json +75 -0
- package/public/.pulse-ui-version +1 -0
- package/public/chippy-bird.css +246 -0
- package/public/examples/contact.css +119 -0
- package/public/examples/counter.css +79 -0
- package/public/examples/pricing.css +132 -0
- package/public/examples/products.css +100 -0
- package/public/examples/quiz.css +200 -0
- package/public/examples/todos.css +137 -0
- package/public/favicon.ico +0 -0
- package/public/log-dashboard.css +383 -0
- package/public/pulse-ui.css +2740 -0
- package/public/pulse-ui.js +236 -0
- package/public/pulse.css +149 -0
- package/scripts/build.js +411 -0
- package/src/agent/checklist.md +111 -0
- package/src/agent/coverage-check.js +66 -0
- package/src/agent/guide-components.md +274 -0
- package/src/agent/guide-examples.md +54 -0
- package/src/agent/guide-routing.md +36 -0
- package/src/agent/guide-server.md +258 -0
- package/src/agent/guide-spec.md +103 -0
- package/src/agent/guide-styles.md +191 -0
- package/src/agent/guide.md +979 -0
- package/src/agent/identity.md +106 -0
- package/src/agent/workflow.md +108 -0
- package/src/cli/cli.test.js +82 -0
- package/src/cli/dev.js +195 -0
- package/src/cli/discover.js +113 -0
- package/src/cli/index.js +361 -0
- package/src/cli/load-report.js +91 -0
- package/src/cli/load-runner.js +121 -0
- package/src/cli/report-server.js +723 -0
- package/src/cli/report.js +116 -0
- package/src/cli/scaffold.archive.js +1371 -0
- package/src/cli/scaffold.js +349 -0
- package/src/cli/start.js +74 -0
- package/src/html.js +19 -0
- package/src/mcp/server.js +884 -0
- package/src/mcp/validate-worker.js +110 -0
- package/src/runtime/image.js +74 -0
- package/src/runtime/image.test.js +111 -0
- package/src/runtime/index.js +621 -0
- package/src/runtime/navigate.js +146 -0
- package/src/runtime/runtime.test.js +773 -0
- package/src/runtime/ssr.js +464 -0
- package/src/runtime/ssr.test.js +421 -0
- package/src/runtime/store.js +92 -0
- package/src/runtime/toast.js +163 -0
- package/src/server/index.js +1386 -0
- package/src/server/server.test.js +1248 -0
- package/src/spec/schema.js +428 -0
- package/src/spec/schema.test.js +291 -0
- package/src/store/index.js +102 -0
- package/src/store/store.test.js +210 -0
- package/src/testing/html.js +283 -0
- package/src/testing/index.js +249 -0
- package/src/testing/testing.test.js +450 -0
- package/src/ui/accordion.js +28 -0
- package/src/ui/alert.js +43 -0
- package/src/ui/app-badge.js +48 -0
- package/src/ui/avatar.js +47 -0
- package/src/ui/badge.js +24 -0
- package/src/ui/banner.js +26 -0
- package/src/ui/breadcrumbs.js +38 -0
- package/src/ui/button.js +66 -0
- package/src/ui/card.js +34 -0
- package/src/ui/carousel.js +59 -0
- package/src/ui/charts.js +321 -0
- package/src/ui/checkbox.js +65 -0
- package/src/ui/cluster.js +44 -0
- package/src/ui/code-window.js +39 -0
- package/src/ui/container.js +24 -0
- package/src/ui/cta.js +37 -0
- package/src/ui/divider.js +29 -0
- package/src/ui/empty.js +33 -0
- package/src/ui/feature.js +33 -0
- package/src/ui/fieldset.js +37 -0
- package/src/ui/fileupload.js +89 -0
- package/src/ui/footer.js +38 -0
- package/src/ui/grid.js +36 -0
- package/src/ui/heading.js +45 -0
- package/src/ui/hero.js +37 -0
- package/src/ui/icons.js +161 -0
- package/src/ui/index.js +89 -0
- package/src/ui/input.js +74 -0
- package/src/ui/list.js +36 -0
- package/src/ui/media.js +44 -0
- package/src/ui/modal.js +80 -0
- package/src/ui/nav.js +61 -0
- package/src/ui/pricing.js +56 -0
- package/src/ui/progress.js +62 -0
- package/src/ui/prose.js +29 -0
- package/src/ui/pullquote.js +34 -0
- package/src/ui/radio.js +102 -0
- package/src/ui/rating.js +93 -0
- package/src/ui/search.js +77 -0
- package/src/ui/section.js +69 -0
- package/src/ui/segmented.js +50 -0
- package/src/ui/select.js +77 -0
- package/src/ui/slider.js +84 -0
- package/src/ui/spinner.js +34 -0
- package/src/ui/stack.js +36 -0
- package/src/ui/stat.js +52 -0
- package/src/ui/stepper.js +46 -0
- package/src/ui/switch.js +57 -0
- package/src/ui/table.js +45 -0
- package/src/ui/testimonial.js +48 -0
- package/src/ui/textarea.js +72 -0
- package/src/ui/timeline.js +72 -0
- package/src/ui/tooltip.js +28 -0
- package/src/ui/ui.test.js +1241 -0
- package/src/ui/uiimage.js +65 -0
- package/tsconfig.json +13 -0
- package/types/html.d.ts +17 -0
- package/types/image.d.ts +70 -0
- package/types/index.d.ts +7 -0
- package/types/navigate.d.ts +38 -0
- package/types/runtime.d.ts +63 -0
- package/types/schema.d.ts +243 -0
- package/types/server.d.ts +145 -0
- package/types/ssr.d.ts +110 -0
- package/types/testing.d.ts +154 -0
- package/types/ui.d.ts +704 -0
package/src/ui/switch.js
ADDED
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pulse UI — Switch
|
|
3
|
+
*
|
|
4
|
+
* iOS-style toggle switch. Renders a visually hidden checkbox with a custom
|
|
5
|
+
* styled track and thumb. Works with FormData — reads as 'on' when checked.
|
|
6
|
+
*
|
|
7
|
+
* @param {object} opts
|
|
8
|
+
* @param {string} opts.name - Field name
|
|
9
|
+
* @param {string} opts.label - Visible label text
|
|
10
|
+
* @param {boolean} opts.checked - Initial checked state
|
|
11
|
+
* @param {boolean} opts.disabled
|
|
12
|
+
* @param {string} opts.hint - Helper text below the switch
|
|
13
|
+
* @param {string} opts.id - Override generated id
|
|
14
|
+
* @param {string} opts.event - data-event binding, e.g. 'change:setEnabled'
|
|
15
|
+
* @param {string} opts.class
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
import { escHtml as e } from '../html.js'
|
|
19
|
+
|
|
20
|
+
export function toggle({
|
|
21
|
+
name = '',
|
|
22
|
+
label = '',
|
|
23
|
+
checked = false,
|
|
24
|
+
disabled = false,
|
|
25
|
+
hint = '',
|
|
26
|
+
id = '',
|
|
27
|
+
event = '',
|
|
28
|
+
class: cls = '',
|
|
29
|
+
} = {}) {
|
|
30
|
+
const fieldId = e(id || `field-${name}`)
|
|
31
|
+
const hintId = `${fieldId}-hint`
|
|
32
|
+
const classes = ['ui-switch', cls].filter(Boolean).join(' ')
|
|
33
|
+
|
|
34
|
+
const hintHtml = hint
|
|
35
|
+
? `<p id="${hintId}" class="ui-hint">${e(hint)}</p>`
|
|
36
|
+
: ''
|
|
37
|
+
|
|
38
|
+
return `<div class="${e(classes)}">
|
|
39
|
+
<label class="ui-switch-label${disabled ? ' ui-switch-label--disabled' : ''}">
|
|
40
|
+
<input
|
|
41
|
+
type="checkbox"
|
|
42
|
+
id="${fieldId}"
|
|
43
|
+
name="${e(name)}"
|
|
44
|
+
class="ui-switch-input"
|
|
45
|
+
${checked ? 'checked' : ''}
|
|
46
|
+
${disabled ? 'disabled' : ''}
|
|
47
|
+
${event ? `data-event="${e(event)}"` : ''}
|
|
48
|
+
${hint ? `aria-describedby="${hintId}"` : ''}
|
|
49
|
+
>
|
|
50
|
+
<span class="ui-switch-track" aria-hidden="true">
|
|
51
|
+
<span class="ui-switch-thumb"></span>
|
|
52
|
+
</span>
|
|
53
|
+
${label ? `<span class="ui-switch-text">${e(label)}</span>` : ''}
|
|
54
|
+
</label>
|
|
55
|
+
${hintHtml}
|
|
56
|
+
</div>`
|
|
57
|
+
}
|
package/src/ui/table.js
ADDED
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pulse UI — Table
|
|
3
|
+
*
|
|
4
|
+
* Responsive, accessible data table.
|
|
5
|
+
* The scroll wrapper has role="region" and tabindex="0" so keyboard users can
|
|
6
|
+
* scroll horizontally. Cells accept HTML strings (not escaped) — escape
|
|
7
|
+
* user data before passing it in.
|
|
8
|
+
*
|
|
9
|
+
* @param {object} opts
|
|
10
|
+
* @param {string[]} opts.headers - Column header labels (escaped automatically)
|
|
11
|
+
* @param {Array[]} opts.rows - 2D array of cell content (HTML strings)
|
|
12
|
+
* @param {string} opts.caption - Accessible caption (also used as aria-label)
|
|
13
|
+
* @param {string} opts.class
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import { escHtml as e } from '../html.js'
|
|
17
|
+
|
|
18
|
+
export function table({
|
|
19
|
+
headers = [],
|
|
20
|
+
rows = [],
|
|
21
|
+
caption = '',
|
|
22
|
+
class: cls = '',
|
|
23
|
+
} = {}) {
|
|
24
|
+
const wrapClasses = ['ui-table-wrap', cls].filter(Boolean).join(' ')
|
|
25
|
+
|
|
26
|
+
const captionHtml = caption
|
|
27
|
+
? `<caption class="ui-table-caption">${e(caption)}</caption>`
|
|
28
|
+
: ''
|
|
29
|
+
|
|
30
|
+
const ths = headers
|
|
31
|
+
.map(h => `<th scope="col">${e(h)}</th>`)
|
|
32
|
+
.join('')
|
|
33
|
+
|
|
34
|
+
const trs = rows
|
|
35
|
+
.map(row => `<tr>${row.map(cell => `<td>${cell}</td>`).join('')}</tr>`)
|
|
36
|
+
.join('')
|
|
37
|
+
|
|
38
|
+
return `<div class="${e(wrapClasses)}" role="region" aria-label="${e(caption || 'Table')}" tabindex="0">
|
|
39
|
+
<table class="ui-table">
|
|
40
|
+
${captionHtml}
|
|
41
|
+
<thead><tr>${ths}</tr></thead>
|
|
42
|
+
<tbody>${trs}</tbody>
|
|
43
|
+
</table>
|
|
44
|
+
</div>`
|
|
45
|
+
}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pulse UI — Testimonial
|
|
3
|
+
*
|
|
4
|
+
* A customer quote with optional star rating, avatar, and attribution.
|
|
5
|
+
*
|
|
6
|
+
* @param {object} opts
|
|
7
|
+
* @param {string} opts.quote - The testimonial text
|
|
8
|
+
* @param {string} opts.name - Author full name
|
|
9
|
+
* @param {string} opts.role - Author role/company (e.g. "CEO at Acme")
|
|
10
|
+
* @param {string} opts.src - Avatar image URL; falls back to initials when omitted
|
|
11
|
+
* @param {number} opts.rating - Star rating 1–5; omit to hide stars
|
|
12
|
+
* @param {string} opts.class
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
import { escHtml as e } from '../html.js'
|
|
16
|
+
|
|
17
|
+
export function testimonial({
|
|
18
|
+
quote = '',
|
|
19
|
+
name = '',
|
|
20
|
+
role = '',
|
|
21
|
+
src = '',
|
|
22
|
+
rating = 0,
|
|
23
|
+
class: cls = '',
|
|
24
|
+
} = {}) {
|
|
25
|
+
const classes = ['ui-testimonial', cls].filter(Boolean).join(' ')
|
|
26
|
+
|
|
27
|
+
const stars = rating > 0
|
|
28
|
+
? `<p class="ui-testimonial-rating" aria-label="${Math.round(rating)} out of 5 stars">${'★'.repeat(Math.min(5, Math.max(1, Math.round(rating))))}</p>`
|
|
29
|
+
: ''
|
|
30
|
+
|
|
31
|
+
const initials = name.split(' ').map(w => w[0]).join('').slice(0, 2).toUpperCase()
|
|
32
|
+
|
|
33
|
+
const avatarHtml = src
|
|
34
|
+
? `<img src="${e(src)}" alt="${e(name)}" class="ui-testimonial-avatar" loading="lazy" width="40" height="40">`
|
|
35
|
+
: `<div class="ui-testimonial-avatar ui-testimonial-avatar--initials" aria-hidden="true">${e(initials)}</div>`
|
|
36
|
+
|
|
37
|
+
return `<figure class="${e(classes)}">
|
|
38
|
+
${stars}
|
|
39
|
+
<blockquote class="ui-testimonial-quote"><p>${e(quote)}</p></blockquote>
|
|
40
|
+
<figcaption class="ui-testimonial-author">
|
|
41
|
+
${avatarHtml}
|
|
42
|
+
<div class="ui-testimonial-meta">
|
|
43
|
+
<p class="ui-testimonial-name">${e(name)}</p>
|
|
44
|
+
${role ? `<p class="ui-testimonial-role">${e(role)}</p>` : ''}
|
|
45
|
+
</div>
|
|
46
|
+
</figcaption>
|
|
47
|
+
</figure>`
|
|
48
|
+
}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pulse UI — Textarea
|
|
3
|
+
*
|
|
4
|
+
* Multi-line text input with label, hint, and error message.
|
|
5
|
+
*
|
|
6
|
+
* @param {object} opts
|
|
7
|
+
* @param {string} opts.name
|
|
8
|
+
* @param {string} opts.label
|
|
9
|
+
* @param {string} opts.placeholder
|
|
10
|
+
* @param {string} opts.value
|
|
11
|
+
* @param {number} opts.rows
|
|
12
|
+
* @param {string} opts.error
|
|
13
|
+
* @param {string} opts.hint
|
|
14
|
+
* @param {boolean} opts.required
|
|
15
|
+
* @param {boolean} opts.disabled
|
|
16
|
+
* @param {string} opts.id
|
|
17
|
+
* @param {string} opts.class
|
|
18
|
+
* @param {object} opts.attrs
|
|
19
|
+
*/
|
|
20
|
+
|
|
21
|
+
import { escHtml as e } from '../html.js'
|
|
22
|
+
|
|
23
|
+
export function textarea({
|
|
24
|
+
name = '',
|
|
25
|
+
label = '',
|
|
26
|
+
placeholder = '',
|
|
27
|
+
value = '',
|
|
28
|
+
rows = 4,
|
|
29
|
+
error = '',
|
|
30
|
+
hint = '',
|
|
31
|
+
required = false,
|
|
32
|
+
disabled = false,
|
|
33
|
+
id = '',
|
|
34
|
+
class: cls = '',
|
|
35
|
+
attrs = {},
|
|
36
|
+
} = {}) {
|
|
37
|
+
const fieldId = e(id || `field-${name}`)
|
|
38
|
+
const errorId = `${fieldId}-error`
|
|
39
|
+
const hintId = `${fieldId}-hint`
|
|
40
|
+
const described = [error ? errorId : '', hint ? hintId : ''].filter(Boolean).join(' ')
|
|
41
|
+
|
|
42
|
+
const wrapClasses = ['ui-field', error ? 'ui-field--error' : '', cls].filter(Boolean).join(' ')
|
|
43
|
+
|
|
44
|
+
const attrsStr = Object.entries(attrs)
|
|
45
|
+
.map(([k, v]) => ` ${e(k)}="${e(String(v))}"`)
|
|
46
|
+
.join('')
|
|
47
|
+
|
|
48
|
+
const labelHtml = label
|
|
49
|
+
? `<label for="${fieldId}" class="ui-label">${e(label)}${required ? ' <span class="ui-required" aria-hidden="true">*</span>' : ''}</label>`
|
|
50
|
+
: ''
|
|
51
|
+
|
|
52
|
+
const hintHtml = hint ? `<p id="${hintId}" class="ui-hint">${e(hint)}</p>` : ''
|
|
53
|
+
const errorHtml = error ? `<p id="${errorId}" class="ui-error" role="alert">${e(error)}</p>` : ''
|
|
54
|
+
|
|
55
|
+
return `<div class="${e(wrapClasses)}">
|
|
56
|
+
${labelHtml}
|
|
57
|
+
<textarea
|
|
58
|
+
id="${fieldId}"
|
|
59
|
+
name="${e(name)}"
|
|
60
|
+
class="ui-textarea"
|
|
61
|
+
rows="${Number(rows) || 4}"
|
|
62
|
+
${placeholder ? `placeholder="${e(placeholder)}"` : ''}
|
|
63
|
+
${required ? 'required aria-required="true"' : ''}
|
|
64
|
+
${disabled ? 'disabled' : ''}
|
|
65
|
+
${described ? `aria-describedby="${described}"` : ''}
|
|
66
|
+
${error ? 'aria-invalid="true"' : ''}
|
|
67
|
+
${attrsStr}
|
|
68
|
+
>${e(value)}</textarea>
|
|
69
|
+
${hintHtml}
|
|
70
|
+
${errorHtml}
|
|
71
|
+
</div>`
|
|
72
|
+
}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pulse UI — Timeline
|
|
3
|
+
*
|
|
4
|
+
* Ordered list of events or steps connected by a line.
|
|
5
|
+
* Supports vertical (default) and horizontal orientations.
|
|
6
|
+
* Each item accepts a raw HTML content slot — pass any component output.
|
|
7
|
+
*
|
|
8
|
+
* @param {object} opts
|
|
9
|
+
* @param {'vertical'|'horizontal'} opts.direction - Orientation (default: 'vertical')
|
|
10
|
+
* @param {Array} opts.items - Array of timelineItem option objects
|
|
11
|
+
* @param {string} opts.content - Raw HTML alternative to opts.items (advanced)
|
|
12
|
+
* @param {string} opts.class
|
|
13
|
+
*
|
|
14
|
+
* timelineItem:
|
|
15
|
+
* @param {string} opts.dot - Raw HTML inside the dot — SVG icon or emoji
|
|
16
|
+
* @param {'accent'|'success'|'warning'|'error'|'muted'} opts.dotColor
|
|
17
|
+
* @param {string} opts.label - Timestamp or step label (escaped)
|
|
18
|
+
* @param {string} opts.content - Raw HTML body — any component output
|
|
19
|
+
*/
|
|
20
|
+
|
|
21
|
+
import { escHtml as e } from '../html.js'
|
|
22
|
+
|
|
23
|
+
const DOT_COLORS = new Set(['accent', 'success', 'warning', 'error', 'muted'])
|
|
24
|
+
|
|
25
|
+
export function timelineItem({
|
|
26
|
+
dot = '',
|
|
27
|
+
dotColor = 'accent',
|
|
28
|
+
label = '',
|
|
29
|
+
content = '',
|
|
30
|
+
class: cls = '',
|
|
31
|
+
} = {}) {
|
|
32
|
+
if (!DOT_COLORS.has(dotColor)) dotColor = 'accent'
|
|
33
|
+
|
|
34
|
+
const dotClasses = [
|
|
35
|
+
'ui-timeline-dot',
|
|
36
|
+
`ui-timeline-dot--${dotColor}`,
|
|
37
|
+
dot ? 'ui-timeline-dot--icon' : '',
|
|
38
|
+
].filter(Boolean).join(' ')
|
|
39
|
+
|
|
40
|
+
const labelHtml = label
|
|
41
|
+
? `<span class="ui-timeline-label">${e(label)}</span>`
|
|
42
|
+
: ''
|
|
43
|
+
|
|
44
|
+
const itemClasses = ['ui-timeline-item', cls].filter(Boolean).join(' ')
|
|
45
|
+
|
|
46
|
+
return `<li class="${itemClasses}">
|
|
47
|
+
<div class="ui-timeline-side">
|
|
48
|
+
<div class="ui-timeline-connector ui-timeline-connector--before"></div>
|
|
49
|
+
<div class="${e(dotClasses)}" aria-hidden="true">${dot}</div>
|
|
50
|
+
<div class="ui-timeline-connector ui-timeline-connector--after"></div>
|
|
51
|
+
</div>
|
|
52
|
+
<div class="ui-timeline-main">
|
|
53
|
+
${labelHtml}
|
|
54
|
+
<div class="ui-timeline-body">${content}</div>
|
|
55
|
+
</div>
|
|
56
|
+
</li>`
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
export function timeline({
|
|
60
|
+
direction = 'vertical',
|
|
61
|
+
items = [],
|
|
62
|
+
content = '',
|
|
63
|
+
class: cls = '',
|
|
64
|
+
} = {}) {
|
|
65
|
+
if (direction !== 'horizontal') direction = 'vertical'
|
|
66
|
+
|
|
67
|
+
const classes = ['ui-timeline', `ui-timeline--${direction}`, cls].filter(Boolean).join(' ')
|
|
68
|
+
|
|
69
|
+
const itemsHtml = content || items.map(item => timelineItem(item)).join('')
|
|
70
|
+
|
|
71
|
+
return `<ol class="${e(classes)}">${itemsHtml}</ol>`
|
|
72
|
+
}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pulse UI — Tooltip
|
|
3
|
+
*
|
|
4
|
+
* CSS-powered tooltip. Wraps any trigger element — no JavaScript required.
|
|
5
|
+
*
|
|
6
|
+
* @param {object} opts
|
|
7
|
+
* @param {string} opts.content - Tooltip text (plain text; HTML not rendered)
|
|
8
|
+
* @param {string} opts.trigger - Raw HTML slot — the element the tooltip appears on
|
|
9
|
+
* @param {'top'|'bottom'|'left'|'right'} opts.position - Placement (default: 'top')
|
|
10
|
+
* @param {string} opts.class
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import { escHtml as e } from '../html.js'
|
|
14
|
+
|
|
15
|
+
const POSITIONS = new Set(['top', 'bottom', 'left', 'right'])
|
|
16
|
+
|
|
17
|
+
export function tooltip({
|
|
18
|
+
content = '',
|
|
19
|
+
trigger = '',
|
|
20
|
+
position = 'top',
|
|
21
|
+
class: cls = '',
|
|
22
|
+
} = {}) {
|
|
23
|
+
if (!POSITIONS.has(position)) position = 'top'
|
|
24
|
+
|
|
25
|
+
const classes = ['ui-tooltip', `ui-tooltip--${position}`, cls].filter(Boolean).join(' ')
|
|
26
|
+
|
|
27
|
+
return `<span class="${e(classes)}" data-tip="${e(content)}" aria-label="${e(content)}">${trigger}</span>`
|
|
28
|
+
}
|