@exxatdesignux/ui 0.2.19 → 0.3.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 (688) hide show
  1. package/CHANGELOG.md +60 -7
  2. package/bin/sync-extras.mjs +116 -29
  3. package/consumer-extras/README.md +42 -7
  4. package/consumer-extras/cursor-rules/exxat-accessibility.mdc +39 -0
  5. package/consumer-extras/cursor-rules/exxat-board-cards.mdc +26 -0
  6. package/consumer-extras/cursor-rules/exxat-breadcrumbs-no-back.mdc +21 -0
  7. package/consumer-extras/cursor-rules/exxat-card-vs-list-rows.mdc +21 -0
  8. package/consumer-extras/cursor-rules/exxat-centralized-list-dataset.mdc +44 -0
  9. package/consumer-extras/cursor-rules/exxat-collaboration-access.mdc +32 -0
  10. package/consumer-extras/cursor-rules/exxat-command-menu.mdc +22 -0
  11. package/consumer-extras/cursor-rules/exxat-dashboard-view-charts.mdc +53 -0
  12. package/consumer-extras/cursor-rules/exxat-data-tables.mdc +41 -0
  13. package/consumer-extras/cursor-rules/exxat-dedicated-search-surfaces.mdc +25 -0
  14. package/consumer-extras/cursor-rules/exxat-drawer-vs-dialog.mdc +22 -0
  15. package/consumer-extras/cursor-rules/exxat-ds-agents.mdc +56 -0
  16. package/consumer-extras/cursor-rules/exxat-fontawesome-icons.mdc +31 -0
  17. package/consumer-extras/cursor-rules/exxat-kbd-shortcuts.mdc +100 -0
  18. package/consumer-extras/cursor-rules/exxat-kpi-flat-band.mdc +28 -0
  19. package/consumer-extras/cursor-rules/exxat-kpi-max-four.mdc +21 -0
  20. package/consumer-extras/cursor-rules/exxat-kpi-trends.mdc +31 -0
  21. package/consumer-extras/cursor-rules/exxat-list-page-connected-views.mdc +24 -0
  22. package/consumer-extras/cursor-rules/exxat-list-page-view-shells.mdc +31 -0
  23. package/consumer-extras/cursor-rules/exxat-mono-ids.mdc +30 -0
  24. package/consumer-extras/cursor-rules/exxat-no-slds-leakage.mdc +78 -0
  25. package/consumer-extras/cursor-rules/exxat-no-toast.mdc +25 -0
  26. package/consumer-extras/cursor-rules/exxat-page-vs-drawer.mdc +23 -0
  27. package/consumer-extras/cursor-rules/exxat-person-identity-display.mdc +47 -0
  28. package/consumer-extras/cursor-rules/exxat-primary-nav-secondary-panel.mdc +52 -0
  29. package/consumer-extras/cursor-rules/exxat-question-bank-hub-header.mdc +28 -0
  30. package/consumer-extras/cursor-rules/exxat-reuse-before-custom.mdc +34 -0
  31. package/consumer-extras/cursor-rules/exxat-table-properties-drawer.mdc +77 -0
  32. package/consumer-extras/cursor-rules/exxat-token-discipline.mdc +103 -0
  33. package/consumer-extras/cursor-skills/exxat-accessibility/SKILL.md +1 -1
  34. package/consumer-extras/cursor-skills/exxat-board-cards/SKILL.md +2 -2
  35. package/consumer-extras/cursor-skills/exxat-centralized-list-dataset/SKILL.md +4 -15
  36. package/consumer-extras/cursor-skills/exxat-ds-skill/SKILL.md +13 -28
  37. package/consumer-extras/cursor-skills/exxat-ds-skill/references/data-table-pattern.md +1 -1
  38. package/consumer-extras/cursor-skills/exxat-primary-nav-secondary-panel/SKILL.md +2 -4
  39. package/consumer-extras/handbook/HANDBOOK.md +185 -0
  40. package/consumer-extras/handbook/glossary.md +57 -0
  41. package/consumer-extras/handbook/reference-implementations.md +126 -0
  42. package/consumer-extras/handbook/voice-and-tone.md +262 -0
  43. package/consumer-extras/patterns/command-menu-pattern.md +1 -1
  44. package/consumer-extras/patterns/consumer-upgrade-checklist.md +0 -20
  45. package/consumer-extras/patterns/data-views-pattern.md +17 -54
  46. package/consumer-extras/patterns/shell-surface-elevation-pattern.md +3 -5
  47. package/dist/components/data-table/filter-date-calendar.d.ts +10 -0
  48. package/dist/components/data-table/filter-date-calendar.js +280 -0
  49. package/dist/components/data-table/filter-date-calendar.js.map +1 -0
  50. package/dist/components/data-table/filter-text-value-input.d.ts +15 -0
  51. package/dist/components/data-table/filter-text-value-input.js +561 -0
  52. package/dist/components/data-table/filter-text-value-input.js.map +1 -0
  53. package/dist/components/data-table/index.d.ts +45 -0
  54. package/dist/components/data-table/index.js +3085 -0
  55. package/dist/components/data-table/index.js.map +1 -0
  56. package/dist/components/data-table/pagination.d.ts +28 -0
  57. package/dist/components/data-table/pagination.js +3264 -0
  58. package/dist/components/data-table/pagination.js.map +1 -0
  59. package/dist/components/data-table/types.d.ts +84 -0
  60. package/dist/components/data-table/types.js +3 -0
  61. package/dist/components/data-table/types.js.map +1 -0
  62. package/dist/components/data-table/use-table-state.d.ts +116 -0
  63. package/dist/components/data-table/use-table-state.js +670 -0
  64. package/dist/components/data-table/use-table-state.js.map +1 -0
  65. package/dist/components/data-views/board-card-primitives.d.ts +22 -0
  66. package/dist/components/data-views/board-card-primitives.js +84 -0
  67. package/dist/components/data-views/board-card-primitives.js.map +1 -0
  68. package/dist/components/data-views/data-row-list.d.ts +33 -0
  69. package/dist/components/data-views/data-row-list.js +106 -0
  70. package/dist/components/data-views/data-row-list.js.map +1 -0
  71. package/dist/components/data-views/finder-panel-view.d.ts +54 -0
  72. package/dist/components/data-views/finder-panel-view.js +388 -0
  73. package/dist/components/data-views/finder-panel-view.js.map +1 -0
  74. package/dist/components/data-views/folder-grid-view.d.ts +22 -0
  75. package/dist/components/data-views/folder-grid-view.js +58 -0
  76. package/dist/components/data-views/folder-grid-view.js.map +1 -0
  77. package/dist/components/data-views/hub-table.d.ts +167 -0
  78. package/dist/components/data-views/hub-table.js +5561 -0
  79. package/dist/components/data-views/hub-table.js.map +1 -0
  80. package/dist/components/data-views/index.d.ts +27 -0
  81. package/dist/components/data-views/index.js +6575 -0
  82. package/dist/components/data-views/index.js.map +1 -0
  83. package/dist/components/data-views/list-page-board-card.d.ts +72 -0
  84. package/dist/components/data-views/list-page-board-card.js +264 -0
  85. package/dist/components/data-views/list-page-board-card.js.map +1 -0
  86. package/dist/components/data-views/list-page-board-template.d.ts +24 -0
  87. package/dist/components/data-views/list-page-board-template.js +137 -0
  88. package/dist/components/data-views/list-page-board-template.js.map +1 -0
  89. package/dist/components/data-views/list-page-connected-view-body.d.ts +19 -0
  90. package/dist/components/data-views/list-page-connected-view-body.js +116 -0
  91. package/dist/components/data-views/list-page-connected-view-body.js.map +1 -0
  92. package/dist/components/data-views/list-page-split-details-placeholder.d.ts +14 -0
  93. package/dist/components/data-views/list-page-split-details-placeholder.js +38 -0
  94. package/dist/components/data-views/list-page-split-details-placeholder.js.map +1 -0
  95. package/dist/components/data-views/list-page-split-hub-chrome.d.ts +17 -0
  96. package/dist/components/data-views/list-page-split-hub-chrome.js +54 -0
  97. package/dist/components/data-views/list-page-split-hub-chrome.js.map +1 -0
  98. package/dist/components/data-views/list-page-split-hub-tokens.d.ts +12 -0
  99. package/dist/components/data-views/list-page-split-hub-tokens.js +8 -0
  100. package/dist/components/data-views/list-page-split-hub-tokens.js.map +1 -0
  101. package/dist/components/data-views/list-page-tree-column-header.d.ts +15 -0
  102. package/dist/components/data-views/list-page-tree-column-header.js +22 -0
  103. package/dist/components/data-views/list-page-tree-column-header.js.map +1 -0
  104. package/dist/components/data-views/list-page-tree-panel-shell.d.ts +25 -0
  105. package/dist/components/data-views/list-page-tree-panel-shell.js +146 -0
  106. package/dist/components/data-views/list-page-tree-panel-shell.js.map +1 -0
  107. package/dist/components/data-views/os-folder-glyph.d.ts +35 -0
  108. package/dist/components/data-views/os-folder-glyph.js +104 -0
  109. package/dist/components/data-views/os-folder-glyph.js.map +1 -0
  110. package/dist/components/data-views/outline-tree-menu.d.ts +36 -0
  111. package/dist/components/data-views/outline-tree-menu.js +131 -0
  112. package/dist/components/data-views/outline-tree-menu.js.map +1 -0
  113. package/dist/components/table-properties/column-row.d.ts +22 -0
  114. package/dist/components/table-properties/column-row.js +153 -0
  115. package/dist/components/table-properties/column-row.js.map +1 -0
  116. package/dist/components/table-properties/draggable-list.d.ts +24 -0
  117. package/dist/components/table-properties/draggable-list.js +53 -0
  118. package/dist/components/table-properties/draggable-list.js.map +1 -0
  119. package/dist/components/table-properties/drawer-button.d.ts +110 -0
  120. package/dist/components/table-properties/drawer-button.js +2748 -0
  121. package/dist/components/table-properties/drawer-button.js.map +1 -0
  122. package/dist/components/table-properties/drawer.d.ts +100 -0
  123. package/dist/components/table-properties/drawer.js +2595 -0
  124. package/dist/components/table-properties/drawer.js.map +1 -0
  125. package/dist/components/table-properties/filter-card.d.ts +24 -0
  126. package/dist/components/table-properties/filter-card.js +854 -0
  127. package/dist/components/table-properties/filter-card.js.map +1 -0
  128. package/dist/components/table-properties/index.d.ts +14 -0
  129. package/dist/components/table-properties/index.js +2768 -0
  130. package/dist/components/table-properties/index.js.map +1 -0
  131. package/dist/components/table-properties/sort-card.d.ts +20 -0
  132. package/dist/components/table-properties/sort-card.js +102 -0
  133. package/dist/components/table-properties/sort-card.js.map +1 -0
  134. package/dist/components/templates/dedicated-search-landing-template.d.ts +21 -0
  135. package/dist/components/templates/dedicated-search-landing-template.js +254 -0
  136. package/dist/components/templates/dedicated-search-landing-template.js.map +1 -0
  137. package/dist/components/templates/dedicated-search-results-template.d.ts +15 -0
  138. package/dist/components/templates/dedicated-search-results-template.js +16 -0
  139. package/dist/components/templates/dedicated-search-results-template.js.map +1 -0
  140. package/dist/components/templates/index.d.ts +9 -0
  141. package/dist/components/templates/index.js +2720 -0
  142. package/dist/components/templates/index.js.map +1 -0
  143. package/dist/components/templates/list-page.d.ts +83 -0
  144. package/dist/components/templates/list-page.js +2433 -0
  145. package/dist/components/templates/list-page.js.map +1 -0
  146. package/dist/components/templates/nested-secondary-panel-shell.d.ts +20 -0
  147. package/dist/components/templates/nested-secondary-panel-shell.js +54 -0
  148. package/dist/components/templates/nested-secondary-panel-shell.js.map +1 -0
  149. package/dist/components/ui/accordion.d.ts +10 -0
  150. package/dist/components/ui/accordion.js +74 -0
  151. package/dist/components/ui/accordion.js.map +1 -0
  152. package/dist/components/ui/alert-dialog.d.ts +37 -0
  153. package/dist/components/ui/alert-dialog.js +201 -0
  154. package/dist/components/ui/alert-dialog.js.map +1 -0
  155. package/dist/components/ui/avatar.d.ts +84 -0
  156. package/dist/components/ui/avatar.js +328 -0
  157. package/dist/components/ui/avatar.js.map +1 -0
  158. package/dist/components/ui/badge.d.ts +13 -0
  159. package/dist/components/ui/badge.js +49 -0
  160. package/dist/components/ui/badge.js.map +1 -0
  161. package/dist/components/ui/banner.d.ts +62 -0
  162. package/dist/components/ui/banner.js +364 -0
  163. package/dist/components/ui/banner.js.map +1 -0
  164. package/dist/components/ui/breadcrumb.d.ts +14 -0
  165. package/dist/components/ui/breadcrumb.js +114 -0
  166. package/dist/components/ui/breadcrumb.js.map +1 -0
  167. package/dist/components/ui/button.d.ts +16 -0
  168. package/dist/components/ui/button.js +59 -0
  169. package/dist/components/ui/button.js.map +1 -0
  170. package/dist/components/ui/calendar.d.ts +13 -0
  171. package/dist/components/ui/calendar.js +238 -0
  172. package/dist/components/ui/calendar.js.map +1 -0
  173. package/dist/components/ui/card.d.ts +14 -0
  174. package/dist/components/ui/card.js +102 -0
  175. package/dist/components/ui/card.js.map +1 -0
  176. package/dist/components/ui/chart.d.ts +58 -0
  177. package/dist/components/ui/chart.js +292 -0
  178. package/dist/components/ui/chart.js.map +1 -0
  179. package/dist/components/ui/checkbox.d.ts +23 -0
  180. package/dist/components/ui/checkbox.js +155 -0
  181. package/dist/components/ui/checkbox.js.map +1 -0
  182. package/dist/components/ui/coach-mark.d.ts +27 -0
  183. package/dist/components/ui/coach-mark.js +306 -0
  184. package/dist/components/ui/coach-mark.js.map +1 -0
  185. package/dist/components/ui/collapsible.d.ts +8 -0
  186. package/dist/components/ui/collapsible.js +35 -0
  187. package/dist/components/ui/collapsible.js.map +1 -0
  188. package/dist/components/ui/command.d.ts +36 -0
  189. package/dist/components/ui/command.js +274 -0
  190. package/dist/components/ui/command.js.map +1 -0
  191. package/dist/components/ui/context-menu.d.ts +32 -0
  192. package/dist/components/ui/context-menu.js +245 -0
  193. package/dist/components/ui/context-menu.js.map +1 -0
  194. package/dist/components/ui/date-picker-field.d.ts +38 -0
  195. package/dist/components/ui/date-picker-field.js +550 -0
  196. package/dist/components/ui/date-picker-field.js.map +1 -0
  197. package/dist/components/ui/dialog.d.ts +22 -0
  198. package/dist/components/ui/dialog.js +200 -0
  199. package/dist/components/ui/dialog.js.map +1 -0
  200. package/dist/components/ui/dot-pattern.d.ts +21 -0
  201. package/dist/components/ui/dot-pattern.js +139 -0
  202. package/dist/components/ui/dot-pattern.js.map +1 -0
  203. package/dist/components/ui/drag-handle-grip.d.ts +10 -0
  204. package/dist/components/ui/drag-handle-grip.js +15 -0
  205. package/dist/components/ui/drag-handle-grip.js.map +1 -0
  206. package/dist/components/ui/drawer.d.ts +16 -0
  207. package/dist/components/ui/drawer.js +125 -0
  208. package/dist/components/ui/drawer.js.map +1 -0
  209. package/dist/components/ui/dropdown-menu.d.ts +45 -0
  210. package/dist/components/ui/dropdown-menu.js +353 -0
  211. package/dist/components/ui/dropdown-menu.js.map +1 -0
  212. package/dist/components/ui/export-drawer.d.ts +11 -0
  213. package/dist/components/ui/export-drawer.js +1658 -0
  214. package/dist/components/ui/export-drawer.js.map +1 -0
  215. package/dist/components/ui/field.d.ts +30 -0
  216. package/dist/components/ui/field.js +249 -0
  217. package/dist/components/ui/field.js.map +1 -0
  218. package/dist/components/ui/form.d.ts +28 -0
  219. package/dist/components/ui/form.js +110 -0
  220. package/dist/components/ui/form.js.map +1 -0
  221. package/dist/components/ui/hover-card.d.ts +9 -0
  222. package/dist/components/ui/hover-card.js +43 -0
  223. package/dist/components/ui/hover-card.js.map +1 -0
  224. package/dist/components/ui/input-group.d.ts +20 -0
  225. package/dist/components/ui/input-group.js +219 -0
  226. package/dist/components/ui/input-group.js.map +1 -0
  227. package/dist/components/ui/input-mask.d.ts +39 -0
  228. package/dist/components/ui/input-mask.js +118 -0
  229. package/dist/components/ui/input-mask.js.map +1 -0
  230. package/dist/components/ui/input.d.ts +5 -0
  231. package/dist/components/ui/input.js +30 -0
  232. package/dist/components/ui/input.js.map +1 -0
  233. package/dist/components/ui/kbd.d.ts +20 -0
  234. package/dist/components/ui/kbd.js +45 -0
  235. package/dist/components/ui/kbd.js.map +1 -0
  236. package/dist/components/ui/key-metrics-context.d.ts +19 -0
  237. package/dist/components/ui/key-metrics-context.js +26 -0
  238. package/dist/components/ui/key-metrics-context.js.map +1 -0
  239. package/dist/components/ui/key-metrics.d.ts +131 -0
  240. package/dist/components/ui/key-metrics.js +1015 -0
  241. package/dist/components/ui/key-metrics.js.map +1 -0
  242. package/dist/components/ui/label.d.ts +6 -0
  243. package/dist/components/ui/label.js +28 -0
  244. package/dist/components/ui/label.js.map +1 -0
  245. package/dist/components/ui/list-page-view-frame.d.ts +22 -0
  246. package/dist/components/ui/list-page-view-frame.js +24 -0
  247. package/dist/components/ui/list-page-view-frame.js.map +1 -0
  248. package/dist/components/ui/page-header.d.ts +51 -0
  249. package/dist/components/ui/page-header.js +372 -0
  250. package/dist/components/ui/page-header.js.map +1 -0
  251. package/dist/components/ui/payment-card-fields.d.ts +10 -0
  252. package/dist/components/ui/payment-card-fields.js +80 -0
  253. package/dist/components/ui/payment-card-fields.js.map +1 -0
  254. package/dist/components/ui/popover.d.ts +10 -0
  255. package/dist/components/ui/popover.js +47 -0
  256. package/dist/components/ui/popover.js.map +1 -0
  257. package/dist/components/ui/radio-group.d.ts +29 -0
  258. package/dist/components/ui/radio-group.js +190 -0
  259. package/dist/components/ui/radio-group.js.map +1 -0
  260. package/dist/components/ui/resizable.d.ts +16 -0
  261. package/dist/components/ui/resizable.js +51 -0
  262. package/dist/components/ui/resizable.js.map +1 -0
  263. package/dist/components/ui/scroll-area.d.ts +8 -0
  264. package/dist/components/ui/scroll-area.js +66 -0
  265. package/dist/components/ui/scroll-area.js.map +1 -0
  266. package/dist/components/ui/select.d.ts +18 -0
  267. package/dist/components/ui/select.js +186 -0
  268. package/dist/components/ui/select.js.map +1 -0
  269. package/dist/components/ui/selection-tile-grid.d.ts +52 -0
  270. package/dist/components/ui/selection-tile-grid.js +347 -0
  271. package/dist/components/ui/selection-tile-grid.js.map +1 -0
  272. package/dist/components/ui/separator.d.ts +7 -0
  273. package/dist/components/ui/separator.js +33 -0
  274. package/dist/components/ui/separator.js.map +1 -0
  275. package/dist/components/ui/sheet.d.ts +18 -0
  276. package/dist/components/ui/sheet.js +181 -0
  277. package/dist/components/ui/sheet.js.map +1 -0
  278. package/dist/components/ui/sidebar.d.ts +94 -0
  279. package/dist/components/ui/sidebar.js +805 -0
  280. package/dist/components/ui/sidebar.js.map +1 -0
  281. package/dist/components/ui/skeleton.d.ts +5 -0
  282. package/dist/components/ui/skeleton.js +22 -0
  283. package/dist/components/ui/skeleton.js.map +1 -0
  284. package/dist/components/ui/slider.d.ts +7 -0
  285. package/dist/components/ui/slider.js +66 -0
  286. package/dist/components/ui/slider.js.map +1 -0
  287. package/dist/components/ui/sonner.d.ts +6 -0
  288. package/dist/components/ui/sonner.js +38 -0
  289. package/dist/components/ui/sonner.js.map +1 -0
  290. package/dist/components/ui/status-badge.d.ts +38 -0
  291. package/dist/components/ui/status-badge.js +77 -0
  292. package/dist/components/ui/status-badge.js.map +1 -0
  293. package/dist/components/ui/table.d.ts +13 -0
  294. package/dist/components/ui/table.js +115 -0
  295. package/dist/components/ui/table.js.map +1 -0
  296. package/dist/components/ui/tabs.d.ts +15 -0
  297. package/dist/components/ui/tabs.js +93 -0
  298. package/dist/components/ui/tabs.js.map +1 -0
  299. package/dist/components/ui/textarea.d.ts +6 -0
  300. package/dist/components/ui/textarea.js +25 -0
  301. package/dist/components/ui/textarea.js.map +1 -0
  302. package/dist/components/ui/tip.d.ts +12 -0
  303. package/dist/components/ui/tip.js +61 -0
  304. package/dist/components/ui/tip.js.map +1 -0
  305. package/dist/components/ui/toggle-group.d.ts +14 -0
  306. package/dist/components/ui/toggle-group.js +104 -0
  307. package/dist/components/ui/toggle-group.js.map +1 -0
  308. package/dist/components/ui/toggle-switch.d.ts +10 -0
  309. package/dist/components/ui/toggle-switch.js +33 -0
  310. package/dist/components/ui/toggle-switch.js.map +1 -0
  311. package/dist/components/ui/toggle.d.ts +13 -0
  312. package/dist/components/ui/toggle.js +51 -0
  313. package/dist/components/ui/toggle.js.map +1 -0
  314. package/dist/components/ui/tooltip.d.ts +10 -0
  315. package/dist/components/ui/tooltip.js +68 -0
  316. package/dist/components/ui/tooltip.js.map +1 -0
  317. package/dist/components/ui/view-segmented-control.d.ts +31 -0
  318. package/dist/components/ui/view-segmented-control.js +167 -0
  319. package/dist/components/ui/view-segmented-control.js.map +1 -0
  320. package/dist/data-list-view-registry-CyBoBML4.d.ts +73 -0
  321. package/dist/hooks/use-app-theme.d.ts +24 -0
  322. package/dist/hooks/use-app-theme.js +286 -0
  323. package/dist/hooks/use-app-theme.js.map +1 -0
  324. package/dist/hooks/use-coach-mark.d.ts +86 -0
  325. package/dist/hooks/use-coach-mark.js +218 -0
  326. package/dist/hooks/use-coach-mark.js.map +1 -0
  327. package/dist/hooks/use-mobile.d.ts +3 -0
  328. package/dist/hooks/use-mobile.js +29 -0
  329. package/dist/hooks/use-mobile.js.map +1 -0
  330. package/dist/hooks/use-mod-key-label.d.ts +6 -0
  331. package/dist/hooks/use-mod-key-label.js +25 -0
  332. package/dist/hooks/use-mod-key-label.js.map +1 -0
  333. package/dist/index.d.ts +120 -0
  334. package/dist/index.js +13324 -0
  335. package/dist/index.js.map +1 -0
  336. package/dist/lib/compose-refs.d.ts +6 -0
  337. package/dist/lib/compose-refs.js +17 -0
  338. package/dist/lib/compose-refs.js.map +1 -0
  339. package/dist/lib/conditional-rule-match.d.ts +30 -0
  340. package/dist/lib/conditional-rule-match.js +66 -0
  341. package/dist/lib/conditional-rule-match.js.map +1 -0
  342. package/dist/lib/data-list-display-options.d.ts +26 -0
  343. package/dist/lib/data-list-display-options.js +14 -0
  344. package/dist/lib/data-list-display-options.js.map +1 -0
  345. package/dist/lib/data-list-view-registry.d.ts +2 -0
  346. package/dist/lib/data-list-view-registry.js +102 -0
  347. package/dist/lib/data-list-view-registry.js.map +1 -0
  348. package/dist/lib/data-list-view-surface.d.ts +2 -0
  349. package/dist/lib/data-list-view-surface.js +80 -0
  350. package/dist/lib/data-list-view-surface.js.map +1 -0
  351. package/dist/lib/data-list-view.d.ts +21 -0
  352. package/dist/lib/data-list-view.js +25 -0
  353. package/dist/lib/data-list-view.js.map +1 -0
  354. package/dist/lib/date-filter.d.ts +22 -0
  355. package/dist/lib/date-filter.js +61 -0
  356. package/dist/lib/date-filter.js.map +1 -0
  357. package/dist/lib/dev-log.d.ts +8 -0
  358. package/dist/lib/dev-log.js +10 -0
  359. package/dist/lib/dev-log.js.map +1 -0
  360. package/dist/lib/dropdown-menu-surface.d.ts +14 -0
  361. package/dist/lib/dropdown-menu-surface.js +6 -0
  362. package/dist/lib/dropdown-menu-surface.js.map +1 -0
  363. package/dist/lib/editable-target.d.ts +12 -0
  364. package/dist/lib/editable-target.js +12 -0
  365. package/dist/lib/editable-target.js.map +1 -0
  366. package/dist/lib/list-page-table-properties.d.ts +35 -0
  367. package/dist/lib/list-page-table-properties.js +81 -0
  368. package/dist/lib/list-page-table-properties.js.map +1 -0
  369. package/dist/lib/raf-throttle.d.ts +23 -0
  370. package/dist/lib/raf-throttle.js +27 -0
  371. package/dist/lib/raf-throttle.js.map +1 -0
  372. package/dist/lib/row-height.d.ts +16 -0
  373. package/dist/lib/row-height.js +10 -0
  374. package/dist/lib/row-height.js.map +1 -0
  375. package/dist/lib/table-properties-types.d.ts +83 -0
  376. package/dist/lib/table-properties-types.js +19 -0
  377. package/dist/lib/table-properties-types.js.map +1 -0
  378. package/dist/lib/utils.d.ts +5 -0
  379. package/dist/lib/utils.js +11 -0
  380. package/dist/lib/utils.js.map +1 -0
  381. package/package.json +83 -19
  382. package/src/components/data-table/filter-date-calendar.tsx +38 -0
  383. package/src/components/data-table/filter-text-value-input.tsx +77 -0
  384. package/src/components/data-table/index.tsx +1678 -0
  385. package/src/components/data-table/pagination.tsx +255 -0
  386. package/src/components/data-table/types.ts +96 -0
  387. package/src/components/data-table/use-table-state.ts +767 -0
  388. package/src/components/data-views/board-card-primitives.tsx +93 -0
  389. package/src/components/data-views/data-row-list.tsx +183 -0
  390. package/src/components/data-views/finder-panel-view.tsx +405 -0
  391. package/src/components/data-views/folder-grid-view.tsx +86 -0
  392. package/src/components/data-views/hub-table.tsx +498 -0
  393. package/src/components/data-views/index.ts +28 -0
  394. package/src/components/data-views/list-page-board-card.tsx +192 -0
  395. package/src/components/data-views/list-page-board-template.tsx +122 -0
  396. package/src/components/data-views/list-page-connected-view-body.tsx +66 -0
  397. package/src/components/data-views/list-page-split-details-placeholder.tsx +39 -0
  398. package/src/components/data-views/list-page-split-hub-chrome.tsx +60 -0
  399. package/src/components/data-views/list-page-split-hub-tokens.ts +16 -0
  400. package/src/components/data-views/list-page-tree-column-header.tsx +31 -0
  401. package/src/components/data-views/list-page-tree-panel-shell.tsx +91 -0
  402. package/src/components/data-views/os-folder-glyph.tsx +141 -0
  403. package/src/components/data-views/outline-tree-menu.tsx +157 -0
  404. package/src/components/table-properties/column-row.tsx +90 -0
  405. package/src/components/table-properties/draggable-list.ts +54 -0
  406. package/src/components/table-properties/drawer-button.tsx +300 -0
  407. package/src/components/table-properties/drawer.tsx +1148 -0
  408. package/src/components/table-properties/filter-card.tsx +251 -0
  409. package/src/components/table-properties/index.ts +36 -0
  410. package/src/components/table-properties/sort-card.tsx +63 -0
  411. package/src/components/templates/dedicated-search-landing-template.tsx +124 -0
  412. package/src/components/templates/dedicated-search-results-template.tsx +19 -0
  413. package/src/components/templates/index.ts +33 -0
  414. package/src/components/templates/list-page.tsx +602 -0
  415. package/src/components/templates/nested-secondary-panel-shell.tsx +70 -0
  416. package/src/components/ui/accordion.tsx +92 -0
  417. package/src/components/ui/alert-dialog.tsx +221 -0
  418. package/src/components/ui/avatar.tsx +13 -2
  419. package/src/components/ui/banner.tsx +2 -2
  420. package/src/components/ui/button.tsx +4 -4
  421. package/src/components/ui/calendar.tsx +1 -1
  422. package/src/components/ui/coach-mark.tsx +1 -1
  423. package/src/components/ui/context-menu.tsx +291 -0
  424. package/src/components/ui/date-picker-field.tsx +2 -2
  425. package/src/components/ui/dot-pattern.tsx +183 -0
  426. package/src/components/ui/export-drawer.tsx +375 -0
  427. package/src/components/ui/hover-card.tsx +66 -0
  428. package/src/components/ui/key-metrics-context.tsx +78 -0
  429. package/src/components/ui/key-metrics.tsx +1133 -0
  430. package/src/components/ui/list-page-view-frame.tsx +64 -0
  431. package/src/components/ui/page-header.tsx +244 -0
  432. package/src/components/ui/payment-card-fields.tsx +2 -2
  433. package/src/components/ui/resizable.tsx +68 -0
  434. package/src/components/ui/scroll-area.tsx +72 -0
  435. package/src/components/ui/selection-tile-grid.tsx +9 -2
  436. package/src/components/ui/sidebar.tsx +84 -12
  437. package/src/components/ui/slider.tsx +83 -0
  438. package/src/globals.css +2201 -7
  439. package/src/globals.d.ts +20 -0
  440. package/src/index.ts +68 -1
  441. package/src/lib/conditional-rule-match.ts +119 -0
  442. package/src/lib/data-list-display-options.ts +35 -0
  443. package/src/lib/data-list-view-registry.ts +104 -0
  444. package/src/lib/data-list-view-surface.ts +83 -0
  445. package/src/lib/data-list-view.ts +47 -0
  446. package/src/lib/dev-log.ts +10 -0
  447. package/src/lib/editable-target.ts +20 -0
  448. package/src/lib/list-page-table-properties.ts +48 -0
  449. package/src/lib/raf-throttle.ts +45 -0
  450. package/src/lib/row-height.ts +19 -0
  451. package/src/lib/table-properties-types.ts +98 -0
  452. package/template/.cursor/rules/exxat-command-menu.mdc +1 -1
  453. package/template/.cursor/rules/exxat-dashboard-view-charts.mdc +3 -3
  454. package/template/.cursor/rules/exxat-data-tables.mdc +1 -1
  455. package/template/.cursor/rules/exxat-ds-agents.mdc +2 -2
  456. package/template/.cursor/rules/exxat-kbd-shortcuts.mdc +2 -2
  457. package/template/.cursor/rules/exxat-table-properties-drawer.mdc +1 -1
  458. package/template/AGENTS.md +104 -78
  459. package/template/app/(app)/dashboard/loading.tsx +15 -3
  460. package/template/app/(app)/dashboard/page.tsx +14 -2
  461. package/template/app/(app)/examples/page.tsx +0 -2
  462. package/template/app/(app)/layout.tsx +17 -4
  463. package/template/app/(app)/loading.tsx +18 -1
  464. package/template/app/(app)/question-bank/find/page.tsx +1 -2
  465. package/template/app/(app)/question-bank/layout.tsx +1 -1
  466. package/template/app/(app)/question-bank/library/page.tsx +1 -2
  467. package/template/app/(app)/question-bank/list/page.tsx +1 -2
  468. package/template/app/(app)/question-bank/new/page.tsx +15 -20
  469. package/template/app/(app)/question-bank/page.tsx +1 -2
  470. package/template/app/(app)/settings/page.tsx +5 -4
  471. package/template/app/globals.css +14 -16
  472. package/template/components/ask-leo-sidebar.tsx +5 -1
  473. package/template/components/brand-color-picker.tsx +2 -2
  474. package/template/components/charts-overview.tsx +1 -1
  475. package/template/components/compliance-board-view.tsx +142 -0
  476. package/template/components/compliance-client.tsx +92 -0
  477. package/template/components/compliance-page-header.tsx +89 -0
  478. package/template/components/compliance-table.tsx +468 -0
  479. package/template/components/dashboard-report-charts.tsx +1 -1
  480. package/template/components/dashboard-tabs.tsx +1 -1
  481. package/template/components/data-table/filter-date-calendar.tsx +1 -38
  482. package/template/components/data-table/filter-text-value-input.tsx +1 -77
  483. package/template/components/data-table/index.tsx +1 -1634
  484. package/template/components/data-table/pagination.tsx +1 -255
  485. package/template/components/data-table/types.ts +1 -94
  486. package/template/components/data-table/use-table-state.test.ts +420 -0
  487. package/template/components/data-table/use-table-state.ts +1 -758
  488. package/template/components/data-view-dashboard-charts-compliance.tsx +963 -0
  489. package/template/components/data-view-dashboard-charts-team.tsx +971 -0
  490. package/template/components/data-view-dashboard-charts.tsx +1503 -0
  491. package/template/components/data-views/board-card-primitives.tsx +1 -93
  492. package/template/components/data-views/data-row-list.tsx +1 -183
  493. package/template/components/data-views/finder-panel-view.tsx +1 -405
  494. package/template/components/data-views/folder-grid-view.tsx +1 -86
  495. package/template/components/data-views/hub-table.tsx +1 -0
  496. package/template/components/data-views/index.ts +50 -37
  497. package/template/components/data-views/list-page-board-card.tsx +1 -192
  498. package/template/components/data-views/list-page-board-template.tsx +1 -122
  499. package/template/components/data-views/list-page-connected-view-body.tsx +1 -66
  500. package/template/components/data-views/list-page-split-details-placeholder.tsx +1 -39
  501. package/template/components/data-views/list-page-split-hub-chrome.tsx +1 -68
  502. package/template/components/data-views/list-page-split-hub-tokens.ts +1 -16
  503. package/template/components/data-views/list-page-tree-column-header.tsx +1 -31
  504. package/template/components/data-views/list-page-tree-panel-shell.tsx +1 -91
  505. package/template/components/data-views/list-page-view-frame.tsx +5 -53
  506. package/template/components/data-views/os-folder-glyph.tsx +1 -129
  507. package/template/components/data-views/outline-tree-menu.tsx +1 -157
  508. package/template/components/export-drawer.test.tsx +71 -0
  509. package/template/components/export-drawer.tsx +1 -375
  510. package/template/components/exxat-product-logo.tsx +5 -5
  511. package/template/components/hub-tree-panel-view.tsx +2 -2
  512. package/template/components/invite-collaborators-drawer.tsx +3 -3
  513. package/template/components/key-metrics-ask-leo-bridge.tsx +40 -0
  514. package/template/components/key-metrics.tsx +1 -1063
  515. package/template/components/leo-insight-indicator.tsx +2 -2
  516. package/template/components/new-placement-back-btn.tsx +28 -0
  517. package/template/components/new-placement-form.tsx +942 -0
  518. package/template/components/new-question-composer.tsx +456 -408
  519. package/template/components/onboarding/index.ts +9 -0
  520. package/template/components/onboarding/onboarding-01.tsx +1 -1
  521. package/template/components/onboarding/onboarding-02.tsx +1 -1
  522. package/template/components/onboarding/onboarding-03.tsx +1 -1
  523. package/template/components/onboarding/onboarding-04.tsx +1 -1
  524. package/template/components/page-header.tsx +8 -226
  525. package/template/components/placement-board-card.tsx +250 -0
  526. package/template/components/placement-detail.tsx +438 -0
  527. package/template/components/placements-board-view.tsx +397 -0
  528. package/template/components/placements-client.tsx +220 -0
  529. package/template/components/placements-list-view.tsx +124 -0
  530. package/template/components/placements-page-header.tsx +166 -0
  531. package/template/components/placements-table-cells.test.tsx +22 -0
  532. package/template/components/placements-table-cells.tsx +173 -0
  533. package/template/components/placements-table-columns.tsx +210 -0
  534. package/template/components/placements-table.tsx +934 -0
  535. package/template/components/product-switcher.tsx +3 -4
  536. package/template/components/product-wordmark.tsx +2 -1
  537. package/template/components/question-bank-client.tsx +5 -5
  538. package/template/components/question-bank-hub-client.tsx +1 -1
  539. package/template/components/question-bank-new-folder-sheet.tsx +2 -2
  540. package/template/components/question-bank-secondary-nav.tsx +3 -3
  541. package/template/components/question-bank-table.tsx +541 -431
  542. package/template/components/rotations-empty-state.tsx +50 -0
  543. package/template/components/rotations-panel-activator.tsx +8 -0
  544. package/template/components/settings-appearance-card.tsx +3 -4
  545. package/template/components/settings-client.tsx +15 -59
  546. package/template/components/settings-form-row.tsx +4 -9
  547. package/template/components/{app-sidebar-dynamic.tsx → sidebar/app-sidebar-dynamic.tsx} +1 -1
  548. package/template/components/{app-sidebar.tsx → sidebar/app-sidebar.tsx} +59 -74
  549. package/template/components/sidebar/index.ts +16 -0
  550. package/template/components/{secondary-nav.tsx → sidebar/secondary-nav.tsx} +2 -2
  551. package/template/components/{secondary-panel.tsx → sidebar/secondary-panel.tsx} +50 -7
  552. package/template/components/{sidebar-auto-collapse.tsx → sidebar/sidebar-auto-collapse.tsx} +6 -2
  553. package/template/components/{sidebar-shell.tsx → sidebar/sidebar-shell.tsx} +1 -1
  554. package/template/components/site-header.tsx +1 -1
  555. package/template/components/sites-board-view.tsx +67 -0
  556. package/template/components/sites-client.tsx +154 -0
  557. package/template/components/sites-table.tsx +249 -0
  558. package/template/components/table-properties/column-row.tsx +1 -90
  559. package/template/components/table-properties/draggable-list.ts +1 -49
  560. package/template/components/table-properties/drawer-button.tsx +1 -262
  561. package/template/components/table-properties/drawer.tsx +1 -1166
  562. package/template/components/table-properties/filter-card.tsx +1 -251
  563. package/template/components/table-properties/sort-card.tsx +1 -59
  564. package/template/components/table-properties/types.ts +28 -71
  565. package/template/components/team-board-view.tsx +122 -0
  566. package/template/components/team-client.tsx +100 -0
  567. package/template/components/team-page-header.tsx +92 -0
  568. package/template/components/team-table.tsx +553 -0
  569. package/template/components/templates/dedicated-search-landing-template.tsx +1 -124
  570. package/template/components/templates/dedicated-search-results-template.tsx +1 -19
  571. package/template/components/templates/list-page.tsx +1 -608
  572. package/template/components/templates/nested-secondary-panel-shell.tsx +1 -63
  573. package/template/components/templates/new-focus-template.tsx +659 -0
  574. package/template/components/templates/secondary-panel-hub-template.tsx +1 -1
  575. package/template/components/ui/accordion.tsx +1 -0
  576. package/template/components/ui/alert-dialog.tsx +1 -0
  577. package/template/components/ui/context-menu.tsx +1 -0
  578. package/template/components/ui/dot-pattern.tsx +1 -183
  579. package/template/components/ui/hover-card.tsx +1 -0
  580. package/template/components/ui/resizable.tsx +1 -68
  581. package/template/components/ui/scroll-area.tsx +1 -0
  582. package/template/components/ui/slider.tsx +1 -0
  583. package/template/docs/blueprints/README.md +86 -0
  584. package/template/docs/blueprints/_template.md +91 -0
  585. package/template/docs/blueprints/board-card.md +123 -0
  586. package/template/docs/blueprints/data-table.md +139 -0
  587. package/template/docs/blueprints/key-metrics.md +128 -0
  588. package/template/docs/blueprints/list-page-template.md +123 -0
  589. package/template/docs/blueprints/page-header.md +130 -0
  590. package/template/docs/command-menu-pattern.md +1 -1
  591. package/template/docs/component-selection-guide.md +224 -0
  592. package/template/docs/components-audit-2026-05.md +158 -0
  593. package/template/docs/data-views-pattern.md +17 -54
  594. package/template/docs/drawer-vs-dialog-pattern.md +1 -3
  595. package/template/docs/migrations/0001-brand-deep-alias-stabilization.md +95 -0
  596. package/template/docs/migrations/0002-exxat-token-namespace.md +154 -0
  597. package/template/docs/migrations/0003-globals-css-canonical.md +110 -0
  598. package/template/docs/migrations/README.md +100 -0
  599. package/template/docs/migrations/_template.md +64 -0
  600. package/template/docs/shell-surface-elevation-pattern.md +3 -5
  601. package/template/docs/token-taxonomy.md +416 -0
  602. package/template/eslint.config.mjs +27 -0
  603. package/template/hooks/use-secondary-panel-hub-nav.ts +1 -1
  604. package/template/lib/command-menu-config.ts +0 -1
  605. package/template/lib/command-menu-search-data.ts +27 -11
  606. package/template/lib/compliance-supported-views.ts +10 -0
  607. package/template/lib/conditional-rule-match.ts +6 -97
  608. package/template/lib/data-list-display-options.ts +1 -49
  609. package/template/lib/data-list-view-registry.ts +1 -104
  610. package/template/lib/data-list-view-surface.ts +1 -83
  611. package/template/lib/data-list-view.ts +1 -47
  612. package/template/lib/data-view-dashboard-placements-layout.ts +215 -0
  613. package/template/lib/data-view-dashboard-storage.ts +35 -38
  614. package/template/lib/dev-log.ts +1 -8
  615. package/template/lib/editable-target.ts +1 -10
  616. package/template/lib/list-page-table-properties.ts +1 -48
  617. package/template/lib/list-status-badges.ts +97 -4
  618. package/template/lib/mock/compliance-kpi.ts +61 -0
  619. package/template/lib/mock/compliance.ts +146 -0
  620. package/template/lib/mock/navigation.tsx +0 -9
  621. package/template/lib/mock/placements-kpi.ts +134 -0
  622. package/template/lib/mock/placements.ts +176 -0
  623. package/template/lib/mock/sites-directory.ts +16 -0
  624. package/template/lib/mock/sites-kpi.ts +25 -0
  625. package/template/lib/mock/team-kpi.ts +60 -0
  626. package/template/lib/mock/team.ts +118 -0
  627. package/template/lib/placement-board-card-layout.ts +79 -0
  628. package/template/lib/placements-supported-views.ts +12 -0
  629. package/template/lib/question-bank-supported-views.ts +0 -1
  630. package/template/lib/raf-throttle.ts +1 -45
  631. package/template/lib/row-height.ts +4 -10
  632. package/template/lib/sidebar-state-cookie.ts +11 -2
  633. package/template/lib/sites-supported-views.ts +10 -0
  634. package/template/lib/table-state-lifecycle.ts +2 -2
  635. package/template/lib/team-supported-views.ts +10 -0
  636. package/template/package.json +1 -0
  637. package/template/tests/setup.ts +25 -0
  638. package/consumer-extras/AGENTS.md +0 -76
  639. package/consumer-extras/cursor-skills/exxat-consumer-app/SKILL.md +0 -37
  640. package/consumer-extras/cursor-skills/exxat-focused-workflow-page/SKILL.md +0 -57
  641. package/consumer-extras/patterns/consumer-app-pattern.md +0 -39
  642. package/consumer-extras/patterns/focused-workflow-page-pattern.md +0 -84
  643. package/src/components/ui/button-group.tsx +0 -81
  644. package/src/theme.css +0 -16
  645. package/src/tokens/README.md +0 -15
  646. package/src/tokens/base.css +0 -337
  647. package/src/tokens/high-contrast.css +0 -1195
  648. package/src/tokens/layers.css +0 -224
  649. package/src/tokens/tailwind-bridge.css +0 -118
  650. package/src/tokens/themes.css +0 -201
  651. package/template/app/(app)/data-list/layout.tsx +0 -43
  652. package/template/app/(app)/data-list/page.tsx +0 -10
  653. package/template/app/(app)/examples/focused-workflow/page.tsx +0 -5
  654. package/template/components/app-route-loading.tsx +0 -14
  655. package/template/components/dashboard-onboarding-gallery.tsx +0 -13
  656. package/template/components/dashboard-onboarding.tsx +0 -21
  657. package/template/components/data-views/list-page-calendar-view.tsx +0 -593
  658. package/template/components/data-views/list-page-folder-columns-panel.tsx +0 -345
  659. package/template/components/examples/focused-workflow-showcase.tsx +0 -183
  660. package/template/components/list-hub-board-view.tsx +0 -68
  661. package/template/components/list-hub-client.tsx +0 -186
  662. package/template/components/list-hub-list-view.tsx +0 -36
  663. package/template/components/list-hub-panel-activator.tsx +0 -8
  664. package/template/components/list-hub-secondary-nav.tsx +0 -121
  665. package/template/components/list-hub-table.tsx +0 -336
  666. package/template/components/question-bank-folder-columns-panel.tsx +0 -104
  667. package/template/components/question-bank-list-view.tsx +0 -53
  668. package/template/components/secondary-panel/nav-link-rows.tsx +0 -83
  669. package/template/components/secondary-panels/list-hub-panel.tsx +0 -39
  670. package/template/components/secondary-panels/question-bank-panel.tsx +0 -39
  671. package/template/components/secondary-panels/registry.tsx +0 -15
  672. package/template/components/section-cards.tsx +0 -106
  673. package/template/components/templates/focused-workflow-layouts.tsx +0 -448
  674. package/template/components/templates/focused-workflow-page-template.tsx +0 -69
  675. package/template/components/templates/page-loading-shell.tsx +0 -262
  676. package/template/components/ui/button-group.tsx +0 -1
  677. package/template/docs/consumer-app-pattern.md +0 -39
  678. package/template/docs/focused-workflow-page-pattern.md +0 -84
  679. package/template/lib/list-hub-nav.ts +0 -121
  680. package/template/lib/mock/list-hub-directory.ts +0 -27
  681. package/template/lib/mock/list-hub-kpi.ts +0 -27
  682. package/template/lib/page-loading-variant.ts +0 -40
  683. /package/template/components/{getting-started.tsx → onboarding/getting-started.tsx} +0 -0
  684. /package/template/components/{nav-documents.tsx → sidebar/nav-documents.tsx} +0 -0
  685. /package/template/components/{nav-main.tsx → sidebar/nav-main.tsx} +0 -0
  686. /package/template/components/{nav-secondary.tsx → sidebar/nav-secondary.tsx} +0 -0
  687. /package/template/components/{nav-user.tsx → sidebar/nav-user.tsx} +0 -0
  688. /package/template/components/{sidebar-auto-open.tsx → sidebar/sidebar-auto-open.tsx} +0 -0
