@mappoh/nova 0.2.3 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (327) hide show
  1. package/README.md +180 -30
  2. package/dist/a11y/index.d.ts +40 -0
  3. package/dist/a11y/index.d.ts.map +1 -0
  4. package/dist/a11y/index.js +162 -0
  5. package/dist/a11y/index.js.map +1 -0
  6. package/dist/alert/alert.d.ts +32 -0
  7. package/dist/alert/alert.d.ts.map +1 -0
  8. package/dist/alert/alert.js +307 -0
  9. package/dist/alert/alert.js.map +1 -0
  10. package/dist/alert/index.d.ts +3 -0
  11. package/dist/alert/index.d.ts.map +1 -0
  12. package/dist/alert/index.js +2 -0
  13. package/dist/alert/index.js.map +1 -0
  14. package/dist/animation/flip.d.ts +28 -0
  15. package/dist/animation/flip.d.ts.map +1 -0
  16. package/dist/animation/flip.js +113 -0
  17. package/dist/animation/flip.js.map +1 -0
  18. package/dist/animation/spring.d.ts +34 -0
  19. package/dist/animation/spring.d.ts.map +1 -0
  20. package/dist/animation/spring.js +86 -0
  21. package/dist/animation/spring.js.map +1 -0
  22. package/dist/animation/stagger.d.ts +43 -0
  23. package/dist/animation/stagger.d.ts.map +1 -0
  24. package/dist/animation/stagger.js +150 -0
  25. package/dist/animation/stagger.js.map +1 -0
  26. package/dist/avatar/avatar.d.ts +27 -0
  27. package/dist/avatar/avatar.d.ts.map +1 -0
  28. package/dist/avatar/avatar.js +132 -0
  29. package/dist/avatar/avatar.js.map +1 -0
  30. package/dist/avatar/index.d.ts +3 -0
  31. package/dist/avatar/index.d.ts.map +1 -0
  32. package/dist/avatar/index.js +2 -0
  33. package/dist/avatar/index.js.map +1 -0
  34. package/dist/badge/badge.d.ts +27 -0
  35. package/dist/badge/badge.d.ts.map +1 -0
  36. package/dist/badge/badge.js +118 -0
  37. package/dist/badge/badge.js.map +1 -0
  38. package/dist/badge/index.d.ts +3 -0
  39. package/dist/badge/index.d.ts.map +1 -0
  40. package/dist/badge/index.js +2 -0
  41. package/dist/badge/index.js.map +1 -0
  42. package/dist/cache/cache.d.ts +28 -0
  43. package/dist/cache/cache.d.ts.map +1 -0
  44. package/dist/cache/cache.js +67 -0
  45. package/dist/cache/cache.js.map +1 -0
  46. package/dist/cache/index.d.ts +3 -0
  47. package/dist/cache/index.d.ts.map +1 -0
  48. package/dist/cache/index.js +2 -0
  49. package/dist/cache/index.js.map +1 -0
  50. package/dist/chart/chart.d.ts +31 -0
  51. package/dist/chart/chart.d.ts.map +1 -0
  52. package/dist/chart/chart.js +372 -0
  53. package/dist/chart/chart.js.map +1 -0
  54. package/dist/chart/index.d.ts +3 -0
  55. package/dist/chart/index.d.ts.map +1 -0
  56. package/dist/chart/index.js +2 -0
  57. package/dist/chart/index.js.map +1 -0
  58. package/dist/color-picker/color-picker.d.ts +18 -0
  59. package/dist/color-picker/color-picker.d.ts.map +1 -0
  60. package/dist/color-picker/color-picker.js +243 -0
  61. package/dist/color-picker/color-picker.js.map +1 -0
  62. package/dist/color-picker/index.d.ts +3 -0
  63. package/dist/color-picker/index.d.ts.map +1 -0
  64. package/dist/color-picker/index.js +2 -0
  65. package/dist/color-picker/index.js.map +1 -0
  66. package/dist/combobox/combobox.d.ts +26 -0
  67. package/dist/combobox/combobox.d.ts.map +1 -0
  68. package/dist/combobox/combobox.js +262 -0
  69. package/dist/combobox/combobox.js.map +1 -0
  70. package/dist/combobox/index.d.ts +3 -0
  71. package/dist/combobox/index.d.ts.map +1 -0
  72. package/dist/combobox/index.js +2 -0
  73. package/dist/combobox/index.js.map +1 -0
  74. package/dist/command-palette/command-palette.d.ts +31 -0
  75. package/dist/command-palette/command-palette.d.ts.map +1 -0
  76. package/dist/command-palette/command-palette.js +590 -0
  77. package/dist/command-palette/command-palette.js.map +1 -0
  78. package/dist/command-palette/index.d.ts +3 -0
  79. package/dist/command-palette/index.d.ts.map +1 -0
  80. package/dist/command-palette/index.js +2 -0
  81. package/dist/command-palette/index.js.map +1 -0
  82. package/dist/component/component.d.ts +20 -2
  83. package/dist/component/component.d.ts.map +1 -1
  84. package/dist/component/component.js +115 -5
  85. package/dist/component/component.js.map +1 -1
  86. package/dist/component/connect.d.ts +50 -0
  87. package/dist/component/connect.d.ts.map +1 -1
  88. package/dist/component/connect.js +135 -0
  89. package/dist/component/connect.js.map +1 -1
  90. package/dist/component/directives.d.ts +20 -0
  91. package/dist/component/directives.d.ts.map +1 -0
  92. package/dist/component/directives.js +42 -0
  93. package/dist/component/directives.js.map +1 -0
  94. package/dist/component/html.d.ts +8 -0
  95. package/dist/component/html.d.ts.map +1 -1
  96. package/dist/component/html.js +11 -0
  97. package/dist/component/html.js.map +1 -1
  98. package/dist/component/index.d.ts +9 -3
  99. package/dist/component/index.d.ts.map +1 -1
  100. package/dist/component/index.js +6 -2
  101. package/dist/component/index.js.map +1 -1
  102. package/dist/component/portal.d.ts +32 -0
  103. package/dist/component/portal.d.ts.map +1 -0
  104. package/dist/component/portal.js +59 -0
  105. package/dist/component/portal.js.map +1 -0
  106. package/dist/component/ref.d.ts +18 -0
  107. package/dist/component/ref.d.ts.map +1 -0
  108. package/dist/component/ref.js +17 -0
  109. package/dist/component/ref.js.map +1 -0
  110. package/dist/component/slot-styles.d.ts +18 -0
  111. package/dist/component/slot-styles.d.ts.map +1 -0
  112. package/dist/component/slot-styles.js +47 -0
  113. package/dist/component/slot-styles.js.map +1 -0
  114. package/dist/component/template.d.ts +2 -0
  115. package/dist/component/template.d.ts.map +1 -1
  116. package/dist/component/template.js +122 -4
  117. package/dist/component/template.js.map +1 -1
  118. package/dist/context/context.d.ts +39 -0
  119. package/dist/context/context.d.ts.map +1 -0
  120. package/dist/context/context.js +111 -0
  121. package/dist/context/context.js.map +1 -0
  122. package/dist/context/index.d.ts +3 -0
  123. package/dist/context/index.d.ts.map +1 -0
  124. package/dist/context/index.js +2 -0
  125. package/dist/context/index.js.map +1 -0
  126. package/dist/data-table/data-table.d.ts +34 -0
  127. package/dist/data-table/data-table.d.ts.map +1 -0
  128. package/dist/data-table/data-table.js +256 -0
  129. package/dist/data-table/data-table.js.map +1 -0
  130. package/dist/data-table/index.d.ts +3 -0
  131. package/dist/data-table/index.d.ts.map +1 -0
  132. package/dist/data-table/index.js +2 -0
  133. package/dist/data-table/index.js.map +1 -0
  134. package/dist/date-picker/date-picker.d.ts +22 -0
  135. package/dist/date-picker/date-picker.d.ts.map +1 -0
  136. package/dist/date-picker/date-picker.js +282 -0
  137. package/dist/date-picker/date-picker.js.map +1 -0
  138. package/dist/date-picker/index.d.ts +3 -0
  139. package/dist/date-picker/index.d.ts.map +1 -0
  140. package/dist/date-picker/index.js +2 -0
  141. package/dist/date-picker/index.js.map +1 -0
  142. package/dist/devtools/devtools.d.ts +54 -0
  143. package/dist/devtools/devtools.d.ts.map +1 -1
  144. package/dist/devtools/devtools.js +86 -0
  145. package/dist/devtools/devtools.js.map +1 -1
  146. package/dist/devtools/index.d.ts +2 -2
  147. package/dist/devtools/index.d.ts.map +1 -1
  148. package/dist/devtools/index.js +1 -1
  149. package/dist/devtools/index.js.map +1 -1
  150. package/dist/editor/editor.d.ts +40 -0
  151. package/dist/editor/editor.d.ts.map +1 -0
  152. package/dist/editor/editor.js +955 -0
  153. package/dist/editor/editor.js.map +1 -0
  154. package/dist/editor/index.d.ts +3 -0
  155. package/dist/editor/index.d.ts.map +1 -0
  156. package/dist/editor/index.js +2 -0
  157. package/dist/editor/index.js.map +1 -0
  158. package/dist/event-bus/event-bus.d.ts +20 -0
  159. package/dist/event-bus/event-bus.d.ts.map +1 -0
  160. package/dist/event-bus/event-bus.js +55 -0
  161. package/dist/event-bus/event-bus.js.map +1 -0
  162. package/dist/event-bus/index.d.ts +3 -0
  163. package/dist/event-bus/index.d.ts.map +1 -0
  164. package/dist/event-bus/index.js +2 -0
  165. package/dist/event-bus/index.js.map +1 -0
  166. package/dist/file-upload/file-upload.d.ts +24 -0
  167. package/dist/file-upload/file-upload.d.ts.map +1 -0
  168. package/dist/file-upload/file-upload.js +177 -0
  169. package/dist/file-upload/file-upload.js.map +1 -0
  170. package/dist/file-upload/index.d.ts +3 -0
  171. package/dist/file-upload/index.d.ts.map +1 -0
  172. package/dist/file-upload/index.js +2 -0
  173. package/dist/file-upload/index.js.map +1 -0
  174. package/dist/forms/form-engine.d.ts +19 -6
  175. package/dist/forms/form-engine.d.ts.map +1 -1
  176. package/dist/forms/form-engine.js +97 -11
  177. package/dist/forms/form-engine.js.map +1 -1
  178. package/dist/forms/validators.d.ts +4 -0
  179. package/dist/forms/validators.d.ts.map +1 -1
  180. package/dist/forms/validators.js +6 -0
  181. package/dist/forms/validators.js.map +1 -1
  182. package/dist/forms/wasm-validators.d.ts +19 -11
  183. package/dist/forms/wasm-validators.d.ts.map +1 -1
  184. package/dist/forms/wasm-validators.js +191 -31
  185. package/dist/forms/wasm-validators.js.map +1 -1
  186. package/dist/gesture/gesture.d.ts +2 -0
  187. package/dist/gesture/gesture.d.ts.map +1 -1
  188. package/dist/gesture/gesture.js +81 -0
  189. package/dist/gesture/gesture.js.map +1 -1
  190. package/dist/http/http.d.ts +8 -0
  191. package/dist/http/http.d.ts.map +1 -1
  192. package/dist/http/http.js +18 -4
  193. package/dist/http/http.js.map +1 -1
  194. package/dist/i18n/i18n.d.ts +6 -0
  195. package/dist/i18n/i18n.d.ts.map +1 -1
  196. package/dist/i18n/i18n.js +71 -9
  197. package/dist/i18n/i18n.js.map +1 -1
  198. package/dist/machine/index.d.ts +3 -0
  199. package/dist/machine/index.d.ts.map +1 -0
  200. package/dist/machine/index.js +2 -0
  201. package/dist/machine/index.js.map +1 -0
  202. package/dist/machine/machine.d.ts +26 -0
  203. package/dist/machine/machine.d.ts.map +1 -0
  204. package/dist/machine/machine.js +79 -0
  205. package/dist/machine/machine.js.map +1 -0
  206. package/dist/modal/modal.d.ts.map +1 -1
  207. package/dist/modal/modal.js +13 -29
  208. package/dist/modal/modal.js.map +1 -1
  209. package/dist/notification-center/index.d.ts +3 -0
  210. package/dist/notification-center/index.d.ts.map +1 -0
  211. package/dist/notification-center/index.js +2 -0
  212. package/dist/notification-center/index.js.map +1 -0
  213. package/dist/notification-center/notification-center.d.ts +55 -0
  214. package/dist/notification-center/notification-center.d.ts.map +1 -0
  215. package/dist/notification-center/notification-center.js +941 -0
  216. package/dist/notification-center/notification-center.js.map +1 -0
  217. package/dist/pagination/index.d.ts +3 -0
  218. package/dist/pagination/index.d.ts.map +1 -0
  219. package/dist/pagination/index.js +2 -0
  220. package/dist/pagination/index.js.map +1 -0
  221. package/dist/pagination/pagination.d.ts +31 -0
  222. package/dist/pagination/pagination.d.ts.map +1 -0
  223. package/dist/pagination/pagination.js +213 -0
  224. package/dist/pagination/pagination.js.map +1 -0
  225. package/dist/progress/progress.d.ts.map +1 -1
  226. package/dist/progress/progress.js +5 -7
  227. package/dist/progress/progress.js.map +1 -1
  228. package/dist/query/index.d.ts +3 -0
  229. package/dist/query/index.d.ts.map +1 -0
  230. package/dist/query/index.js +2 -0
  231. package/dist/query/index.js.map +1 -0
  232. package/dist/query/query.d.ts +31 -0
  233. package/dist/query/query.d.ts.map +1 -0
  234. package/dist/query/query.js +150 -0
  235. package/dist/query/query.js.map +1 -0
  236. package/dist/radio-group/index.d.ts +3 -0
  237. package/dist/radio-group/index.d.ts.map +1 -0
  238. package/dist/radio-group/index.js +2 -0
  239. package/dist/radio-group/index.js.map +1 -0
  240. package/dist/radio-group/radio-group.d.ts +37 -0
  241. package/dist/radio-group/radio-group.d.ts.map +1 -0
  242. package/dist/radio-group/radio-group.js +251 -0
  243. package/dist/radio-group/radio-group.js.map +1 -0
  244. package/dist/rating/index.d.ts +3 -0
  245. package/dist/rating/index.d.ts.map +1 -0
  246. package/dist/rating/index.js +2 -0
  247. package/dist/rating/index.js.map +1 -0
  248. package/dist/rating/rating.d.ts +31 -0
  249. package/dist/rating/rating.d.ts.map +1 -0
  250. package/dist/rating/rating.js +187 -0
  251. package/dist/rating/rating.js.map +1 -0
  252. package/dist/router/index.d.ts +1 -1
  253. package/dist/router/index.d.ts.map +1 -1
  254. package/dist/router/index.js +1 -1
  255. package/dist/router/index.js.map +1 -1
  256. package/dist/router/router.d.ts +30 -1
  257. package/dist/router/router.d.ts.map +1 -1
  258. package/dist/router/router.js +131 -16
  259. package/dist/router/router.js.map +1 -1
  260. package/dist/skeleton/index.d.ts +3 -0
  261. package/dist/skeleton/index.d.ts.map +1 -0
  262. package/dist/skeleton/index.js +2 -0
  263. package/dist/skeleton/index.js.map +1 -0
  264. package/dist/skeleton/skeleton.d.ts +24 -0
  265. package/dist/skeleton/skeleton.d.ts.map +1 -0
  266. package/dist/skeleton/skeleton.js +91 -0
  267. package/dist/skeleton/skeleton.js.map +1 -0
  268. package/dist/slider/index.d.ts +3 -0
  269. package/dist/slider/index.d.ts.map +1 -0
  270. package/dist/slider/index.js +2 -0
  271. package/dist/slider/index.js.map +1 -0
  272. package/dist/slider/slider.d.ts +33 -0
  273. package/dist/slider/slider.d.ts.map +1 -0
  274. package/dist/slider/slider.js +248 -0
  275. package/dist/slider/slider.js.map +1 -0
  276. package/dist/spinner/index.d.ts +3 -0
  277. package/dist/spinner/index.d.ts.map +1 -0
  278. package/dist/spinner/index.js +2 -0
  279. package/dist/spinner/index.js.map +1 -0
  280. package/dist/spinner/spinner.d.ts +23 -0
  281. package/dist/spinner/spinner.d.ts.map +1 -0
  282. package/dist/spinner/spinner.js +82 -0
  283. package/dist/spinner/spinner.js.map +1 -0
  284. package/dist/state/store.d.ts +4 -0
  285. package/dist/state/store.d.ts.map +1 -1
  286. package/dist/state/store.js +27 -7
  287. package/dist/state/store.js.map +1 -1
  288. package/dist/sw/sw.d.ts.map +1 -1
  289. package/dist/sw/sw.js +39 -7
  290. package/dist/sw/sw.js.map +1 -1
  291. package/dist/switch/index.d.ts +3 -0
  292. package/dist/switch/index.d.ts.map +1 -0
  293. package/dist/switch/index.js +2 -0
  294. package/dist/switch/index.js.map +1 -0
  295. package/dist/switch/switch.d.ts +27 -0
  296. package/dist/switch/switch.d.ts.map +1 -0
  297. package/dist/switch/switch.js +163 -0
  298. package/dist/switch/switch.js.map +1 -0
  299. package/dist/theme/index.d.ts +2 -0
  300. package/dist/theme/index.d.ts.map +1 -1
  301. package/dist/theme/index.js +1 -0
  302. package/dist/theme/index.js.map +1 -1
  303. package/dist/theme/scale.d.ts +40 -0
  304. package/dist/theme/scale.d.ts.map +1 -0
  305. package/dist/theme/scale.js +62 -0
  306. package/dist/theme/scale.js.map +1 -0
  307. package/dist/tree-view/index.d.ts +3 -0
  308. package/dist/tree-view/index.d.ts.map +1 -0
  309. package/dist/tree-view/index.js +2 -0
  310. package/dist/tree-view/index.js.map +1 -0
  311. package/dist/tree-view/tree-view.d.ts +29 -0
  312. package/dist/tree-view/tree-view.d.ts.map +1 -0
  313. package/dist/tree-view/tree-view.js +273 -0
  314. package/dist/tree-view/tree-view.js.map +1 -0
  315. package/dist/utils/index.d.ts +29 -0
  316. package/dist/utils/index.d.ts.map +1 -0
  317. package/dist/utils/index.js +114 -0
  318. package/dist/utils/index.js.map +1 -0
  319. package/dist/websocket/index.d.ts +3 -0
  320. package/dist/websocket/index.d.ts.map +1 -0
  321. package/dist/websocket/index.js +2 -0
  322. package/dist/websocket/index.js.map +1 -0
  323. package/dist/websocket/websocket.d.ts +31 -0
  324. package/dist/websocket/websocket.d.ts.map +1 -0
  325. package/dist/websocket/websocket.js +164 -0
  326. package/dist/websocket/websocket.js.map +1 -0
  327. package/package.json +135 -2
