@happyvertical/smrt-ui 0.30.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 (330) hide show
  1. package/AGENTS.md +50 -0
  2. package/CLAUDE.md +1 -0
  3. package/LICENSE +7 -0
  4. package/dist/actions/__tests__/ripple.test.js +28 -0
  5. package/dist/actions/permission.d.ts +34 -0
  6. package/dist/actions/permission.d.ts.map +1 -0
  7. package/dist/actions/permission.js +70 -0
  8. package/dist/actions/ripple.d.ts +7 -0
  9. package/dist/actions/ripple.d.ts.map +1 -0
  10. package/dist/actions/ripple.js +65 -0
  11. package/dist/components/calendar/Calendar.svelte +520 -0
  12. package/dist/components/calendar/Calendar.svelte.d.ts +17 -0
  13. package/dist/components/calendar/Calendar.svelte.d.ts.map +1 -0
  14. package/dist/components/calendar/DayView.svelte +389 -0
  15. package/dist/components/calendar/DayView.svelte.d.ts +13 -0
  16. package/dist/components/calendar/DayView.svelte.d.ts.map +1 -0
  17. package/dist/components/calendar/index.d.ts +6 -0
  18. package/dist/components/calendar/index.d.ts.map +1 -0
  19. package/dist/components/calendar/index.js +5 -0
  20. package/dist/components/chat/MessageBubble.svelte +126 -0
  21. package/dist/components/chat/MessageBubble.svelte.d.ts +30 -0
  22. package/dist/components/chat/MessageBubble.svelte.d.ts.map +1 -0
  23. package/dist/components/chat/ReactionPicker.svelte +89 -0
  24. package/dist/components/chat/ReactionPicker.svelte.d.ts +19 -0
  25. package/dist/components/chat/ReactionPicker.svelte.d.ts.map +1 -0
  26. package/dist/components/chat/TypingIndicator.svelte +90 -0
  27. package/dist/components/chat/TypingIndicator.svelte.d.ts +17 -0
  28. package/dist/components/chat/TypingIndicator.svelte.d.ts.map +1 -0
  29. package/dist/components/chat/__tests__/chat-primitives.test.js +67 -0
  30. package/dist/components/chat/index.d.ts +10 -0
  31. package/dist/components/chat/index.d.ts.map +1 -0
  32. package/dist/components/chat/index.js +9 -0
  33. package/dist/components/data/DataTable.svelte +519 -0
  34. package/dist/components/data/DataTable.svelte.d.ts +49 -0
  35. package/dist/components/data/DataTable.svelte.d.ts.map +1 -0
  36. package/dist/components/data/__tests__/DataTable.test.js +48 -0
  37. package/dist/components/data/__tests__/data-table-helpers.test.js +36 -0
  38. package/dist/components/data/index.d.ts +6 -0
  39. package/dist/components/data/index.d.ts.map +1 -0
  40. package/dist/components/data/index.js +5 -0
  41. package/dist/components/data/types.d.ts +104 -0
  42. package/dist/components/data/types.d.ts.map +1 -0
  43. package/dist/components/data/types.js +45 -0
  44. package/dist/components/display/ConfidenceBadge.svelte +142 -0
  45. package/dist/components/display/ConfidenceBadge.svelte.d.ts +25 -0
  46. package/dist/components/display/ConfidenceBadge.svelte.d.ts.map +1 -0
  47. package/dist/components/display/CurrencyDisplay.svelte +106 -0
  48. package/dist/components/display/CurrencyDisplay.svelte.d.ts +30 -0
  49. package/dist/components/display/CurrencyDisplay.svelte.d.ts.map +1 -0
  50. package/dist/components/display/DateDisplay.svelte +122 -0
  51. package/dist/components/display/DateDisplay.svelte.d.ts +24 -0
  52. package/dist/components/display/DateDisplay.svelte.d.ts.map +1 -0
  53. package/dist/components/display/Icon.svelte +77 -0
  54. package/dist/components/display/Icon.svelte.d.ts +28 -0
  55. package/dist/components/display/Icon.svelte.d.ts.map +1 -0
  56. package/dist/components/display/StatusBadge.svelte +256 -0
  57. package/dist/components/display/StatusBadge.svelte.d.ts +24 -0
  58. package/dist/components/display/StatusBadge.svelte.d.ts.map +1 -0
  59. package/dist/components/display/__tests__/ConfidenceBadge.test.js +96 -0
  60. package/dist/components/display/__tests__/CurrencyDisplay.test.js +114 -0
  61. package/dist/components/display/__tests__/DateDisplay.test.js +114 -0
  62. package/dist/components/display/__tests__/Icon.test.js +93 -0
  63. package/dist/components/display/__tests__/StatusBadge.test.js +98 -0
  64. package/dist/components/display/index.d.ts +10 -0
  65. package/dist/components/display/index.d.ts.map +1 -0
  66. package/dist/components/display/index.js +9 -0
  67. package/dist/components/display/types.d.ts +5 -0
  68. package/dist/components/display/types.d.ts.map +1 -0
  69. package/dist/components/display/types.js +4 -0
  70. package/dist/components/feedback/ConfirmDialog.svelte +226 -0
  71. package/dist/components/feedback/ConfirmDialog.svelte.d.ts +25 -0
  72. package/dist/components/feedback/ConfirmDialog.svelte.d.ts.map +1 -0
  73. package/dist/components/feedback/LoadingOverlay.svelte +281 -0
  74. package/dist/components/feedback/LoadingOverlay.svelte.d.ts +31 -0
  75. package/dist/components/feedback/LoadingOverlay.svelte.d.ts.map +1 -0
  76. package/dist/components/feedback/Modal.svelte +393 -0
  77. package/dist/components/feedback/Modal.svelte.d.ts +46 -0
  78. package/dist/components/feedback/Modal.svelte.d.ts.map +1 -0
  79. package/dist/components/feedback/ProgressBar.svelte +162 -0
  80. package/dist/components/feedback/ProgressBar.svelte.d.ts +21 -0
  81. package/dist/components/feedback/ProgressBar.svelte.d.ts.map +1 -0
  82. package/dist/components/feedback/__tests__/ConfirmDialog.test.js +111 -0
  83. package/dist/components/feedback/__tests__/LoadingOverlay.test.js +99 -0
  84. package/dist/components/feedback/__tests__/Modal.test.js +72 -0
  85. package/dist/components/feedback/__tests__/ProgressBar.test.js +89 -0
  86. package/dist/components/feedback/index.d.ts +8 -0
  87. package/dist/components/feedback/index.d.ts.map +1 -0
  88. package/dist/components/feedback/index.js +10 -0
  89. package/dist/components/layout/Container.svelte +53 -0
  90. package/dist/components/layout/Container.svelte.d.ts +11 -0
  91. package/dist/components/layout/Container.svelte.d.ts.map +1 -0
  92. package/dist/components/layout/EmptyState.svelte +187 -0
  93. package/dist/components/layout/EmptyState.svelte.d.ts +28 -0
  94. package/dist/components/layout/EmptyState.svelte.d.ts.map +1 -0
  95. package/dist/components/layout/Footer.svelte +63 -0
  96. package/dist/components/layout/Footer.svelte.d.ts +8 -0
  97. package/dist/components/layout/Footer.svelte.d.ts.map +1 -0
  98. package/dist/components/layout/Grid.svelte +241 -0
  99. package/dist/components/layout/Grid.svelte.d.ts +56 -0
  100. package/dist/components/layout/Grid.svelte.d.ts.map +1 -0
  101. package/dist/components/layout/Header.svelte +86 -0
  102. package/dist/components/layout/Header.svelte.d.ts +9 -0
  103. package/dist/components/layout/Header.svelte.d.ts.map +1 -0
  104. package/dist/components/layout/Masthead.svelte +219 -0
  105. package/dist/components/layout/Masthead.svelte.d.ts +13 -0
  106. package/dist/components/layout/Masthead.svelte.d.ts.map +1 -0
  107. package/dist/components/layout/PageHeader.svelte +131 -0
  108. package/dist/components/layout/PageHeader.svelte.d.ts +26 -0
  109. package/dist/components/layout/PageHeader.svelte.d.ts.map +1 -0
  110. package/dist/components/layout/SummaryCard.svelte +203 -0
  111. package/dist/components/layout/SummaryCard.svelte.d.ts +20 -0
  112. package/dist/components/layout/SummaryCard.svelte.d.ts.map +1 -0
  113. package/dist/components/layout/__tests__/Container.test.js +62 -0
  114. package/dist/components/layout/__tests__/EmptyState.test.js +83 -0
  115. package/dist/components/layout/__tests__/Footer.test.js +50 -0
  116. package/dist/components/layout/__tests__/Grid.test.js +121 -0
  117. package/dist/components/layout/__tests__/Header.test.js +48 -0
  118. package/dist/components/layout/__tests__/Masthead.test.js +93 -0
  119. package/dist/components/layout/__tests__/PageHeader.test.js +80 -0
  120. package/dist/components/layout/__tests__/SummaryCard.test.js +82 -0
  121. package/dist/components/layout/index.d.ts +12 -0
  122. package/dist/components/layout/index.d.ts.map +1 -0
  123. package/dist/components/layout/index.js +11 -0
  124. package/dist/components/memberships/MembershipCard.svelte +163 -0
  125. package/dist/components/memberships/MembershipCard.svelte.d.ts +12 -0
  126. package/dist/components/memberships/MembershipCard.svelte.d.ts.map +1 -0
  127. package/dist/components/memberships/MembershipList.svelte +98 -0
  128. package/dist/components/memberships/MembershipList.svelte.d.ts +19 -0
  129. package/dist/components/memberships/MembershipList.svelte.d.ts.map +1 -0
  130. package/dist/components/nav/FilterChips.svelte +152 -0
  131. package/dist/components/nav/FilterChips.svelte.d.ts +19 -0
  132. package/dist/components/nav/FilterChips.svelte.d.ts.map +1 -0
  133. package/dist/components/nav/Tabs.svelte +252 -0
  134. package/dist/components/nav/Tabs.svelte.d.ts +34 -0
  135. package/dist/components/nav/Tabs.svelte.d.ts.map +1 -0
  136. package/dist/components/nav/__tests__/FilterChips.test.js +94 -0
  137. package/dist/components/nav/__tests__/Tabs.test.js +128 -0
  138. package/dist/components/nav/index.d.ts +7 -0
  139. package/dist/components/nav/index.d.ts.map +1 -0
  140. package/dist/components/nav/index.js +6 -0
  141. package/dist/components/nav/types.d.ts +24 -0
  142. package/dist/components/nav/types.d.ts.map +1 -0
  143. package/dist/components/nav/types.js +4 -0
  144. package/dist/components/permissions/PermissionCheck.svelte +45 -0
  145. package/dist/components/permissions/PermissionCheck.svelte.d.ts +19 -0
  146. package/dist/components/permissions/PermissionCheck.svelte.d.ts.map +1 -0
  147. package/dist/components/roles/RoleBadge.svelte +84 -0
  148. package/dist/components/roles/RoleBadge.svelte.d.ts +13 -0
  149. package/dist/components/roles/RoleBadge.svelte.d.ts.map +1 -0
  150. package/dist/components/roles/RoleSelector.svelte +216 -0
  151. package/dist/components/roles/RoleSelector.svelte.d.ts +13 -0
  152. package/dist/components/roles/RoleSelector.svelte.d.ts.map +1 -0
  153. package/dist/components/theme/ThemeProvider.svelte +71 -0
  154. package/dist/components/theme/ThemeProvider.svelte.d.ts +10 -0
  155. package/dist/components/theme/ThemeProvider.svelte.d.ts.map +1 -0
  156. package/dist/components/theme/context.svelte.d.ts +15 -0
  157. package/dist/components/theme/context.svelte.d.ts.map +1 -0
  158. package/dist/components/theme/context.svelte.js +42 -0
  159. package/dist/components/theme/index.d.ts +3 -0
  160. package/dist/components/theme/index.d.ts.map +1 -0
  161. package/dist/components/theme/index.js +2 -0
  162. package/dist/components/ui/Avatar.svelte +167 -0
  163. package/dist/components/ui/Avatar.svelte.d.ts +26 -0
  164. package/dist/components/ui/Avatar.svelte.d.ts.map +1 -0
  165. package/dist/components/ui/Badge.svelte +70 -0
  166. package/dist/components/ui/Badge.svelte.d.ts +12 -0
  167. package/dist/components/ui/Badge.svelte.d.ts.map +1 -0
  168. package/dist/components/ui/Button.svelte +226 -0
  169. package/dist/components/ui/Button.svelte.d.ts +28 -0
  170. package/dist/components/ui/Button.svelte.d.ts.map +1 -0
  171. package/dist/components/ui/Card.svelte +122 -0
  172. package/dist/components/ui/Card.svelte.d.ts +15 -0
  173. package/dist/components/ui/Card.svelte.d.ts.map +1 -0
  174. package/dist/components/ui/Chip.svelte +167 -0
  175. package/dist/components/ui/Chip.svelte.d.ts +33 -0
  176. package/dist/components/ui/Chip.svelte.d.ts.map +1 -0
  177. package/dist/components/ui/Dropdown.svelte +250 -0
  178. package/dist/components/ui/Dropdown.svelte.d.ts +20 -0
  179. package/dist/components/ui/Dropdown.svelte.d.ts.map +1 -0
  180. package/dist/components/ui/Pagination.svelte +294 -0
  181. package/dist/components/ui/Pagination.svelte.d.ts +21 -0
  182. package/dist/components/ui/Pagination.svelte.d.ts.map +1 -0
  183. package/dist/components/ui/Skeleton.svelte +113 -0
  184. package/dist/components/ui/Skeleton.svelte.d.ts +24 -0
  185. package/dist/components/ui/Skeleton.svelte.d.ts.map +1 -0
  186. package/dist/components/ui/Tooltip.svelte +120 -0
  187. package/dist/components/ui/Tooltip.svelte.d.ts +24 -0
  188. package/dist/components/ui/Tooltip.svelte.d.ts.map +1 -0
  189. package/dist/components/ui/Tree.svelte +209 -0
  190. package/dist/components/ui/Tree.svelte.d.ts +17 -0
  191. package/dist/components/ui/Tree.svelte.d.ts.map +1 -0
  192. package/dist/components/ui/__tests__/Badge.test.js +76 -0
  193. package/dist/components/ui/__tests__/Button.test.js +69 -0
  194. package/dist/components/ui/__tests__/Card.test.js +103 -0
  195. package/dist/components/ui/__tests__/Pagination.test.js +99 -0
  196. package/dist/components/ui/__tests__/gap-primitives-interactive.test.js +112 -0
  197. package/dist/components/ui/__tests__/gap-primitives.test.js +84 -0
  198. package/dist/components/ui/index.d.ts +14 -0
  199. package/dist/components/ui/index.d.ts.map +1 -0
  200. package/dist/components/ui/index.js +18 -0
  201. package/dist/i18n/Trans.svelte +29 -0
  202. package/dist/i18n/Trans.svelte.d.ts +24 -0
  203. package/dist/i18n/Trans.svelte.d.ts.map +1 -0
  204. package/dist/i18n/__tests__/i18n.test.js +74 -0
  205. package/dist/i18n/__tests__/render-parity.spec.js +37 -0
  206. package/dist/i18n/context.svelte.d.ts +43 -0
  207. package/dist/i18n/context.svelte.d.ts.map +1 -0
  208. package/dist/i18n/context.svelte.js +69 -0
  209. package/dist/i18n/index.d.ts +17 -0
  210. package/dist/i18n/index.d.ts.map +1 -0
  211. package/dist/i18n/index.js +24 -0
  212. package/dist/i18n/registry.d.ts +44 -0
  213. package/dist/i18n/registry.d.ts.map +1 -0
  214. package/dist/i18n/registry.js +60 -0
  215. package/dist/i18n/render.d.ts +22 -0
  216. package/dist/i18n/render.d.ts.map +1 -0
  217. package/dist/i18n/render.js +44 -0
  218. package/dist/i18n/strings.d.ts +7 -0
  219. package/dist/i18n/strings.d.ts.map +1 -0
  220. package/dist/i18n/strings.js +19 -0
  221. package/dist/i18n/strings.ui.d.ts +34 -0
  222. package/dist/i18n/strings.ui.d.ts.map +1 -0
  223. package/dist/i18n/strings.ui.js +44 -0
  224. package/dist/i18n/use-i18n.d.ts +20 -0
  225. package/dist/i18n/use-i18n.d.ts.map +1 -0
  226. package/dist/i18n/use-i18n.js +21 -0
  227. package/dist/index.d.ts +28 -0
  228. package/dist/index.d.ts.map +1 -0
  229. package/dist/index.js +38 -0
  230. package/dist/registry/index.d.ts +6 -0
  231. package/dist/registry/index.d.ts.map +1 -0
  232. package/dist/registry/index.js +4 -0
  233. package/dist/registry/module-registry.d.ts +58 -0
  234. package/dist/registry/module-registry.d.ts.map +1 -0
  235. package/dist/registry/module-registry.js +94 -0
  236. package/dist/styles/index.d.ts +4 -0
  237. package/dist/styles/index.d.ts.map +1 -0
  238. package/dist/styles/index.js +6 -0
  239. package/dist/styles/tokens.css +76 -0
  240. package/dist/test-support/a11y.d.ts +16 -0
  241. package/dist/test-support/a11y.d.ts.map +1 -0
  242. package/dist/test-support/a11y.js +32 -0
  243. package/dist/test-support/setup.d.ts +11 -0
  244. package/dist/test-support/setup.d.ts.map +1 -0
  245. package/dist/test-support/setup.js +33 -0
  246. package/dist/theme/ThemeProvider.svelte +207 -0
  247. package/dist/theme/ThemeProvider.svelte.d.ts +22 -0
  248. package/dist/theme/ThemeProvider.svelte.d.ts.map +1 -0
  249. package/dist/theme/context.d.ts +49 -0
  250. package/dist/theme/context.d.ts.map +1 -0
  251. package/dist/theme/context.js +32 -0
  252. package/dist/theme/index.d.ts +7 -0
  253. package/dist/theme/index.d.ts.map +1 -0
  254. package/dist/theme/index.js +9 -0
  255. package/dist/theme/tokens.d.ts +309 -0
  256. package/dist/theme/tokens.d.ts.map +1 -0
  257. package/dist/theme/tokens.js +418 -0
  258. package/dist/themes/CUSTOM_THEME_GUIDE.md +341 -0
  259. package/dist/themes/README.md +675 -0
  260. package/dist/themes/ThemeProvider.svelte +275 -0
  261. package/dist/themes/ThemeProvider.svelte.d.ts +24 -0
  262. package/dist/themes/ThemeProvider.svelte.d.ts.map +1 -0
  263. package/dist/themes/__tests__/css-generator.test.js +32 -0
  264. package/dist/themes/__tests__/registry.test.js +43 -0
  265. package/dist/themes/__tests__/token-aliases.test.js +176 -0
  266. package/dist/themes/components/ColorSchemeToggle.svelte +205 -0
  267. package/dist/themes/components/ColorSchemeToggle.svelte.d.ts +14 -0
  268. package/dist/themes/components/ColorSchemeToggle.svelte.d.ts.map +1 -0
  269. package/dist/themes/components/ThemeSwitcher.svelte +188 -0
  270. package/dist/themes/components/ThemeSwitcher.svelte.d.ts +14 -0
  271. package/dist/themes/components/ThemeSwitcher.svelte.d.ts.map +1 -0
  272. package/dist/themes/components/index.d.ts +8 -0
  273. package/dist/themes/components/index.d.ts.map +1 -0
  274. package/dist/themes/components/index.js +7 -0
  275. package/dist/themes/context.svelte.d.ts +30 -0
  276. package/dist/themes/context.svelte.d.ts.map +1 -0
  277. package/dist/themes/context.svelte.js +42 -0
  278. package/dist/themes/create-theme.d.ts +99 -0
  279. package/dist/themes/create-theme.d.ts.map +1 -0
  280. package/dist/themes/create-theme.js +389 -0
  281. package/dist/themes/css-generator.d.ts +44 -0
  282. package/dist/themes/css-generator.d.ts.map +1 -0
  283. package/dist/themes/css-generator.js +226 -0
  284. package/dist/themes/glass/index.d.ts +14 -0
  285. package/dist/themes/glass/index.d.ts.map +1 -0
  286. package/dist/themes/glass/index.js +286 -0
  287. package/dist/themes/index.d.ts +31 -0
  288. package/dist/themes/index.d.ts.map +1 -0
  289. package/dist/themes/index.js +37 -0
  290. package/dist/themes/material/index.d.ts +13 -0
  291. package/dist/themes/material/index.d.ts.map +1 -0
  292. package/dist/themes/material/index.js +269 -0
  293. package/dist/themes/registry.d.ts +64 -0
  294. package/dist/themes/registry.d.ts.map +1 -0
  295. package/dist/themes/registry.js +122 -0
  296. package/dist/themes/shared.d.ts +78 -0
  297. package/dist/themes/shared.d.ts.map +1 -0
  298. package/dist/themes/shared.js +179 -0
  299. package/dist/themes/studio/index.d.ts +14 -0
  300. package/dist/themes/studio/index.d.ts.map +1 -0
  301. package/dist/themes/studio/index.js +270 -0
  302. package/dist/themes/styles/all.css +12 -0
  303. package/dist/themes/styles/glass.css +432 -0
  304. package/dist/themes/styles/index.d.ts +22 -0
  305. package/dist/themes/styles/index.d.ts.map +1 -0
  306. package/dist/themes/styles/index.js +23 -0
  307. package/dist/themes/styles/material.css +364 -0
  308. package/dist/themes/styles/studio.css +416 -0
  309. package/dist/themes/types.d.ts +273 -0
  310. package/dist/themes/types.d.ts.map +1 -0
  311. package/dist/themes/types.js +15 -0
  312. package/dist/types-generic.d.ts +75 -0
  313. package/dist/types-generic.d.ts.map +1 -0
  314. package/dist/types-generic.js +1 -0
  315. package/dist/utils/forms/__tests__/formatters.test.js +27 -0
  316. package/dist/utils/forms/formatters.d.ts +14 -0
  317. package/dist/utils/forms/formatters.d.ts.map +1 -0
  318. package/dist/utils/forms/formatters.js +77 -0
  319. package/dist/utils/import-optional.d.ts +5 -0
  320. package/dist/utils/import-optional.d.ts.map +1 -0
  321. package/dist/utils/import-optional.js +7 -0
  322. package/dist/utils/theme/__tests__/color.test.js +72 -0
  323. package/dist/utils/theme/__tests__/typography.test.js +11 -0
  324. package/dist/utils/theme/color.d.ts +70 -0
  325. package/dist/utils/theme/color.d.ts.map +1 -0
  326. package/dist/utils/theme/color.js +221 -0
  327. package/dist/utils/theme/typography.d.ts +27 -0
  328. package/dist/utils/theme/typography.d.ts.map +1 -0
  329. package/dist/utils/theme/typography.js +30 -0
  330. package/package.json +143 -0