@@ -0,0 +1,416 @@
1
+ # Exxat DS — Token Taxonomy
2
+
3
+ **Status:** Authoritative · **Audience:** humans + AI agents · **Standard:** WCAG 2.1 AA
4
+ **Machine-readable index:** [`packages/ui/tokens/hooks-index.json`](../../packages/ui/tokens/hooks-index.json) (generated)
5
+ **Defining file (canonical):** [`packages/ui/src/globals.css`](../../packages/ui/src/globals.css). Consumer apps (`apps/web/app/globals.css`, the `create-exxat-app` starter at `packages/ui/template/app/globals.css`) are thin shells that `@import "@exxatdesignux/ui/globals.css"` and only declare their own `@source` directive. The dedicated `packages/ui/src/theme.css` file has been **retired** — see [`docs/migrations/0003-globals-css-canonical.md`](./migrations/0003-globals-css-canonical.md).
6
+
7
+ This document formalizes the **naming, layering, and ownership** of every CSS
8
+ custom property the design system ships. It is the answer to "what do I name a
9
+ new token?" and "is there already a token for this?". If something here
10
+ disagrees with the CSS, the CSS wins — and this file needs an edit.
11
+
12
+ ---
13
+
14
+ ## 1. Layering — where a token lives
15
+
16
+ Tokens are declared in **four** layers. Higher layers may consume lower ones;
17
+ they MUST NOT redefine names from a higher layer with a different meaning.
18
+
19
+ | Layer | File | Purpose | Edit when |
20
+ |---|---|---|---|
21
+ | **L0 — Exxat canonical** | `packages/ui/src/globals.css` `:root` (block tagged `Exxat L0 — canonical namespace`) | SLDS-style flat namespace (`--exxat-color-surface-1`, `--exxat-radius-2`, …). The **official** names the DS scales on. Today these are `var()` aliases of L1; the canonical OKLCH will move here over time. | New product surface tokens, broader brand picker, anything you would otherwise put under "L1 with an Exxat brand prefix" |
22
+ | **L1 — shadcn semantic** | `packages/ui/src/globals.css` `:root`/`.dark` | OKLCH literals + back-compat vocabulary (`--background`, `--foreground`, `--brand-color`, …). Required for the upstream shadcn primitives in `components/ui/*`. **Frozen** — do not extend; new tokens go to L0. | Brand color changes, accessibility-driven contrast lifts, new chip/chart slot tied to shadcn names |
23
+ | **L2 — Tailwind bridge** | `packages/ui/src/globals.css` `@theme inline` block | Maps primitives to `--color-*` and `--radius-*` so Tailwind v4 emits utility classes (`bg-brand`, `bg-surface-1`, `text-chip-3`, `rounded-2`, …) | Whenever an L0/L1 token needs a Tailwind utility class |
24
+ | **L3 — Surface override** | Theme blocks (`.theme-one`, `.theme-prism`, `[data-contrast="high"]`, `[data-text-size]`) | Per-brand / per-mode rebindings of L1 names — never new names | New brand, new contrast mode, new density step |
25
+
26
+ > **Rules:**
27
+ > 1. A token is **introduced exactly once**. New product tokens land in **L0**.
28
+ > 2. L1 (shadcn names) is **frozen** — keep working, do not extend.
29
+ > 3. L2 bridges L0/L1 → Tailwind utilities, exactly once per name.
30
+ > 4. L3 re-binds existing L1 OKLCH values per theme. L0 inherits via `var(L1)` automatically.
31
+
32
+ ---
33
+
34
+ ## 2. Namespace map (all current prefixes)
35
+
36
+ Every prefix below has a clear role. **New tokens MUST land in one of these
37
+ prefixes** — if you cannot place a token, the design system is missing a
38
+ prefix; propose one in a PR and document it here first.
39
+
40
+ ### 2.0 Exxat L0 — canonical namespace (`--exxat-*`)
41
+
42
+ The **official** name layer. Mirrors the SLDS `--slds-g-*` pattern but flat
43
+ (no `-g-` for "global" yet, since we have a single product namespace). Today
44
+ these resolve to L1 via `var(...)` so every existing theme override flows
45
+ through automatically. New components SHOULD prefer L0 names; the linter will
46
+ not yet *force* this, but `apps/web/docs/migrations/0002-exxat-token-namespace.md`
47
+ tracks the rollout.
48
+
49
+ | Category | L0 token | Resolves to L1 |
50
+ |---|---|---|
51
+ | Surface | `--exxat-color-surface-1` … `-3` | `--background` / `--card` / `--popover` |
52
+ | Surface | `--exxat-color-surface-muted` / `-accent` / `-secondary` / `-sidebar` / `-input` | `--muted` / `--accent` / `--secondary` / `--sidebar` / `--input-background` |
53
+ | Ink | `--exxat-color-ink-1` / `-2` | `--foreground` / `--muted-foreground` |
54
+ | Ink | `--exxat-color-ink-on-surface-2` / `-on-surface-3` | `--card-foreground` / `--popover-foreground` |
55
+ | Ink | `--exxat-color-ink-on-brand` / `-on-primary` / `-on-secondary` / `-on-accent` / `-on-destructive` | foreground tokens |
56
+ | Brand | `--exxat-color-brand-1` / `-2` / `-3` / `-deep` | `--brand-color` / `-dark` / `-light` / `-deep` |
57
+ | Brand | `--exxat-color-brand-tint-1` / `-2` / `-3` | `--brand-tint` / `-subtle` / `-light` |
58
+ | Action | `--exxat-color-action-primary` / `-secondary` / `-destructive` | `--primary` / `--secondary` / `--destructive` |
59
+ | Border | `--exxat-color-border-1` / `-control-subtle` / `-control-1` / `-control-2` | `--border` ladder (see §6) |
60
+ | Focus | `--exxat-color-focus-ring` | `--ring` |
61
+ | Overlay | `--exxat-color-overlay` | `--overlay` |
62
+ | Chart | `--exxat-color-chart-1` … `-5` | `--chart-1` … `-5` |
63
+ | Chip | `--exxat-color-chip-1` … `-5` / `-destructive` | `--chip-1` … `-5` / `-destructive` |
64
+ | Radius | `--exxat-radius-1` … `-6` | `4px` / `8px` / `12px` / `16px` / `20px` / `24px` |
65
+ | Spacing | `--exxat-spacing-1` / `-2` / `-3` / `-4` / `-5` / `-6` / `-8` / `-12` | Tailwind base scale |
66
+ | Control | `--exxat-control-height-1` / `-2` / `-3` | `--control-height-sm` / `-` / `-touch` |
67
+
68
+ **Tailwind utilities** are emitted via L2 bridges in `@theme inline`:
69
+
70
+ | L0 token | Tailwind utility |
71
+ |---|---|
72
+ | `--exxat-color-surface-1` | `bg-surface-1`, `text-surface-1`, `border-surface-1`, … |
73
+ | `--exxat-color-ink-1` / `-2` | `text-ink-1`, `text-ink-2` |
74
+ | `--exxat-color-brand-1` … `-3` | `bg-brand-1`, `bg-brand-2`, `bg-brand-3` |
75
+ | `--exxat-color-brand-tint-1` … `-3` | `bg-brand-tint-1`, … |
76
+ | `--exxat-color-border-1` | `border-1` |
77
+ | `--exxat-color-focus-ring` | `ring-focus-ring` |
78
+ | `--exxat-radius-1` … `-6` | `rounded-1` … `rounded-6` |
79
+
80
+ > The existing `bg-background`, `bg-brand`, `rounded-md`, etc. are not going
81
+ > away — they alias to the same OKLCH. Both forms work side-by-side; **new code**
82
+ > should reach for the L0 form for clarity and grep-ability (`grep --exxat-`
83
+ > finds every product token without false positives from shadcn names).
84
+
85
+ ### 2.1 Semantic surface (shadcn core, L1 — frozen)
86
+
87
+ Inherited from shadcn / Radix and treated as the **stable** semantic vocabulary
88
+ across every shadcn component. Do not rename — downstream `components/ui/*`
89
+ expects these.
90
+
91
+ | Token | Role |
92
+ |---|---|
93
+ | `--background` / `--foreground` | Page canvas + ink |
94
+ | `--card` / `--card-foreground` | Raised card surfaces |
95
+ | `--popover` / `--popover-foreground` | Floating menus, tooltips, dropdowns |
96
+ | `--primary` / `--primary-foreground` | Default CTA color (neutral charcoal, **not** brand — see §3) |
97
+ | `--secondary` / `--secondary-foreground` | Secondary button + de-emphasized surface |
98
+ | `--muted` / `--muted-foreground` | Muted body copy, subtle backgrounds |
99
+ | `--accent` / `--accent-foreground` | Hovered list rows, low-contrast chips |
100
+ | `--destructive` / `--destructive-foreground` | Errors, delete confirms |
101
+ | `--border` | Decorative dividers (no AA requirement) |
102
+ | `--border-control` / `--border-control-3` / `--border-control-35` / `--control-border` | Form-field borders — see §6 for the contrast ladder |
103
+ | `--input` / `--input-background` | Input outline + fill |
104
+ | `--ring` | Focus ring (≥ 3:1, SC 2.4.11) |
105
+ | `--overlay` | Modal / sheet / drawer scrim |
106
+
107
+ ### 2.2 Brand (`--brand-*`)
108
+
109
+ The **product accent** — different from `--primary` (which is neutral). Exxat
110
+ ships **two** brand themes (Exxat One = lavender 286.1, Exxat Prism = rose 343)
111
+ plus user-picked custom brand.
112
+
113
+ | Token | Role |
114
+ |---|---|
115
+ | `--brand-color` | Solid brand fill (used on chips, links, charts when product) |
116
+ | `--brand-color-light` / `--brand-color-dark` / `--brand-color-deep` | Brand scale (light / dark / deepest) |
117
+ | `--brand-foreground` | Ink on a solid brand fill |
118
+ | `--brand-tint` / `--brand-tint-light` / `--brand-tint-subtle` | Wash surfaces (sidebar, secondary panel) |
119
+ | `--brand-preview-one` / `--brand-preview-prism` | Fixed swatches for the brand picker — **never change with the active theme** |
120
+
121
+ ### 2.3 Sidebar + secondary panel (`--sidebar-*`, `--secondary-panel-bg`)
122
+
123
+ Three-level brand chrome stack. See `docs/shell-surface-elevation-pattern.md`.
124
+
125
+ | Token | Role |
126
+ |---|---|
127
+ | `--sidebar` / `--sidebar-foreground` | Primary sidebar (= `--brand-tint` on product themes) |
128
+ | `--sidebar-primary` / `--sidebar-primary-foreground` | Solid pill / active hint on sidebar |
129
+ | `--sidebar-accent` / `--sidebar-accent-foreground` | Hovered/active row in sidebar |
130
+ | `--sidebar-border` | Inner divider |
131
+ | `--sidebar-ring` | Focus ring inside sidebar |
132
+ | `--sidebar-section-label-foreground` | Section title — mixed against real `--sidebar`, not `--background` |
133
+ | `--secondary-panel-bg` | Nested panel (Question bank) — level 1 between sidebar and canvas |
134
+
135
+ ### 2.4 Chips / badges (`--chip-*`)
136
+
137
+ A five-slot **AA-compliant** palette for tags, kanban badges, and small status
138
+ chips. All slots maintain ≥ 4.5:1 against `--background`.
139
+
140
+ | Token | Hue (light) |
141
+ |---|---|
142
+ | `--chip-1` | indigo (264) |
143
+ | `--chip-2` | teal (184) |
144
+ | `--chip-3` | slate (227) |
145
+ | `--chip-4` | amber (84) |
146
+ | `--chip-5` | orange (70) |
147
+ | `--chip-destructive` | red (25) |
148
+
149
+ Use through `lib/list-status-badges.ts` (`LIST_HUB_STATUS_TINT_*`). Do not pick a
150
+ chip slot to mean "this entity uses indigo" — pick by **semantic intent**
151
+ (success / warning / neutral / danger / info).
152
+
153
+ ### 2.5 Charts (`--chart-*`)
154
+
155
+ Five slots scoped for `recharts` series. They are **darker** than chip slots
156
+ because chart areas are larger and need denser pigment.
157
+
158
+ | Token | Hue |
159
+ |---|---|
160
+ | `--chart-1` | blue (264) |
161
+ | `--chart-2` | green-teal (184) |
162
+ | `--chart-3` | slate (227) |
163
+ | `--chart-4` | amber (84) |
164
+ | `--chart-5` | orange (70) |
165
+
166
+ Insights derived from charts use `--insight-severity-*-bg` / `-fg` (KPI strip).
167
+
168
+ ### 2.6 Interactive hover (`--interactive-hover-*`)
169
+
170
+ Single source for ghost-button hover, list-row hover, table chrome hover.
171
+ Replaces ad-hoc `hover:bg-muted` so theming + dark mode flip together.
172
+
173
+ | Token | Role |
174
+ |---|---|
175
+ | `--interactive-hover` | Default opaque hover fill |
176
+ | `--interactive-hover-foreground` | Ink on hover |
177
+ | `--interactive-hover-subtle` / `-soft` / `-medium` / `-strong` | Opacity ladder (50 / 40 / 60 / 70 mix) |
178
+ | `--interactive-hover-row` | List/table row hover (accent-mixed) |
179
+
180
+ ### 2.7 DataTable (`--dt-*`)
181
+
182
+ Pinned cells must be **opaque** so they sit above scrolled rows. These tokens
183
+ exist because `var(--muted)` alone gives translucent surfaces in dark mode.
184
+
185
+ | Token | Role |
186
+ |---|---|
187
+ | `--dt-row-bg` | Base row fill (opaque) |
188
+ | `--dt-row-hover` | Hovered row |
189
+ | `--dt-row-selected` / `--dt-row-selected-fg` | Selection state |
190
+ | `--dt-header-bg` | Header row |
191
+ | `--dt-group-bg` | Group / divider row |
192
+ | `--dt-new-row-bg` / `--dt-new-row-border` | "Just created" highlight |
193
+
194
+ ### 2.8 KPI strip (`--key-metrics-*`, `--insight-severity-*`)
195
+
196
+ Used by `KeyMetrics` (both `variant="flat"` and `variant="card"`). See
197
+ `docs/kpi-flat-band-pattern.md`.
198
+
199
+ | Token | Role |
200
+ |---|---|
201
+ | `--key-metrics-flat-cell-bg` | Transparent on `variant="flat"` |
202
+ | `--key-metrics-flat-divider` | Cell-border hairline |
203
+ | `--key-metrics-flat-band-radial` | OKLCH brand glow at bottom |
204
+ | `--key-metrics-flat-band-shadow` | `none` on flat |
205
+ | `--key-metrics-card-glow-radial` | Card-variant glow |
206
+ | `--insight-severity-warning-bg` / `-fg` | Yellow severity (chart-4 mix) |
207
+ | `--insight-severity-info-bg` / `-fg` | Blue severity (chart-1 mix) |
208
+
209
+ ### 2.9 Conditional formatting (`--conditional-rule-*`)
210
+
211
+ Row backgrounds for table conditional-format rules. Already six slots — adding
212
+ more requires expanding the rule type in `lib/table-conditional-format.ts`.
213
+
214
+ ### 2.10 Icon discs (`--icon-disc-*`)
215
+
216
+ Soft tinted backgrounds for icon "discs" (KPI cards, banner avatars).
217
+ Bg ≈ 14% mix, fg from `--chip-*` / `--brand-color-dark`.
218
+
219
+ ### 2.11 Avatar (`--avatar-initials-*`)
220
+
221
+ Bg + fg for initials avatars in `DataTable` cells. Pair brand wash with deep
222
+ brand ink for ≥ 4.5:1 on white rows.
223
+
224
+ ### 2.12 Leo (Ask Leo) (`--leo-*`)
225
+
226
+ | Token | Role |
227
+ |---|---|
228
+ | `--leo-surface-tint-a` / `-b` | Top / bottom of the AI panel wash |
229
+ | `--leo-surface-gradient` | Full linear wash |
230
+
231
+ ### 2.13 Layout + density (`--header-height`, `--control-*`, `--table-row-height`, `--scaling`)
232
+
233
+ Sized in **px** so JS can `parseFloat` them. Multiply by `--scaling` for future
234
+ density modes.
235
+
236
+ ### 2.14 Border radius (`--radius`, `--radius-sm`/`-md`/`-lg`/`-xl`/`-2xl`/`-3xl`)
237
+
238
+ `--radius` is the **base** (8px). The named scale lives at L2 (`@theme inline`)
239
+ so Tailwind emits `rounded-sm`, `rounded-md`, etc.
240
+
241
+ ### 2.15 Shadows + transitions (`--shadow-*`, `--transition-*`)
242
+
243
+ | Token | Role |
244
+ |---|---|
245
+ | `--shadow-sm` / `-md` / `-lg` | Three-step elevation |
246
+ | `--transition-fast` / `-normal` / `-colors` | Standard durations |
247
+ | `--sticky-edge-fade` | Edge-of-pinned-column gradient |
248
+
249
+ ### 2.16 Promo / banner (`--banner-prism-bg`)
250
+
251
+ Rose hue 343, used **universally** as the Exxat Prism promo highlight — does
252
+ not flip with the active theme.
253
+
254
+ ### 2.17 OS chrome (`--theme-color-chrome`)
255
+
256
+ Mirrored into `<meta name="theme-color">` so the browser titlebar matches the
257
+ sidebar. Hex literal here is **intentional** — it is consumed by the browser
258
+ parser, not CSS.
259
+
260
+ ---
261
+
262
+ ## 3. Color identity rules
263
+
264
+ 1. **`--primary` is neutral**, not brand. The product accent is `--brand-color`.
265
+ This matches Exxat's design language (neutral charcoal CTAs, lavender/rose
266
+ reserved for surfaces and chips). When a button **must** carry the brand,
267
+ use `bg-brand-color text-brand-foreground` — not `bg-primary`.
268
+ 2. **Brand washes (`--brand-tint*`) are surfaces; brand colors
269
+ (`--brand-color*`) are ink/fills.** Mixing the roles produces low-contrast
270
+ chips and washed-out sidebars.
271
+ 3. **Chip ≠ chart.** Chip tokens are sized for inline chips; chart tokens are
272
+ sized for filled SVG areas. Re-use a chip color on a chart fill (or vice
273
+ versa) only when the design explicitly calls for it.
274
+ 4. **Every theme MUST keep `--brand-preview-one` and `--brand-preview-prism`
275
+ fixed.** They drive the brand picker swatch — flipping them with the
276
+ selected theme defeats the picker.
277
+
278
+ ---
279
+
280
+ ## 4. Naming a new token — checklist
281
+
282
+ Run this list **before** opening `globals.css`:
283
+
284
+ - [ ] **Does an L0 alias already exist?** (`--exxat-color-surface-1`,
285
+ `--exxat-color-brand-1`, `--exxat-radius-2`, …) — prefer L0 for new
286
+ consumers; the L1 shadcn name is the back-compat alias.
287
+ - [ ] If the token is new, **name it at L0 first** (`--exxat-<category>-<slot>`).
288
+ Add an L1 mirror **only** if a shadcn primitive in `components/ui/*`
289
+ needs it — that's the only place L1 is still required.
290
+ - [ ] Can it be expressed with an existing semantic token? (`--exxat-color-surface-muted`
291
+ instead of `--exxat-color-surface-4`, `--brand-color-dark` instead of
292
+ `--brand-deeper`)
293
+ - [ ] Does it belong to a **scoped surface** (`KeyMetrics`, `DataTable`, Leo,
294
+ sidebar, secondary panel)? If yes, **prefix with that surface**, e.g.
295
+ `--key-metrics-*`, `--dt-*`, `--leo-*`. Do **not** drop scoped tokens at
296
+ the top level.
297
+ - [ ] Is it a **decoration** or a **decision**? Decorations (gradients, fades,
298
+ shadows) live alongside the surface tokens they decorate. Decisions
299
+ (semantic intent: warning, info, success) live with chips / charts /
300
+ insight tokens.
301
+ - [ ] Does it need a Tailwind utility (`bg-*`, `text-*`, `border-*`)? If yes,
302
+ add the `--color-<name>` bridge in `@theme inline` in **both** globals
303
+ files.
304
+ - [ ] Does dark mode / high contrast / Prism need different OKLCH? Override at
305
+ L3 — never branch by emitting a new token name.
306
+ - [ ] Is it derived (e.g. 14% mix of `--brand-color`)? Use `color-mix(in oklch,
307
+ …)` so theme overrides cascade automatically.
308
+ - [ ] Add the bridge in `packages/ui/src/globals.css` `@theme inline`. App
309
+ consumers @import that file, so a single edit is enough — no manual
310
+ mirroring across `apps/web/app/globals.css` / `template/app/globals.css`.
311
+
312
+ If you tick all the boxes, also:
313
+
314
+ 1. Add an entry to this file (the right §).
315
+ 2. Re-run `pnpm --filter @exxatdesignux/ui tokens:index` (see §7) to refresh
316
+ `packages/ui/tokens/hooks-index.json`.
317
+
318
+ ---
319
+
320
+ ## 5. Deprecation policy
321
+
322
+ Tokens follow the same **add → mark deprecated → remove** lifecycle as code:
323
+
324
+ 1. **Add** the new token; switch components to consume it.
325
+ 2. **Mark deprecated** in `packages/ui/src/globals.css` with a comment:
326
+ ```css
327
+ /* @deprecated v0.2.18 — use --brand-color-dark; remove in v0.4.0 */
328
+ --brand-deep: var(--brand-color-dark);
329
+ ```
330
+ 3. **Migrate** consumers (script + grep + manual review). Document in
331
+ `docs/migrations/<NNNN>-<slug>.md`.
332
+ 4. **Remove** in the version named in the deprecation comment. Bump
333
+ `@exxatdesignux/ui` major or document in `CHANGELOG.md`.
334
+
335
+ Active deprecations are listed in `docs/migrations/README.md`.
336
+
337
+ ---
338
+
339
+ ## 6. Form-field border contrast ladder
340
+
341
+ Form-field borders are a recurring source of WCAG SC 1.4.11 (3:1 UI contrast)
342
+ failures. Use the right rung — do not guess:
343
+
344
+ | Token | OKLCH L | Contrast vs `oklch(1 0 0)` | When to use |
345
+ |---|---|---|---|
346
+ | `--border` | 0.92 | < 3:1 | **Decorative only** — card dividers, list separators |
347
+ | `--border-control` | 0.82 | < 3:1 | Layout chrome where contrast is not required |
348
+ | `--border-control-3` | 0.62 | ≈ 3:1 | Form-field border **minimum** — passes AA |
349
+ | `--border-control-35` | 0.25 | > 3.5:1 | Form-field border **recommended** — passes AA + 16% buffer |
350
+ | `--control-border` (alias of `--border-control-3`) | — | — | Default form-field border slot |
351
+
352
+ If you find yourself reaching for `--border` on an `<input>`, stop — use
353
+ `--control-border`.
354
+
355
+ ---
356
+
357
+ ## 7. Machine-readable hooks index
358
+
359
+ Tooling (linters, codegens, design-tool sync) needs to discover tokens
360
+ programmatically. Run:
361
+
362
+ ```bash
363
+ pnpm --filter @exxatdesignux/ui tokens:index
364
+ ```
365
+
366
+ This regenerates [`packages/ui/tokens/hooks-index.json`](../../packages/ui/tokens/hooks-index.json),
367
+ which mirrors SLDS's `hooks-index.json` shape. The current index contains
368
+ **195 tokens** across **41 namespaces** (including the 12 Exxat L0 sub-namespaces
369
+ `exxat-surface`, `exxat-ink`, `exxat-brand`, `exxat-action`, `exxat-border`,
370
+ `exxat-focus`, `exxat-overlay`, `exxat-chart`, `exxat-chip`, `exxat-radius`,
371
+ `exxat-spacing`, `exxat-control`):
372
+
373
+ ```jsonc
374
+ {
375
+ "version": "0.2.18",
376
+ "source": "packages/ui/src/globals.css",
377
+ "generatedAt": "ISO-8601",
378
+ "tokens": {
379
+ "--exxat-color-brand-1": {
380
+ "namespace": "exxat-brand",
381
+ "category": "color",
382
+ "values": { "light": "var(--brand-color)" },
383
+ "tailwindUtilities": ["bg-brand-1", "text-brand-1", "border-brand-1", "ring-brand-1"],
384
+ "deprecated": false
385
+ },
386
+ "--brand-color": {
387
+ "namespace": "brand",
388
+ "category": "color",
389
+ "values": {
390
+ "light": "oklch(0.50 0.14 286.1)",
391
+ "dark": "oklch(0.50 0.14 286.1)",
392
+ "theme-one": "oklch(0.50 0.14 286.1)",
393
+ "theme-prism": "oklch(0.57 0.24 342)",
394
+ "high-contrast": "oklch(0.06 0 0)"
395
+ },
396
+ "tailwindUtilities": ["bg-brand", "text-brand", "border-brand", "ring-brand"],
397
+ "deprecated": false
398
+ }
399
+ }
400
+ }
401
+ ```
402
+
403
+ CI runs `tokens:index` and fails if the committed JSON is stale.
404
+
405
+ ---
406
+
407
+ ## 8. References
408
+
409
+ - `packages/ui/src/globals.css` — **single source of truth** for L1 / L2 / L3 primitives + Tailwind bridges
410
+ - `apps/web/app/globals.css`, `packages/ui/template/app/globals.css` — thin shells (`@import` + `@source`)
411
+ - `apps/web/docs/shell-surface-elevation-pattern.md` — sidebar / secondary panel
412
+ - `apps/web/docs/kpi-flat-band-pattern.md` — `--key-metrics-flat-*`
413
+ - `apps/web/docs/kpi-trend-pattern.md` — `--insight-severity-*` polarity
414
+ - `.cursor/rules/exxat-token-discipline.mdc` — enforcement rule (don't ship hex
415
+ literals, don't use deprecated tokens)
416
+ - `docs/migrations/` — token rename + removal history
@@ -1,6 +1,7 @@
1
1
  import { defineConfig, globalIgnores } from "eslint/config";
2
2
  import nextVitals from "eslint-config-next/core-web-vitals";
3
3
  import nextTs from "eslint-config-next/typescript";
4
+ import exxatDs from "@exxatdesignux/eslint-plugin";
4
5
 
5
6
  const eslintConfig = defineConfig([
6
7
  ...nextVitals,
@@ -31,6 +32,32 @@ const eslintConfig = defineConfig([
31
32
  ],
32
33
  },
33
34
  },
35
+ // -------------------------------------------------------------------------
36
+ // Exxat DS guardrails (@exxatdesignux/eslint-plugin — packages/eslint-plugin-exxat-ds).
37
+ // - no-hex-color: token discipline (no hex literals in JSX/style)
38
+ // - no-deprecated-tokens: reads @exxatdesignux/ui tokens/hooks-index.json
39
+ // - no-sonner-toast: enforces .cursor/rules/exxat-no-toast.mdc
40
+ // - no-slds-classes: enforces .cursor/rules/exxat-no-slds-leakage.mdc
41
+ // - no-lightning-elements: enforces .cursor/rules/exxat-no-slds-leakage.mdc
42
+ // -------------------------------------------------------------------------
43
+ {
44
+ files: ["app/**/*.{ts,tsx}", "components/**/*.{ts,tsx}", "lib/**/*.{ts,tsx}", "hooks/**/*.{ts,tsx}", "contexts/**/*.{ts,tsx}", "stores/**/*.{ts,tsx}"],
45
+ plugins: { "exxat-ds": exxatDs },
46
+ rules: {
47
+ "exxat-ds/no-hex-color": [
48
+ "warn",
49
+ {
50
+ // Files where hex is legitimately needed (CSS variable definitions,
51
+ // OS theme-color meta, etc.) are intentionally excluded by path.
52
+ allowFiles: ["/app/globals.css", "/packages/ui/src/globals.css", "/lib/theme-color", "/lib/windows-contrast-theme"],
53
+ },
54
+ ],
55
+ "exxat-ds/no-deprecated-tokens": "error",
56
+ "exxat-ds/no-sonner-toast": "error",
57
+ "exxat-ds/no-slds-classes": "error",
58
+ "exxat-ds/no-lightning-elements": "error",
59
+ },
60
+ },
34
61
  ]);