@@ -0,0 +1,27 @@
1
+ /**
2
+ * Nova Engine — Switch / Toggle
3
+ *
4
+ * Accessible toggle switch with animated thumb, keyboard interaction,
5
+ * ARIA roles, and CSS custom property theming.
6
+ */
7
+ export interface SwitchOptions {
8
+ /** Initial checked state. Default: false */
9
+ checked?: boolean;
10
+ /** Accessible label for the switch */
11
+ label?: string;
12
+ /** Disable interaction. Default: false */
13
+ disabled?: boolean;
14
+ /** Callback when the value changes */
15
+ onChange?: (checked: boolean) => void;
16
+ }
17
+ export interface SwitchInstance {
18
+ /** Get the current checked state */
19
+ getValue(): boolean;
20
+ /** Set the checked state programmatically */
21
+ setValue(checked: boolean): void;
22
+ /** Remove from DOM and clean up */
23
+ destroy(): void;
24
+ }
25
+ /** Create an accessible toggle switch. */
26
+ export declare function createSwitch(container: string | HTMLElement, options?: SwitchOptions): SwitchInstance;
27
+ //# sourceMappingURL=switch.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"switch.d.ts","sourceRoot":"","sources":["../../src/switch/switch.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,MAAM,WAAW,aAAa;IAC5B,4CAA4C;IAC5C,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,sCAAsC;IACtC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,0CAA0C;IAC1C,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,sCAAsC;IACtC,QAAQ,CAAC,EAAE,CAAC,OAAO,EAAE,OAAO,KAAK,IAAI,CAAC;CACvC;AAED,MAAM,WAAW,cAAc;IAC7B,oCAAoC;IACpC,QAAQ,IAAI,OAAO,CAAC;IACpB,6CAA6C;IAC7C,QAAQ,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAAC;IACjC,mCAAmC;IACnC,OAAO,IAAI,IAAI,CAAC;CACjB;AA+ED,0CAA0C;AAC1C,wBAAgB,YAAY,CAC1B,SAAS,EAAE,MAAM,GAAG,WAAW,EAC/B,OAAO,GAAE,aAAkB,GAC1B,cAAc,CAgGhB"}
@@ -0,0 +1,163 @@
1
+ /**
2
+ * Nova Engine — Switch / Toggle
3
+ *
4
+ * Accessible toggle switch with animated thumb, keyboard interaction,
5
+ * ARIA roles, and CSS custom property theming.
6
+ */
7
+ /* ---- Injected styles ---- */
8
+ const STYLE_ID = 'nova-switch-styles';
9
+ function injectStyles() {
10
+ if (document.getElementById(STYLE_ID))
11
+ return;
12
+ const s = document.createElement('style');
13
+ s.id = STYLE_ID;
14
+ s.textContent = `
15
+ .nova-switch {
16
+ display: inline-flex;
17
+ align-items: center;
18
+ gap: 10px;
19
+ cursor: pointer;
20
+ -webkit-tap-highlight-color: transparent;
21
+ user-select: none;
22
+ }
23
+ .nova-switch--disabled {
24
+ cursor: not-allowed;
25
+ opacity: 0.5;
26
+ }
27
+ .nova-switch__track {
28
+ position: relative;
29
+ width: 44px;
30
+ height: 24px;
31
+ border-radius: 12px;
32
+ background: var(--color-switch-track, rgba(255, 255, 255, 0.12));
33
+ border: 1px solid var(--color-border, rgba(255, 255, 255, 0.08));
34
+ transition: background 0.2s ease, border-color 0.2s ease;
35
+ flex-shrink: 0;
36
+ }
37
+ .nova-switch__track--checked {
38
+ background: var(--color-accent, #a78bfa);
39
+ border-color: var(--color-accent, #a78bfa);
40
+ }
41
+ .nova-switch__thumb {
42
+ position: absolute;
43
+ top: 2px;
44
+ left: 2px;
45
+ width: 18px;
46
+ height: 18px;
47
+ border-radius: 50%;
48
+ background: var(--color-switch-thumb, #fff);
49
+ box-shadow: 0 1px 3px rgba(0, 0, 0, 0.3);
50
+ transition: transform 0.2s ease;
51
+ }
52
+ .nova-switch__track--checked .nova-switch__thumb {
53
+ transform: translateX(20px);
54
+ }
55
+ .nova-switch__label {
56
+ font: 14px/1.5 -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;
57
+ color: var(--color-text-primary, #e5e5e5);
58
+ }
59
+ .nova-switch:focus-visible .nova-switch__track {
60
+ outline: 2px solid var(--color-accent, #a78bfa);
61
+ outline-offset: 2px;
62
+ }
63
+ @media (prefers-reduced-motion: reduce) {
64
+ .nova-switch__thumb,
65
+ .nova-switch__track { transition: none; }
66
+ }`;
67
+ document.head.appendChild(s);
68
+ }
69
+ /* ---- Resolve container ---- */
70
+ function resolveContainer(container) {
71
+ if (typeof container === 'string') {
72
+ const el = document.querySelector(container);
73
+ if (!el)
74
+ throw new Error(`[Nova Switch] Container not found: ${container}`);
75
+ return el;
76
+ }
77
+ return container;
78
+ }
79
+ /* ---- Factory ---- */
80
+ /** Create an accessible toggle switch. */
81
+ export function createSwitch(container, options = {}) {
82
+ const { checked: initialChecked = false, label, disabled = false, onChange, } = options;
83
+ const root = resolveContainer(container);
84
+ let destroyed = false;
85
+ let checked = initialChecked;
86
+ let isDisabled = disabled;
87
+ injectStyles();
88
+ // --- DOM construction ---
89
+ const el = document.createElement('div');
90
+ el.className = 'nova-switch' + (isDisabled ? ' nova-switch--disabled' : '');
91
+ el.setAttribute('role', 'switch');
92
+ el.setAttribute('aria-checked', String(checked));
93
+ el.setAttribute('tabindex', isDisabled ? '-1' : '0');
94
+ if (label)
95
+ el.setAttribute('aria-label', label);
96
+ const track = document.createElement('div');
97
+ track.className = 'nova-switch__track' + (checked ? ' nova-switch__track--checked' : '');
98
+ const thumb = document.createElement('div');
99
+ thumb.className = 'nova-switch__thumb';
100
+ track.appendChild(thumb);
101
+ el.appendChild(track);
102
+ if (label) {
103
+ const labelEl = document.createElement('span');
104
+ labelEl.className = 'nova-switch__label';
105
+ labelEl.textContent = label;
106
+ el.appendChild(labelEl);
107
+ }
108
+ root.appendChild(el);
109
+ // --- Toggle logic ---
110
+ function toggle() {
111
+ if (destroyed || isDisabled)
112
+ return;
113
+ checked = !checked;
114
+ el.setAttribute('aria-checked', String(checked));
115
+ if (checked) {
116
+ track.classList.add('nova-switch__track--checked');
117
+ }
118
+ else {
119
+ track.classList.remove('nova-switch__track--checked');
120
+ }
121
+ onChange?.(checked);
122
+ }
123
+ // --- Event listeners ---
124
+ const listeners = [];
125
+ function on(target, evt, fn) {
126
+ target.addEventListener(evt, fn);
127
+ listeners.push([target, evt, fn]);
128
+ }
129
+ on(el, 'click', () => toggle());
130
+ on(el, 'keydown', ((e) => {
131
+ if (e.key === ' ' || e.key === 'Enter') {
132
+ e.preventDefault();
133
+ toggle();
134
+ }
135
+ }));
136
+ // --- Instance ---
137
+ return {
138
+ getValue() {
139
+ return checked;
140
+ },
141
+ setValue(value) {
142
+ if (destroyed || value === checked)
143
+ return;
144
+ checked = value;
145
+ el.setAttribute('aria-checked', String(checked));
146
+ if (checked) {
147
+ track.classList.add('nova-switch__track--checked');
148
+ }
149
+ else {
150
+ track.classList.remove('nova-switch__track--checked');
151
+ }
152
+ },
153
+ destroy() {
154
+ if (destroyed)
155
+ return;
156
+ destroyed = true;
157
+ for (const [t, e, fn] of listeners)
158
+ t.removeEventListener(e, fn);
159
+ el.remove();
160
+ },
161
+ };
162
+ }
163
+ //# sourceMappingURL=switch.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"switch.js","sourceRoot":"","sources":[""],"names":[],"mappings":"AAAA;;;;;GAKG;AAsBH,+BAA+B;AAE/B,MAAM,QAAQ,GAAG,oBAAoB,CAAC;AAEtC,SAAS,YAAY;IACnB,IAAI,QAAQ,CAAC,cAAc,CAAC,QAAQ,CAAC;QAAE,OAAO;IAC9C,MAAM,CAAC,GAAG,QAAQ,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;IAC1C,CAAC,CAAC,EAAE,GAAG,QAAQ,CAAC;IAChB,CAAC,CAAC,WAAW,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAoDhB,CAAC;IACD,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;AAC/B,CAAC;AAED,iCAAiC;AAEjC,SAAS,gBAAgB,CAAC,SAA+B;IACvD,IAAI,OAAO,SAAS,KAAK,QAAQ,EAAE,CAAC;QAClC,MAAM,EAAE,GAAG,QAAQ,CAAC,aAAa,CAAc,SAAS,CAAC,CAAC;QAC1D,IAAI,CAAC,EAAE;YAAE,MAAM,IAAI,KAAK,CAAC,sCAAsC,SAAS,EAAE,CAAC,CAAC;QAC5E,OAAO,EAAE,CAAC;IACZ,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,uBAAuB;AAEvB,0CAA0C;AAC1C,MAAM,UAAU,YAAY,CAC1B,SAA+B,EAC/B,UAAyB,EAAE;IAE3B,MAAM,EACJ,OAAO,EAAE,cAAc,GAAG,KAAK,EAC/B,KAAK,EACL,QAAQ,GAAG,KAAK,EAChB,QAAQ,GACT,GAAG,OAAO,CAAC;IAEZ,MAAM,IAAI,GAAG,gBAAgB,CAAC,SAAS,CAAC,CAAC;IACzC,IAAI,SAAS,GAAG,KAAK,CAAC;IACtB,IAAI,OAAO,GAAG,cAAc,CAAC;IAC7B,IAAI,UAAU,GAAG,QAAQ,CAAC;IAE1B,YAAY,EAAE,CAAC;IAEf,2BAA2B;IAE3B,MAAM,EAAE,GAAG,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;IACzC,EAAE,CAAC,SAAS,GAAG,aAAa,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAC5E,EAAE,CAAC,YAAY,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;IAClC,EAAE,CAAC,YAAY,CAAC,cAAc,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC;IACjD,EAAE,CAAC,YAAY,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;IACrD,IAAI,KAAK;QAAE,EAAE,CAAC,YAAY,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;IAEhD,MAAM,KAAK,GAAG,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;IAC5C,KAAK,CAAC,SAAS,GAAG,oBAAoB,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,8BAA8B,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAEzF,MAAM,KAAK,GAAG,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;IAC5C,KAAK,CAAC,SAAS,GAAG,oBAAoB,CAAC;IAEvC,KAAK,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;IACzB,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;IAEtB,IAAI,KAAK,EAAE,CAAC;QACV,MAAM,OAAO,GAAG,QAAQ,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QAC/C,OAAO,CAAC,SAAS,GAAG,oBAAoB,CAAC;QACzC,OAAO,CAAC,WAAW,GAAG,KAAK,CAAC;QAC5B,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;IAC1B,CAAC;IAED,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;IAErB,uBAAuB;IAEvB,SAAS,MAAM;QACb,IAAI,SAAS,IAAI,UAAU;YAAE,OAAO;QACpC,OAAO,GAAG,CAAC,OAAO,CAAC;QACnB,EAAE,CAAC,YAAY,CAAC,cAAc,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC;QACjD,IAAI,OAAO,EAAE,CAAC;YACZ,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;QACrD,CAAC;aAAM,CAAC;YACN,KAAK,CAAC,SAAS,CAAC,MAAM,CAAC,6BAA6B,CAAC,CAAC;QACxD,CAAC;QACD,QAAQ,EAAE,CAAC,OAAO,CAAC,CAAC;IACtB,CAAC;IAED,0BAA0B;IAE1B,MAAM,SAAS,GAAgD,EAAE,CAAC;IAClE,SAAS,EAAE,CAAC,MAAmB,EAAE,GAAW,EAAE,EAAiB;QAC7D,MAAM,CAAC,gBAAgB,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QACjC,SAAS,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,GAAG,EAAE,EAAE,CAAC,CAAC,CAAC;IACpC,CAAC;IAED,EAAE,CAAC,EAAE,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC,CAAC;IAEhC,EAAE,CAAC,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC,CAAgB,EAAE,EAAE;QACtC,IAAI,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,GAAG,KAAK,OAAO,EAAE,CAAC;YACvC,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,MAAM,EAAE,CAAC;QACX,CAAC;IACH,CAAC,CAAkB,CAAC,CAAC;IAErB,mBAAmB;IAEnB,OAAO;QACL,QAAQ;YACN,OAAO,OAAO,CAAC;QACjB,CAAC;QACD,QAAQ,CAAC,KAAc;YACrB,IAAI,SAAS,IAAI,KAAK,KAAK,OAAO;gBAAE,OAAO;YAC3C,OAAO,GAAG,KAAK,CAAC;YAChB,EAAE,CAAC,YAAY,CAAC,cAAc,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC;YACjD,IAAI,OAAO,EAAE,CAAC;gBACZ,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;YACrD,CAAC;iBAAM,CAAC;gBACN,KAAK,CAAC,SAAS,CAAC,MAAM,CAAC,6BAA6B,CAAC,CAAC;YACxD,CAAC;QACH,CAAC;QACD,OAAO;YACL,IAAI,SAAS;gBAAE,OAAO;YACtB,SAAS,GAAG,IAAI,CAAC;YACjB,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,SAAS;gBAAE,CAAC,CAAC,mBAAmB,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACjE,EAAE,CAAC,MAAM,EAAE,CAAC;QACd,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -1,3 +1,5 @@
1
1
  export { createTheme } from './theme';