@@ -0,0 +1,519 @@
1
+ <script lang="ts" generics="T">
2
+ /**
3
+ * DataTable - A flexible, accessible data table component
4
+ *
5
+ * Features:
6
+ * - Sortable columns with custom sort functions
7
+ * - Row selection (single and multi-select)
8
+ * - Custom cell and header renderers via Snippets
9
+ * - Loading and empty states
10
+ * - Responsive with sticky header option
11
+ * - Material 3 styling with theme token support
12
+ */
13
+
14
+ import type { Snippet } from 'svelte';
15
+ import { M } from '../../i18n/strings.js';
16
+ import Trans from '../../i18n/Trans.svelte';
17
+ import { useI18n } from '../../i18n/use-i18n.js';
18
+ import type {
19
+ DataTableColumn,
20
+ DataTableProps,
21
+ SortDirection,
22
+ SortState,
23
+ } from './types.js';
24
+ import { defaultSort, getNestedValue } from './types.js';
25
+
26
+ const { t } = useI18n();
27
+
28
+ interface ExtendedProps<T> extends DataTableProps<T> {
29
+ /** Global cell renderer - takes precedence over column.cell */
30
+ cell?: Snippet<
31
+ [{ column: DataTableColumn<T>; row: T; value: unknown; index: number }]
32
+ >;
33
+ }
34
+
35
+ let {
36
+ data = [],
37
+ columns = [],
38
+ rowKey,
39
+ selectable = false,
40
+ selected = $bindable(new Set<string | number>()),
41
+ onSelectionChange,
42
+ onRowClick,
43
+ sortable = false,
44
+ sort = $bindable({ columnId: null, direction: null }),
45
+ onSortChange,
46
+ loading = false,
47
+ empty,
48
+ rowClass,
49
+ size = 'md',
50
+ striped = false,
51
+ hoverable = true,
52
+ stickyHeader = false,
53
+ caption,
54
+ dense = false,
55
+ cell,
56
+ }: ExtendedProps<T> = $props();
57
+
58
+ // Get row key value
59
+ function getRowKey(row: T, index: number): string | number {
60
+ if (!rowKey) return index;
61
+ if (typeof rowKey === 'function') return rowKey(row);
62
+ return row[rowKey] as string | number;
63
+ }
64
+
65
+ // Handle sort click
66
+ function handleSort(column: DataTableColumn<T>) {
67
+ if (!sortable || !column.sortable) return;
68
+
69
+ const newSort: SortState = {
70
+ columnId: column.id,
71
+ direction:
72
+ sort.columnId === column.id
73
+ ? sort.direction === 'asc'
74
+ ? 'desc'
75
+ : sort.direction === 'desc'
76
+ ? null
77
+ : 'asc'
78
+ : 'asc',
79
+ };
80
+
81
+ if (newSort.direction === null) {
82
+ newSort.columnId = null;
83
+ }
84
+
85
+ sort = newSort;
86
+ onSortChange?.(newSort);
87
+ }
88
+
89
+ // Handle row selection
90
+ function handleRowSelect(key: string | number, event: Event) {
91
+ event.stopPropagation();
92
+ const newSelected = new Set(selected);
93
+
94
+ if (newSelected.has(key)) {
95
+ newSelected.delete(key);
96
+ } else {
97
+ newSelected.add(key);
98
+ }
99
+
100
+ selected = newSelected;
101
+ onSelectionChange?.(newSelected);
102
+ }
103
+
104
+ // Handle select all
105
+ function handleSelectAll() {
106
+ if (allSelected) {
107
+ selected = new Set();
108
+ } else {
109
+ selected = new Set(sortedData.map((row, i) => getRowKey(row, i)));
110
+ }
111
+ onSelectionChange?.(selected);
112
+ }
113
+
114
+ // Handle row click
115
+ function handleRowClick(row: T, index: number) {
116
+ onRowClick?.(row, index);
117
+ }
118
+
119
+ // Action to set indeterminate state (can't be set via HTML attribute)
120
+ function setIndeterminate(node: HTMLInputElement, value: boolean) {
121
+ node.indeterminate = value;
122
+ return {
123
+ update(newValue: boolean) {
124
+ node.indeterminate = newValue;
125
+ },
126
+ };
127
+ }
128
+
129
+ // Get visible columns
130
+ const visibleColumns = $derived(columns.filter((col) => !col.hidden));
131
+
132
+ // Sort data
133
+ const sortedData = $derived.by(() => {
134
+ if (!sort.columnId || !sort.direction) return data;
135
+
136
+ const column = columns.find((c) => c.id === sort.columnId);
137
+ if (!column) return data;
138
+
139
+ const accessor = column.accessor ?? column.id;
140
+ const direction = sort.direction;
141
+
142
+ return [...data].sort((a, b) => {
143
+ if (column.sortFn) {
144
+ return column.sortFn(a, b, direction);
145
+ }
146
+ return defaultSort(a, b, String(accessor), direction);
147
+ });
148
+ });
149
+
150
+ // Selection state
151
+ const allSelected = $derived(
152
+ data.length > 0 && data.every((row, i) => selected.has(getRowKey(row, i))),
153
+ );
154
+ const someSelected = $derived(
155
+ data.some((row, i) => selected.has(getRowKey(row, i))) && !allSelected,
156
+ );
157
+
158
+ // Get cell value
159
+ function getCellValue(row: T, column: DataTableColumn<T>): unknown {
160
+ const accessor = column.accessor ?? column.id;
161
+ return getNestedValue(row, String(accessor));
162
+ }
163
+
164
+ // Size classes
165
+ const sizeClasses = {
166
+ sm: 'data-table--sm',
167
+ md: 'data-table--md',
168
+ lg: 'data-table--lg',
169
+ };
170
+ </script>
171
+
172
+ <div
173
+ class="data-table-container"
174
+ class:data-table-container--sticky={stickyHeader}
175
+ >
176
+ <table
177
+ class="data-table {sizeClasses[size]}"
178
+ class:data-table--striped={striped}
179
+ class:data-table--hoverable={hoverable}
180
+ class:data-table--dense={dense}
181
+ class:data-table--loading={loading}
182
+ >
183
+ {#if caption}
184
+ <caption class="data-table__caption">{caption}</caption>
185
+ {/if}
186
+
187
+ <thead class="data-table__head">
188
+ <tr class="data-table__row data-table__row--header">
189
+ {#if selectable}
190
+ <th class="data-table__cell data-table__cell--checkbox" scope="col">
191
+ <input
192
+ type="checkbox"
193
+ checked={allSelected}
194
+ use:setIndeterminate={someSelected}
195
+ onchange={handleSelectAll}
196
+ aria-label={t(M['ui.data_table.select_all'])}
197
+ class="data-table__checkbox"
198
+ />
199
+ </th>
200
+ {/if}
201
+
202
+ {#each visibleColumns as column (column.id)}
203
+ <th
204
+ class="data-table__cell data-table__cell--header"
205
+ class:data-table__cell--sortable={sortable && column.sortable}
206
+ class:data-table__cell--sorted={sort.columnId === column.id}
207
+ style:width={column.width}
208
+ style:min-width={column.minWidth}
209
+ style:max-width={column.maxWidth}
210
+ style:text-align={column.align}
211
+ scope="col"
212
+ aria-sort={sort.columnId === column.id
213
+ ? sort.direction === 'asc'
214
+ ? 'ascending'
215
+ : 'descending'
216
+ : undefined}
217
+ >
218
+ {#if column.header}
219
+ {@render column.header({ column })}
220
+ {:else if sortable && column.sortable}
221
+ <button
222
+ type="button"
223
+ class="data-table__sort-button"
224
+ onclick={() => handleSort(column)}
225
+ >
226
+ <span>{column.label}</span>
227
+ <span class="data-table__sort-icon" aria-hidden="true">
228
+ {#if sort.columnId === column.id}
229
+ {sort.direction === 'asc' ? '↑' : '↓'}
230
+ {:else}
231
+
232
+ {/if}
233
+ </span>
234
+ </button>
235
+ {:else}
236
+ {column.label}
237
+ {/if}
238
+ </th>
239
+ {/each}
240
+ </tr>
241
+ </thead>
242
+
243
+ <tbody class="data-table__body">
244
+ {#if loading}
245
+ <tr class="data-table__row data-table__row--loading">
246
+ <td
247
+ class="data-table__cell data-table__cell--loading"
248
+ colspan={visibleColumns.length + (selectable ? 1 : 0)}
249
+ >
250
+ <div class="data-table__loading-indicator">
251
+ <span class="data-table__spinner"></span>
252
+ <span><Trans key={M['ui.data_table.loading']} /></span>
253
+ </div>
254
+ </td>
255
+ </tr>
256
+ {:else if sortedData.length === 0}
257
+ <tr class="data-table__row data-table__row--empty">
258
+ <td
259
+ class="data-table__cell data-table__cell--empty"
260
+ colspan={visibleColumns.length + (selectable ? 1 : 0)}
261
+ >
262
+ {#if empty}
263
+ {@render empty()}
264
+ {:else}
265
+ <div class="data-table__empty-state">
266
+ <span><Trans key={M['ui.data_table.empty']} /></span>
267
+ </div>
268
+ {/if}
269
+ </td>
270
+ </tr>
271
+ {:else}
272
+ {#each sortedData as row, index (getRowKey(row, index))}
273
+ {@const key = getRowKey(row, index)}
274
+ {@const isSelected = selected.has(key)}
275
+ <tr
276
+ class="data-table__row {rowClass?.(row, index) ?? ''}"
277
+ class:data-table__row--selected={isSelected}
278
+ onclick={() => handleRowClick(row, index)}
279
+ role={onRowClick ? 'button' : undefined}
280
+ tabindex={onRowClick ? 0 : undefined}
281
+ onkeydown={(e) => {
282
+ if (onRowClick && (e.key === 'Enter' || e.key === ' ')) {
283
+ e.preventDefault();
284
+ handleRowClick(row, index);
285
+ }
286
+ }}
287
+ >
288
+ {#if selectable}
289
+ <td class="data-table__cell data-table__cell--checkbox">
290
+ <input
291
+ type="checkbox"
292
+ checked={isSelected}
293
+ onchange={(e) => handleRowSelect(key, e)}
294
+ aria-label={t(M['ui.data_table.select_row'])}
295
+ class="data-table__checkbox"
296
+ />
297
+ </td>
298
+ {/if}
299
+
300
+ {#each visibleColumns as column (column.id)}
301
+ {@const value = getCellValue(row, column)}
302
+ <td
303
+ class="data-table__cell {column.className ?? ''}"
304
+ style:text-align={column.align}
305
+ >
306
+ {#if cell}
307
+ {@render cell({ column, row, value, index })}
308
+ {:else if column.cell}
309
+ {@render column.cell({ row, value, index })}
310
+ {:else}
311
+ {value ?? ''}
312
+ {/if}
313
+ </td>
314
+ {/each}
315
+ </tr>
316
+ {/each}
317
+ {/if}
318
+ </tbody>
319
+ </table>
320
+ </div>
321
+
322
+ <style>
323
+ .data-table-container {
324
+ width: 100%;
325
+ overflow-x: auto;
326
+ }
327
+
328
+ .data-table-container--sticky {
329
+ max-height: 100%;
330
+ overflow-y: auto;
331
+ }
332
+
333
+ .data-table {
334
+ width: 100%;
335
+ border-collapse: collapse;
336
+ border-spacing: 0;
337
+ font-family: var(--smrt-font-family, inherit);
338
+ font-size: var(--smrt-typography-body-medium-size, 0.875rem);
339
+ color: var(--smrt-color-on-surface, #111827);
340
+ background: var(--smrt-color-surface, #ffffff);
341
+ }
342
+
343
+ .data-table__caption {
344
+ padding: var(--smrt-spacing-3, 0.75rem);
345
+ font-weight: var(--smrt-typography-weight-medium, 500);
346
+ text-align: left;
347
+ color: var(--smrt-color-on-surface-variant, #6b7280);
348
+ caption-side: top;
349
+ }
350
+
351
+ /* Header */
352
+ .data-table__head {
353
+ background: var(--smrt-color-surface-container, #f3f4f6);
354
+ }
355
+
356
+ .data-table-container--sticky .data-table__head {
357
+ position: sticky;
358
+ top: 0;
359
+ z-index: 1;
360
+ }
361
+
362
+ .data-table__row--header {
363
+ border-bottom: 1px solid var(--smrt-color-outline-variant, #e5e7eb);
364
+ }
365
+
366
+ .data-table__cell--header {
367
+ padding: var(--smrt-spacing-3, 0.75rem) var(--smrt-spacing-4, 1rem);
368
+ font-weight: var(--smrt-typography-weight-semibold, 600);
369
+ text-align: left;
370
+ white-space: nowrap;
371
+ color: var(--smrt-color-on-surface, #111827);
372
+ }
373
+
374
+ .data-table__cell--sortable {
375
+ cursor: pointer;
376
+ user-select: none;
377
+ }
378
+
379
+ .data-table__sort-button {
380
+ display: inline-flex;
381
+ align-items: center;
382
+ gap: var(--smrt-spacing-1, 0.25rem);
383
+ padding: 0;
384
+ border: none;
385
+ background: none;
386
+ font: inherit;
387
+ font-weight: var(--smrt-typography-weight-semibold, 600);
388
+ color: inherit;
389
+ cursor: pointer;
390
+ }
391
+
392
+ .data-table__sort-button:hover {
393
+ color: var(--smrt-color-primary, #3b82f6);
394
+ }
395
+
396
+ .data-table__sort-icon {
397
+ font-size: var(--smrt-typography-label-medium-size, 0.75rem);
398
+ opacity: 0.5;
399
+ }
400
+
401
+ .data-table__cell--sorted .data-table__sort-icon {
402
+ opacity: 1;
403
+ color: var(--smrt-color-primary, #3b82f6);
404
+ }
405
+
406
+ /* Body */
407
+ .data-table__body {
408
+ background: var(--smrt-color-surface, #ffffff);
409
+ }
410
+
411
+ .data-table__row {
412
+ border-bottom: 1px solid var(--smrt-color-outline-variant, #e5e7eb);
413
+ transition: background-color var(--smrt-duration-fast, 150ms) var(--smrt-easing-standard, ease);
414
+ }
415
+
416
+ .data-table--hoverable .data-table__row:hover:not(.data-table__row--loading):not(.data-table__row--empty) {
417
+ background: var(--smrt-color-surface-container-low, #f9fafb);
418
+ }
419
+
420
+ .data-table__row--selected {
421
+ background: var(--smrt-color-primary-container, #dbeafe) !important;
422
+ }
423
+
424
+ .data-table--striped .data-table__row:nth-child(even) {
425
+ background: var(--smrt-color-surface-container-lowest, #fafafa);
426
+ }
427
+
428
+ .data-table__row[role='button'] {
429
+ cursor: pointer;
430
+ }
431
+
432
+ .data-table__row[role='button']:focus-visible {
433
+ outline: 2px solid var(--smrt-color-primary, #3b82f6);
434
+ outline-offset: -2px;
435
+ }
436
+
437
+ .data-table__cell {
438
+ padding: var(--smrt-spacing-3, 0.75rem) var(--smrt-spacing-4, 1rem);
439
+ vertical-align: middle;
440
+ }
441
+
442
+ .data-table__cell--checkbox {
443
+ width: 48px;
444
+ text-align: center;
445
+ }
446
+
447
+ .data-table__checkbox {
448
+ width: 18px;
449
+ height: 18px;
450
+ cursor: pointer;
451
+ accent-color: var(--smrt-color-primary, #3b82f6);
452
+ }
453
+
454
+ /* Loading */
455
+ .data-table__cell--loading,
456
+ .data-table__cell--empty {
457
+ padding: var(--smrt-spacing-8, 2rem);
458
+ text-align: center;
459
+ }
460
+
461
+ .data-table__loading-indicator {
462
+ display: flex;
463
+ align-items: center;
464
+ justify-content: center;
465
+ gap: var(--smrt-spacing-2, 0.5rem);
466
+ color: var(--smrt-color-on-surface-variant, #6b7280);
467
+ }
468
+
469
+ .data-table__spinner {
470
+ width: 20px;
471
+ height: 20px;
472
+ border: 2px solid var(--smrt-color-outline-variant, #e5e7eb);
473
+ border-top-color: var(--smrt-color-primary, #3b82f6);
474
+ border-radius: var(--smrt-radius-full, 9999px);
475
+ animation: spin 0.8s linear infinite;
476
+ }
477
+
478
+ @keyframes spin {
479
+ to {
480
+ transform: rotate(360deg);
481
+ }
482
+ }
483
+
484
+ .data-table__empty-state {
485
+ color: var(--smrt-color-on-surface-variant, #6b7280);
486
+ }
487
+
488
+ /* Size variants */
489
+ .data-table--sm .data-table__cell {
490
+ padding: var(--smrt-spacing-2, 0.5rem) var(--smrt-spacing-3, 0.75rem);
491
+ }
492
+
493
+ .data-table--sm .data-table__cell--header {
494
+ padding: var(--smrt-spacing-2, 0.5rem) var(--smrt-spacing-3, 0.75rem);
495
+ }
496
+
497
+ .data-table--lg .data-table__cell {
498
+ padding: var(--smrt-spacing-4, 1rem) var(--smrt-spacing-5, 1.25rem);
499
+ }
500
+
501
+ .data-table--lg .data-table__cell--header {
502
+ padding: var(--smrt-spacing-4, 1rem) var(--smrt-spacing-5, 1.25rem);
503
+ }
504
+
505
+ /* Dense mode */
506
+ .data-table--dense .data-table__cell {
507
+ padding: var(--smrt-spacing-1, 0.25rem) var(--smrt-spacing-2, 0.5rem);
508
+ }
509
+
510
+ .data-table--dense .data-table__cell--header {
511
+ padding: var(--smrt-spacing-2, 0.5rem) var(--smrt-spacing-2, 0.5rem);
512
+ }
513
+
514
+ /* Loading overlay */
515
+ .data-table--loading {
516
+ opacity: 0.7;
517
+ pointer-events: none;
518
+ }
519
+ </style>
@@ -0,0 +1,49 @@
1
+ /**
2
+ * DataTable - A flexible, accessible data table component
3
+ *
4
+ * Features:
5
+ * - Sortable columns with custom sort functions
6
+ * - Row selection (single and multi-select)
7
+ * - Custom cell and header renderers via Snippets
8
+ * - Loading and empty states
9
+ * - Responsive with sticky header option
10
+ * - Material 3 styling with theme token support
11
+ */
12
+ import type { Snippet } from 'svelte';
13
+ import type { DataTableColumn, DataTableProps } from './types.js';
14
+ interface ExtendedProps<T> extends DataTableProps<T> {
15
+ /** Global cell renderer - takes precedence over column.cell */
16
+ cell?: Snippet<[
17
+ {
18
+ column: DataTableColumn<T>;
19
+ row: T;
20
+ value: unknown;
21
+ index: number;
22
+ }
23
+ ]>;
24
+ }
25
+ declare function $$render<T>(): {
26
+ props: ExtendedProps<T>;
27
+ exports: {};
28
+ bindings: "sort" | "selected";
29
+ slots: {};
30
+ events: {};
31
+ };
32
+ declare class __sveltets_Render<T> {
33
+ props(): ReturnType<typeof $$render<T>>['props'];
34
+ events(): ReturnType<typeof $$render<T>>['events'];
35
+ slots(): ReturnType<typeof $$render<T>>['slots'];
36
+ bindings(): "sort" | "selected";
37
+ exports(): {};
38
+ }
39
+ interface $$IsomorphicComponent {
40
+ new <T>(options: import('svelte').ComponentConstructorOptions<ReturnType<__sveltets_Render<T>['props']>>): import('svelte').SvelteComponent<ReturnType<__sveltets_Render<T>['props']>, ReturnType<__sveltets_Render<T>['events']>, ReturnType<__sveltets_Render<T>['slots']>> & {
41
+ $$bindings?: ReturnType<__sveltets_Render<T>['bindings']>;
42
+ } & ReturnType<__sveltets_Render<T>['exports']>;
43
+ <T>(internal: unknown, props: ReturnType<__sveltets_Render<T>['props']> & {}): ReturnType<__sveltets_Render<T>['exports']>;
44
+ z_$$bindings?: ReturnType<__sveltets_Render<any>['bindings']>;
45
+ }
46
+ declare const DataTable: $$IsomorphicComponent;
47
+ type DataTable<T> = InstanceType<typeof DataTable<T>>;
48
+ export default DataTable;
49
+ //# sourceMappingURL=DataTable.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DataTable.svelte.d.ts","sourceRoot":"","sources":["../../../src/components/data/DataTable.svelte.ts"],"names":[],"mappings":"AAGA;;;;;;;;;;GAUG;AACH,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,QAAQ,CAAC;AAItC,OAAO,KAAK,EACV,eAAe,EACf,cAAc,EAGf,MAAM,YAAY,CAAC;AAIpB,UAAU,aAAa,CAAC,CAAC,CAAE,SAAQ,cAAc,CAAC,CAAC,CAAC;IAClD,+DAA+D;IAC/D,IAAI,CAAC,EAAE,OAAO,CACZ;QAAC;YAAE,MAAM,EAAE,eAAe,CAAC,CAAC,CAAC,CAAC;YAAC,GAAG,EAAE,CAAC,CAAC;YAAC,KAAK,EAAE,OAAO,CAAC;YAAC,KAAK,EAAE,MAAM,CAAA;SAAE;KAAC,CACxE,CAAC;CACH;AAAC,iBAAS,QAAQ,CAAC,CAAC;WA0PQ,aAAa,CAAC,CAAC,CAAC;;;;;EAA4F;AACzI,cAAM,iBAAiB,CAAC,CAAC;IACrB,KAAK,IAAI,UAAU,CAAC,OAAO,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IAChD,MAAM,IAAI,UAAU,CAAC,OAAO,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC;IAClD,KAAK,IAAI,UAAU,CAAC,OAAO,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;IAChD,QAAQ;IACR,OAAO;CACV;AAED,UAAU,qBAAqB;IAC3B,KAAK,CAAC,EAAE,OAAO,EAAE,OAAO,QAAQ,EAAE,2BAA2B,CAAC,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,QAAQ,EAAE,eAAe,CAAC,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,EAAE,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG;QAAE,UAAU,CAAC,EAAE,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAA;KAAE,GAAG,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;IAC5X,CAAC,CAAC,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,GAAG,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;IAC3H,YAAY,CAAC,EAAE,UAAU,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC;CACjE;AACD,QAAA,MAAM,SAAS,EAAE,qBAAmC,CAAC;AACnC,KAAK,SAAS,CAAC,CAAC,IAAI,YAAY,CAAC,OAAO,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;AACxD,eAAe,SAAS,CAAC"}
@@ -0,0 +1,48 @@
1
+ /**
2
+ * Golden test for DataTable (Sweep L4, #1423).
3
+ *
4
+ * Covers the semantic table structure (caption → accessible name, column
5
+ * headers, cells, row count), the empty state, sortable-header interaction
6
+ * (aria-sort transition), and axe-cleanliness.
7
+ */
8
+ import { render, screen } from '@testing-library/svelte';
9
+ import userEvent from '@testing-library/user-event';
10
+ import { describe, expect, it } from 'vitest';
11
+ import { expectNoA11yViolations } from '../../../test-support/a11y';
12
+ import DataTable from '../DataTable.svelte';
13
+ const columns = [
14
+ { id: 'name', label: 'Name', accessor: 'name', sortable: true },
15
+ { id: 'age', label: 'Age', accessor: 'age' },
16
+ ];
17
+ const data = [
18
+ { name: 'Ada', age: 36 },
19
+ { name: 'Linus', age: 54 },
20
+ ];
21
+ describe('DataTable', () => {
22
+ it('renders caption, headers, cells, and a row per datum', () => {
23
+ render(DataTable, { props: { data, columns, caption: 'People' } });
24
+ expect(screen.getByRole('table', { name: 'People' })).toBeInTheDocument();
25
+ expect(screen.getByRole('columnheader', { name: 'Name' })).toBeInTheDocument();
26
+ expect(screen.getByRole('cell', { name: 'Ada' })).toBeInTheDocument();
27
+ // header row + one row per datum
28
+ expect(screen.getAllByRole('row')).toHaveLength(data.length + 1);
29
+ });
30
+ it('renders the empty state when there is no data', () => {
31
+ render(DataTable, { props: { data: [], columns } });
32
+ expect(screen.getByText('No data available')).toBeInTheDocument();
33
+ });
34
+ it('toggles aria-sort when a sortable header is activated', async () => {
35
+ render(DataTable, { props: { data, columns, sortable: true } });
36
+ const nameHeader = screen.getByRole('columnheader', { name: 'Name' });
37
+ // aria-sort is only present on the actively-sorted column.
38
+ expect(nameHeader).not.toHaveAttribute('aria-sort');
39
+ await userEvent.click(screen.getByRole('button', { name: 'Name' }));
40
+ expect(nameHeader).toHaveAttribute('aria-sort', 'ascending');
41
+ });
42
+ it('is axe-clean', async () => {
43
+ const { container } = render(DataTable, {
44
+ props: { data, columns, caption: 'People' },
45
+ });
46
+ await expectNoA11yViolations(container);
47
+ });
48
+ });
@@ -0,0 +1,36 @@
1
+ /**
2
+ * Unit tests for DataTable sort/accessor helpers (coverage uplift, S6 gate).
3
+ */
4
+ import { describe, expect, it } from 'vitest';
5
+ import { defaultSort, getNestedValue } from '../types';
6
+ describe('getNestedValue', () => {
7
+ it('reads dot-notation paths', () => {
8
+ expect(getNestedValue({ a: { b: 5 } }, 'a.b')).toBe(5);
9
+ expect(getNestedValue({ a: 1 }, 'a')).toBe(1);
10
+ });
11
+ it('returns undefined for missing paths / non-objects', () => {
12
+ expect(getNestedValue({ a: 1 }, 'x.y')).toBeUndefined();
13
+ expect(getNestedValue(null, 'a')).toBeUndefined();
14
+ });
15
+ });
16
+ describe('defaultSort', () => {
17
+ it('returns 0 when direction is null', () => {
18
+ expect(defaultSort({ a: 1 }, { a: 2 }, 'a', null)).toBe(0);
19
+ });
20
+ it('sorts numbers by direction', () => {
21
+ expect(defaultSort({ a: 1 }, { a: 2 }, 'a', 'asc')).toBeLessThan(0);
22
+ expect(defaultSort({ a: 1 }, { a: 2 }, 'a', 'desc')).toBeGreaterThan(0);
23
+ });
24
+ it('sorts strings and dates', () => {
25
+ expect(defaultSort({ a: 'apple' }, { a: 'banana' }, 'a', 'asc')).toBeLessThan(0);
26
+ expect(defaultSort({ a: new Date(1) }, { a: new Date(2) }, 'a', 'asc')).toBeLessThan(0);
27
+ });
28
+ it('handles nullish values', () => {
29
+ expect(defaultSort({ a: null }, { a: 1 }, 'a', 'asc')).toBeGreaterThan(0);
30
+ expect(defaultSort({ a: 1 }, { a: null }, 'a', 'asc')).toBeLessThan(0);
31
+ expect(defaultSort({ a: null }, { a: null }, 'a', 'asc')).toBe(0);
32
+ });
33
+ it('falls back to string compare for mixed types', () => {
34
+ expect(typeof defaultSort({ a: true }, { a: false }, 'a', 'asc')).toBe('number');
35
+ });
36
+ });
@@ -0,0 +1,6 @@
1
+ /**
2
+ * Data display components
3
+ */
4
+ export { default as DataTable } from './DataTable.svelte';
5
+ export * from './types.js';
6
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/components/data/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,IAAI,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAC1D,cAAc,YAAY,CAAC"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Data display components
3
+ */
4
+ export { default as DataTable } from './DataTable.svelte';
5
+ export * from './types.js';