35
62
 
36
63
  export default eslintConfig;
@@ -3,7 +3,7 @@
3
3
  import * as React from "react"
4
4
  import { usePathname, useRouter, useSearchParams } from "next/navigation"
5
5
 
6
- import { useSecondaryPanel } from "@/components/secondary-panel"
6
+ import { useSecondaryPanel } from "@/components/sidebar"
7
7
  import { QUESTION_BANK_HUB_FIND_PATH, QUESTION_BANK_LIBRARY_PATH, QUESTION_BANK_LIST_PATH } from "@/lib/question-bank-nav"
8
8
 
9
9
  function rewriteLibraryCanonicalToDedicatedSurface(pathname: string, nextHref: string, hash: string): string {
@@ -108,7 +108,6 @@ const STATIC_COMMAND_GROUPS: CommandMenuGroup[] = [
108
108
  href: "/question-bank/library",
109
109
  keywords: "folders assessment items tree panel table",
110
110
  },
111
- { id: "nav-data-list", icon: "fa-light fa-table", label: "List hub", href: "/data-list" },
112
111
  { id: "nav-settings", icon: "fa-light fa-gear", label: "Settings", href: "/settings" },
113
112
  { id: "nav-help", icon: "fa-light fa-circle-question", label: "Help", href: "/help" },
114
113
  ],