2
2
  export type { ThemeOptions, ThemeInstance } from './theme';
3
+ export { createScale } from './scale';
4
+ export type { ScaleOptions, ScaleInstance } from './scale';
3
5
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/theme/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AACtC,YAAY,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/theme/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AACtC,YAAY,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAC3D,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AACtC,YAAY,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC"}
@@ -1,2 +1,3 @@
1
1
  export { createTheme } from './theme';
2
+ export { createScale } from './scale';
2
3
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":[""],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":[""],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAEtC,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC"}
@@ -0,0 +1,40 @@
1
+ /**
2
+ * Nova Engine — Type Scale
3
+ *
4
+ * Mathematical font-size scale generator using CSS custom properties.
5
+ * Replaces manual `_applyFontSize()` patterns with a systematic approach.
6
+ */
7
+ export interface ScaleOptions {
8
+ /** Base font size in px. Default: 16 */
9
+ base?: number;
10
+ /** Scale ratio (e.g. 1.125 minor second, 1.25 major third, 1.333 perfect fourth). Default: 1.25 */
11
+ ratio?: number;
12
+ /** Number of scale steps above base. Default: 6 */
13
+ steps?: number;
14
+ /** CSS variable prefix. Default: '--font-size' */
15
+ prefix?: string;
16
+ /** Target element to apply CSS vars. Default: document.documentElement */
17
+ target?: HTMLElement;
18
+ }
19
+ export interface ScaleInstance {
20
+ /** Get the computed size for a step (positive = larger, negative = smaller) */
21
+ get(step: number): string;
22
+ /** Re-apply all CSS custom properties to the target */
23
+ apply(): void;
24
+ /** Update the base size and re-apply */
25
+ setBase(px: number): void;
26
+ /** Update the ratio and re-apply */
27
+ setRatio(ratio: number): void;
28
+ /** Remove all applied CSS custom properties */
29
+ destroy(): void;
30
+ }
31
+ /**
32
+ * Create a mathematical type scale that sets CSS custom properties.
33
+ *
34
+ * Generated variables (with default prefix):
35
+ * --font-size-sm-2, --font-size-sm-1
36
+ * --font-size-base
37
+ * --font-size-lg-1 through --font-size-lg-{steps}
38
+ */
39
+ export declare function createScale(options?: ScaleOptions): ScaleInstance;
40
+ //# sourceMappingURL=scale.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scale.d.ts","sourceRoot":"","sources":["../../src/theme/scale.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,MAAM,WAAW,YAAY;IAC3B,wCAAwC;IACxC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,mGAAmG;IACnG,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,mDAAmD;IACnD,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,kDAAkD;IAClD,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,0EAA0E;IAC1E,MAAM,CAAC,EAAE,WAAW,CAAC;CACtB;AAED,MAAM,WAAW,aAAa;IAC5B,+EAA+E;IAC/E,GAAG,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC;IAC1B,uDAAuD;IACvD,KAAK,IAAI,IAAI,CAAC;IACd,wCAAwC;IACxC,OAAO,CAAC,EAAE,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,oCAAoC;IACpC,QAAQ,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,+CAA+C;IAC/C,OAAO,IAAI,IAAI,CAAC;CACjB;AAED;;;;;;;GAOG;AACH,wBAAgB,WAAW,CAAC,OAAO,GAAE,YAAiB,GAAG,aAAa,CAyDrE"}
@@ -0,0 +1,62 @@
1
+ /**
2
+ * Nova Engine — Type Scale
3
+ *
4
+ * Mathematical font-size scale generator using CSS custom properties.
5
+ * Replaces manual `_applyFontSize()` patterns with a systematic approach.
6
+ */
7
+ /**
8
+ * Create a mathematical type scale that sets CSS custom properties.
9
+ *
10
+ * Generated variables (with default prefix):
11
+ * --font-size-sm-2, --font-size-sm-1
12
+ * --font-size-base
13
+ * --font-size-lg-1 through --font-size-lg-{steps}
14
+ */
15
+ export function createScale(options = {}) {
16
+ let { base = 16, ratio = 1.25, } = options;
17
+ const { steps = 6, prefix = '--font-size', target = document.documentElement, } = options;
18
+ function compute(step) {
19
+ return step >= 0
20
+ ? base * Math.pow(ratio, step)
21
+ : base / Math.pow(ratio, Math.abs(step));
22
+ }
23
+ function apply() {
24
+ target.style.setProperty(`${prefix}-base`, `${base}px`);
25
+ // Smaller steps
26
+ for (let i = 1; i <= 2; i++) {
27
+ target.style.setProperty(`${prefix}-sm-${i}`, `${compute(-i).toFixed(2)}px`);
28
+ }
29
+ // Larger steps
30
+ for (let i = 1; i <= steps; i++) {
31
+ target.style.setProperty(`${prefix}-lg-${i}`, `${compute(i).toFixed(2)}px`);
32
+ }
33
+ // Also set a --font-scale var for easy multiplication
34
+ target.style.setProperty(`${prefix}-scale`, String(base / 16));
35
+ }
36
+ apply();
37
+ return {
38
+ get(step) {
39
+ return `${compute(step).toFixed(2)}px`;
40
+ },
41
+ apply,
42
+ setBase(px) {
43
+ base = px;
44
+ apply();
45
+ },
46
+ setRatio(r) {
47
+ ratio = r;
48
+ apply();
49
+ },
50
+ destroy() {
51
+ target.style.removeProperty(`${prefix}-base`);
52
+ target.style.removeProperty(`${prefix}-scale`);
53
+ for (let i = 1; i <= 2; i++) {
54
+ target.style.removeProperty(`${prefix}-sm-${i}`);
55
+ }
56
+ for (let i = 1; i <= steps; i++) {
57
+ target.style.removeProperty(`${prefix}-lg-${i}`);
58
+ }
59
+ },
60
+ };
61
+ }
62
+ //# sourceMappingURL=scale.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scale.js","sourceRoot":"","sources":[""],"names":[],"mappings":"AAAA;;;;;GAKG;AA4BH;;;;;;;GAOG;AACH,MAAM,UAAU,WAAW,CAAC,UAAwB,EAAE;IACpD,IAAI,EACF,IAAI,GAAG,EAAE,EACT,KAAK,GAAG,IAAI,GACb,GAAG,OAAO,CAAC;IACZ,MAAM,EACJ,KAAK,GAAG,CAAC,EACT,MAAM,GAAG,aAAa,EACtB,MAAM,GAAG,QAAQ,CAAC,eAAe,GAClC,GAAG,OAAO,CAAC;IAEZ,SAAS,OAAO,CAAC,IAAY;QAC3B,OAAO,IAAI,IAAI,CAAC;YACd,CAAC,CAAC,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC;YAC9B,CAAC,CAAC,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;IAC7C,CAAC;IAED,SAAS,KAAK;QACZ,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,MAAM,OAAO,EAAE,GAAG,IAAI,IAAI,CAAC,CAAC;QACxD,gBAAgB;QAChB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;YAC5B,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,MAAM,OAAO,CAAC,EAAE,EAAE,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC/E,CAAC;QACD,eAAe;QACf,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC;YAChC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,MAAM,OAAO,CAAC,EAAE,EAAE,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC9E,CAAC;QACD,sDAAsD;QACtD,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,MAAM,QAAQ,EAAE,MAAM,CAAC,IAAI,GAAG,EAAE,CAAC,CAAC,CAAC;IACjE,CAAC;IAED,KAAK,EAAE,CAAC;IAER,OAAO;QACL,GAAG,CAAC,IAAY;YACd,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC;QACzC,CAAC;QACD,KAAK;QACL,OAAO,CAAC,EAAU;YAChB,IAAI,GAAG,EAAE,CAAC;YACV,KAAK,EAAE,CAAC;QACV,CAAC;QACD,QAAQ,CAAC,CAAS;YAChB,KAAK,GAAG,CAAC,CAAC;YACV,KAAK,EAAE,CAAC;QACV,CAAC;QACD,OAAO;YACL,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,GAAG,MAAM,OAAO,CAAC,CAAC;YAC9C,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,GAAG,MAAM,QAAQ,CAAC,CAAC;YAC/C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;gBAC5B,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,GAAG,MAAM,OAAO,CAAC,EAAE,CAAC,CAAC;YACnD,CAAC;YACD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC;gBAChC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,GAAG,MAAM,OAAO,CAAC,EAAE,CAAC,CAAC;YACnD,CAAC;QACH,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,3 @@
1
+ export { createTreeView } from './tree-view';
2
+ export type { TreeNode, TreeViewOptions, TreeViewInstance } from './tree-view';
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/tree-view/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAC7C,YAAY,EAAE,QAAQ,EAAE,eAAe,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { createTreeView } from './tree-view';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":[""],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC"}
@@ -0,0 +1,29 @@
1
+ /**
2
+ * Nova Engine — Tree View
3
+ *
4
+ * Accessible hierarchical tree with expand/collapse, selection, and checkboxes.
5
+ */
6
+ export interface TreeNode {
7
+ id: string;
8
+ label: string;
9
+ children?: TreeNode[];
10
+ icon?: string;
11
+ data?: unknown;
12
+ }
13
+ export interface TreeViewOptions {
14
+ nodes: TreeNode[];
15
+ selectable?: boolean;
16
+ checkable?: boolean;
17
+ onSelect?: (node: TreeNode) => void;
18
+ onCheck?: (checked: TreeNode[]) => void;
19
+ onExpand?: (node: TreeNode, expanded: boolean) => void;
20
+ }
21
+ export interface TreeViewInstance {
22
+ getSelected(): TreeNode | null;
23
+ getChecked(): TreeNode[];
24
+ expandAll(): void;
25
+ collapseAll(): void;
26
+ destroy(): void;
27
+ }
28
+ export declare function createTreeView(container: HTMLElement, options: TreeViewOptions): TreeViewInstance;
29
+ //# sourceMappingURL=tree-view.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tree-view.d.ts","sourceRoot":"","sources":["../../src/tree-view/tree-view.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,MAAM,WAAW,QAAQ;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,QAAQ,EAAE,CAAC;IACtB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB;AAED,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,QAAQ,EAAE,CAAC;IAClB,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,QAAQ,CAAC,EAAE,CAAC,IAAI,EAAE,QAAQ,KAAK,IAAI,CAAC;IACpC,OAAO,CAAC,EAAE,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,IAAI,CAAC;IACxC,QAAQ,CAAC,EAAE,CAAC,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,KAAK,IAAI,CAAC;CACxD;AAED,MAAM,WAAW,gBAAgB;IAC/B,WAAW,IAAI,QAAQ,GAAG,IAAI,CAAC;IAC/B,UAAU,IAAI,QAAQ,EAAE,CAAC;IACzB,SAAS,IAAI,IAAI,CAAC;IAClB,WAAW,IAAI,IAAI,CAAC;IACpB,OAAO,IAAI,IAAI,CAAC;CACjB;AAyBD,wBAAgB,cAAc,CAC5B,SAAS,EAAE,WAAW,EACtB,OAAO,EAAE,eAAe,GACvB,gBAAgB,CAoQlB"}
@@ -0,0 +1,273 @@
1
+ /**
2
+ * Nova Engine — Tree View
3
+ *
4
+ * Accessible hierarchical tree with expand/collapse, selection, and checkboxes.
5
+ */
6
+ const STYLE_ID = 'nova-tree-view-styles';
7
+ function injectStyles() {
8
+ if (document.getElementById(STYLE_ID))
9
+ return;
10
+ const style = document.createElement('style');
11
+ style.id = STYLE_ID;
12
+ style.textContent = `
13
+ .nova-tv{font-family:inherit;font-size:0.875rem}
14
+ .nova-tv ul{list-style:none;margin:0;padding-left:1.25rem}
15
+ .nova-tv>ul{padding-left:0}
16
+ .nova-tv-item{display:flex;align-items:center;gap:4px;padding:3px 6px;border-radius:4px;cursor:pointer;user-select:none}
17
+ .nova-tv-item:hover{background:#f3f4f6}
18
+ .nova-tv-item[aria-selected="true"]{background:#dbeafe}
19
+ .nova-tv-toggle{display:inline-flex;width:16px;height:16px;align-items:center;justify-content:center;font-size:0.625rem;flex-shrink:0;transition:transform 0.15s}
20
+ .nova-tv-toggle[data-expanded]{transform:rotate(90deg)}
21
+ .nova-tv-toggle[data-leaf]{visibility:hidden}
22
+ .nova-tv-icon{flex-shrink:0}
23
+ .nova-tv-label{flex:1;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}
24
+ .nova-tv-check{margin-right:2px;cursor:pointer}
25
+ .nova-tv ul[data-collapsed]{display:none}`;
26
+ document.head.appendChild(style);
27
+ }
28
+ export function createTreeView(container, options) {
29
+ injectStyles();
30
+ const expanded = new Set();
31
+ const checked = new Set();
32
+ let selectedId = null;
33
+ const root = document.createElement('div');
34
+ root.className = 'nova-tv';
35
+ root.setAttribute('role', 'tree');
36
+ container.appendChild(root);
37
+ // Flatten for keyboard nav
38
+ function flatNodes() {
39
+ const result = [];
40
+ function walk(nodes) {
41
+ for (const node of nodes) {
42
+ result.push(node);
43
+ if (node.children?.length && expanded.has(node.id)) {
44
+ walk(node.children);
45
+ }
46
+ }
47
+ }
48
+ walk(options.nodes);
49
+ return result;
50
+ }
51
+ function findNode(id, nodes = options.nodes) {
52
+ for (const node of nodes) {
53
+ if (node.id === id)
54
+ return node;
55
+ if (node.children) {
56
+ const found = findNode(id, node.children);
57
+ if (found)
58
+ return found;
59
+ }
60
+ }
61
+ return null;
62
+ }
63
+ function getAllDescendants(node) {
64
+ const result = [];
65
+ function walk(n) {
66
+ result.push(n);
67
+ n.children?.forEach(walk);
68
+ }
69
+ walk(node);
70
+ return result;
71
+ }
72
+ function renderTree(nodes, parentUl, level) {
73
+ for (const node of nodes) {
74
+ const hasChildren = !!node.children?.length;
75
+ const isExpanded = expanded.has(node.id);
76
+ const li = document.createElement('li');
77
+ li.setAttribute('role', 'treeitem');
78
+ li.setAttribute('aria-level', String(level));
79
+ if (hasChildren)
80
+ li.setAttribute('aria-expanded', String(isExpanded));
81
+ const item = document.createElement('div');
82
+ item.className = 'nova-tv-item';
83
+ item.dataset.id = node.id;
84
+ item.tabIndex = selectedId === node.id ? 0 : -1;
85
+ if (selectedId === node.id)
86
+ item.setAttribute('aria-selected', 'true');
87
+ // Toggle
88
+ const toggle = document.createElement('span');
89
+ toggle.className = 'nova-tv-toggle';
90
+ if (hasChildren) {
91
+ toggle.textContent = '\u25B6';
92
+ if (isExpanded)
93
+ toggle.setAttribute('data-expanded', '');
94
+ }
95
+ else {
96
+ toggle.setAttribute('data-leaf', '');
97
+ }
98
+ item.appendChild(toggle);
99
+ // Checkbox
100
+ if (options.checkable) {
101
+ const cb = document.createElement('input');
102
+ cb.type = 'checkbox';
103
+ cb.className = 'nova-tv-check';
104
+ cb.checked = checked.has(node.id);
105
+ cb.setAttribute('aria-label', `Check ${node.label}`);
106
+ cb.addEventListener('click', (e) => {
107
+ e.stopPropagation();
108
+ const descendants = getAllDescendants(node);
109
+ if (cb.checked) {
110
+ descendants.forEach((d) => checked.add(d.id));
111
+ }
112
+ else {
113
+ descendants.forEach((d) => checked.delete(d.id));
114
+ }
115
+ render();
116
+ options.onCheck?.(getCheckedNodes());
117
+ });
118
+ item.appendChild(cb);
119
+ }
120
+ // Icon
121
+ if (node.icon) {
122
+ const icon = document.createElement('span');
123
+ icon.className = 'nova-tv-icon';
124
+ icon.textContent = node.icon;
125
+ item.appendChild(icon);
126
+ }
127
+ // Label
128
+ const label = document.createElement('span');
129
+ label.className = 'nova-tv-label';
130
+ label.textContent = node.label;
131
+ item.appendChild(label);
132
+ // Click handlers
133
+ item.addEventListener('click', () => {
134
+ if (hasChildren) {
135
+ if (isExpanded)
136
+ expanded.delete(node.id);
137
+ else
138
+ expanded.add(node.id);
139
+ options.onExpand?.(node, !isExpanded);
140
+ render();
141
+ }
142
+ if (options.selectable) {
143
+ selectedId = node.id;
144
+ options.onSelect?.(node);
145
+ render();
146
+ }
147
+ });
148
+ li.appendChild(item);
149
+ // Children
150
+ if (hasChildren) {
151
+ const childUl = document.createElement('ul');
152
+ childUl.setAttribute('role', 'group');
153
+ if (!isExpanded)
154
+ childUl.setAttribute('data-collapsed', '');
155
+ renderTree(node.children, childUl, level + 1);
156
+ li.appendChild(childUl);
157
+ }
158
+ parentUl.appendChild(li);
159
+ }
160
+ }
161
+ function render() {
162
+ while (root.firstChild)
163
+ root.removeChild(root.firstChild);
164
+ const ul = document.createElement('ul');
165
+ ul.setAttribute('role', 'tree');
166
+ renderTree(options.nodes, ul, 1);
167
+ // Keyboard nav
168
+ ul.addEventListener('keydown', (e) => {
169
+ const flat = flatNodes();
170
+ const currentIdx = flat.findIndex((n) => n.id === selectedId);
171
+ if (currentIdx === -1)
172
+ return;
173
+ const current = flat[currentIdx];
174
+ switch (e.key) {
175
+ case 'ArrowDown':
176
+ e.preventDefault();
177
+ if (currentIdx < flat.length - 1) {
178
+ selectedId = flat[currentIdx + 1].id;
179
+ options.onSelect?.(flat[currentIdx + 1]);
180
+ render();
181
+ }
182
+ break;
183
+ case 'ArrowUp':
184
+ e.preventDefault();
185
+ if (currentIdx > 0) {
186
+ selectedId = flat[currentIdx - 1].id;
187
+ options.onSelect?.(flat[currentIdx - 1]);
188
+ render();
189
+ }
190
+ break;
191
+ case 'ArrowRight':
192
+ e.preventDefault();
193
+ if (current.children?.length && !expanded.has(current.id)) {
194
+ expanded.add(current.id);
195
+ options.onExpand?.(current, true);
196
+ render();
197
+ }
198
+ break;
199
+ case 'ArrowLeft':
200
+ e.preventDefault();
201
+ if (expanded.has(current.id)) {
202
+ expanded.delete(current.id);
203
+ options.onExpand?.(current, false);
204
+ render();
205
+ }
206
+ break;
207
+ case ' ':
208
+ e.preventDefault();
209
+ if (options.checkable) {
210
+ if (checked.has(current.id)) {
211
+ getAllDescendants(current).forEach((d) => checked.delete(d.id));
212
+ }
213
+ else {
214
+ getAllDescendants(current).forEach((d) => checked.add(d.id));
215
+ }
216
+ render();
217
+ options.onCheck?.(getCheckedNodes());
218
+ }
219
+ break;
220
+ }
221
+ });
222
+ root.appendChild(ul);
223
+ // Focus the selected item
224
+ const selectedItem = root.querySelector(`[data-id="${selectedId}"]`);
225
+ selectedItem?.focus();
226
+ }
227
+ function getCheckedNodes() {
228
+ const result = [];
229
+ function walk(nodes) {
230
+ for (const node of nodes) {
231
+ if (checked.has(node.id))
232
+ result.push(node);
233
+ if (node.children)
234
+ walk(node.children);
235
+ }
236
+ }
237
+ walk(options.nodes);
238
+ return result;
239
+ }
240
+ // Initial render — select first item
241
+ if (options.selectable && options.nodes.length > 0) {
242
+ selectedId = options.nodes[0].id;
243
+ }
244
+ render();
245
+ return {
246
+ getSelected() {
247
+ return selectedId ? findNode(selectedId) : null;
248
+ },
249
+ getChecked() {
250
+ return getCheckedNodes();
251
+ },
252
+ expandAll() {
253
+ function walk(nodes) {
254
+ for (const node of nodes) {
255
+ if (node.children?.length) {
256
+ expanded.add(node.id);
257
+ walk(node.children);
258
+ }
259
+ }
260
+ }
261
+ walk(options.nodes);
262
+ render();
263
+ },
264
+ collapseAll() {
265
+ expanded.clear();
266
+ render();
267
+ },
268
+ destroy() {
269
+ root.remove();
270
+ },
271
+ };
272
+ }
273
+ //# sourceMappingURL=tree-view.js.map