@@ -3,21 +3,36 @@
3
3
  */
4
4
 
5
5
  import type { CommandMenuGroup, CommandMenuItem } from "@/lib/command-menu-config"
6
- import { LIST_HUB_DIRECTORY } from "@/lib/mock/list-hub-directory"
6
+ import { ALL_PLACEMENTS } from "@/lib/mock/placements"
7
7
 
8
8
  function sampleRowSearchItems(): CommandMenuItem[] {
9
- return LIST_HUB_DIRECTORY.map(row => ({
10
- id: `sample-row-${row.id}`,
11
- label: row.title,
12
- icon: "fa-light fa-calendar-days",
13
- href: "/data-list",
14
- keywords: [row.id, row.title, row.category, row.eventDate, "list hub", "calendar"]
15
- .filter(Boolean)
16
- .join(" "),
17
- }))
9
+ return ALL_PLACEMENTS.map((p) => {
10
+ const nameParts = p.student.trim().split(/\s+/)
11
+ return {
12
+ id: `sample-row-${p.id}`,
13
+ label: `Row ${p.id} — ${p.student}`,
14
+ icon: "fa-light fa-table",
15
+ href: `/data-list/${p.id}`,
16
+ keywords: [
17
+ `row ${p.id}`,
18
+ p.student,
19
+ ...nameParts,
20
+ p.program,
21
+ p.site,
22
+ p.internship,
23
+ p.specialization,
24
+ p.email,
25
+ p.supervisor,
26
+ p.status,
27
+ p.compliance,
28
+ ]
29
+ .filter(Boolean)
30
+ .join(" "),
31
+ }
32
+ })
18
33
  }
19
34
 
20
- /** Built once at module load — avoids remapping rows on every layout render. */
35
+ /** Built once at module load — avoids remapping all placement rows on every layout render. */
21
36
  export const COMMAND_MENU_SEARCH_DATA_GROUPS: CommandMenuGroup[] = [
22
37
  {
23
38
  id: "sample-rows",
@@ -27,6 +42,7 @@ export const COMMAND_MENU_SEARCH_DATA_GROUPS: CommandMenuGroup[] = [
27
42
  },
28
43
  ]
29
44
 
45
+ /** Demo rows for the list hub — search-only so the palette stays lightweight on open. */
30
46
  export function getCommandMenuSearchDataGroups(): CommandMenuGroup[] {
31
47
  return COMMAND_MENU_SEARCH_DATA_GROUPS
32
48
  }
@@ -0,0 +1,10 @@
1
+ import type { DataListViewType } from "@/lib/data-list-view"
2
+
3
+ /** Views implemented in `ComplianceTable` — keep in sync with the renderers passed to `HubTable`. */
4
+ export const COMPLIANCE_SUPPORTED_VIEWS = [
5
+ "table",
6
+ "list",
7
+ "board",
8
+ "panel",
9
+ "dashboard",
10
+ ] as const satisfies readonly DataListViewType[]
@@ -1,97 +1,6 @@
1
- import type { ConditionalRule, FilterTextMask } from "@/components/table-properties/types"
2
-
3
- export type ConditionalColumnHint = {
4
- key: string
5
- sortKey?: string
6
- filter?: { type?: string; textMask?: FilterTextMask }
7
- }
8
-
9
- function rowValueForRule<T extends Record<string, unknown>>(
10
- row: T,
11
- rule: ConditionalRule,
12
- columns?: ConditionalColumnHint[],
13
- ): string {
14
- const col = columns?.find(c => c.key === rule.fieldKey)
15
- const dataKey = (col?.sortKey ?? rule.fieldKey) as keyof T
16
- return String(row[dataKey] ?? "")
17
- }
18
-
19
- function ruleHasActiveValues(
20
- rule: ConditionalRule,
21
- columns?: ConditionalColumnHint[],
22
- ): boolean {
23
- if (rule.values.length === 0) return false
24
- const col = columns?.find(c => c.key === rule.fieldKey)
25
- if (col?.filter?.type === "text") return (rule.values[0] ?? "").trim().length > 0
26
- return true
27
- }
28
-
29
- function conditionalTextMatches(
30
- cellVal: string,
31
- needle: string,
32
- op: "contains" | "not_contains",
33
- textMask: FilterTextMask | undefined,
34
- ) {
35
- const v = cellVal.trim()
36
- const n = needle.trim()
37
- if (!n) return op === "not_contains"
38
- if (textMask === "phone" || textMask === "zip") {
39
- const nd = n.replace(/\D/g, "")
40
- const hay = v.replace(/\D/g, "")
41
- if (!nd) return op === "not_contains"
42
- const hit = hay.includes(nd)
43
- return op === "contains" ? hit : !hit
44
- }
45
- const hit = v.toLowerCase().includes(n.toLowerCase())
46
- return op === "contains" ? hit : !hit
47
- }
48
-
49
- /** Whether a conditional rule matches a row (same logic as DataTable cells). */
50
- export function conditionalRuleMatchesRow<T extends Record<string, unknown>>(
51
- row: T,
52
- rule: ConditionalRule,
53
- columns?: ConditionalColumnHint[],
54
- ): boolean {
55
- if (!ruleHasActiveValues(rule, columns)) return false
56
- const v = rowValueForRule(row, rule, columns).trim()
57
- const col = columns?.find(c => c.key === rule.fieldKey)
58
- const textMask = col?.filter?.type === "text" ? col.filter.textMask : undefined
59
- switch (rule.operator) {
60
- case "is":
61
- return rule.values.includes(v)
62
- case "is_not":
63
- return !rule.values.includes(v)
64
- case "contains":
65
- return rule.values.some(val => conditionalTextMatches(v, val, "contains", textMask))
66
- case "not_contains":
67
- return !rule.values.some(val => conditionalTextMatches(v, val, "contains", textMask))
68
- default:
69
- return false
70
- }
71
- }
72
-
73
- /** First matching conditional rule background for a row (list/board row tint). */
74
- export function getConditionalRowBackground<T extends Record<string, unknown>>(
75
- row: T,
76
- rules: ConditionalRule[] | undefined,
77
- columns?: ConditionalColumnHint[],
78
- ): string | undefined {
79
- if (!rules?.length) return undefined
80
- for (const rule of rules) {
81
- if (conditionalRuleMatchesRow(row, rule, columns)) return rule.bgColor
82
- }
83
- return undefined
84
- }
85
-
86
- /** Background for one table cell from conditional rules on that column. */
87
- export function getConditionalCellBackground<T extends Record<string, unknown>>(
88
- row: T,
89
- colKey: string,
90
- rules: ConditionalRule[] | undefined,
91
- columns?: ConditionalColumnHint[],
92
- ): string | undefined {
93
- if (!rules?.length) return undefined
94
- const rule = rules.find(r => r.fieldKey === colKey)
95
- if (!rule || !conditionalRuleMatchesRow(row, rule, columns)) return undefined
96
- return rule.bgColor
97
- }
1
+ export {
2
+ conditionalRuleMatchesRow,
3
+ getConditionalRowBackground,
4
+ getConditionalCellBackground,
5
+ type ConditionalColumnHint,
6
+ } from "@exxatdesignux/ui/lib/conditional-rule-match"