@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.
- package/CHANGELOG.md +60 -7
- package/bin/sync-extras.mjs +116 -29
- package/consumer-extras/README.md +42 -7
- package/consumer-extras/cursor-rules/exxat-accessibility.mdc +39 -0
- package/consumer-extras/cursor-rules/exxat-board-cards.mdc +26 -0
- package/consumer-extras/cursor-rules/exxat-breadcrumbs-no-back.mdc +21 -0
- package/consumer-extras/cursor-rules/exxat-card-vs-list-rows.mdc +21 -0
- package/consumer-extras/cursor-rules/exxat-centralized-list-dataset.mdc +44 -0
- package/consumer-extras/cursor-rules/exxat-collaboration-access.mdc +32 -0
- package/consumer-extras/cursor-rules/exxat-command-menu.mdc +22 -0
- package/consumer-extras/cursor-rules/exxat-dashboard-view-charts.mdc +53 -0
- package/consumer-extras/cursor-rules/exxat-data-tables.mdc +41 -0
- package/consumer-extras/cursor-rules/exxat-dedicated-search-surfaces.mdc +25 -0
- package/consumer-extras/cursor-rules/exxat-drawer-vs-dialog.mdc +22 -0
- package/consumer-extras/cursor-rules/exxat-ds-agents.mdc +56 -0
- package/consumer-extras/cursor-rules/exxat-fontawesome-icons.mdc +31 -0
- package/consumer-extras/cursor-rules/exxat-kbd-shortcuts.mdc +100 -0
- package/consumer-extras/cursor-rules/exxat-kpi-flat-band.mdc +28 -0
- package/consumer-extras/cursor-rules/exxat-kpi-max-four.mdc +21 -0
- package/consumer-extras/cursor-rules/exxat-kpi-trends.mdc +31 -0
- package/consumer-extras/cursor-rules/exxat-list-page-connected-views.mdc +24 -0
- package/consumer-extras/cursor-rules/exxat-list-page-view-shells.mdc +31 -0
- package/consumer-extras/cursor-rules/exxat-mono-ids.mdc +30 -0
- package/consumer-extras/cursor-rules/exxat-no-slds-leakage.mdc +78 -0
- package/consumer-extras/cursor-rules/exxat-no-toast.mdc +25 -0
- package/consumer-extras/cursor-rules/exxat-page-vs-drawer.mdc +23 -0
- package/consumer-extras/cursor-rules/exxat-person-identity-display.mdc +47 -0
- package/consumer-extras/cursor-rules/exxat-primary-nav-secondary-panel.mdc +52 -0
- package/consumer-extras/cursor-rules/exxat-question-bank-hub-header.mdc +28 -0
- package/consumer-extras/cursor-rules/exxat-reuse-before-custom.mdc +34 -0
- package/consumer-extras/cursor-rules/exxat-table-properties-drawer.mdc +77 -0
- package/consumer-extras/cursor-rules/exxat-token-discipline.mdc +103 -0
- package/consumer-extras/cursor-skills/exxat-accessibility/SKILL.md +1 -1
- package/consumer-extras/cursor-skills/exxat-board-cards/SKILL.md +2 -2
- package/consumer-extras/cursor-skills/exxat-centralized-list-dataset/SKILL.md +4 -15
- package/consumer-extras/cursor-skills/exxat-ds-skill/SKILL.md +13 -28
- package/consumer-extras/cursor-skills/exxat-ds-skill/references/data-table-pattern.md +1 -1
- package/consumer-extras/cursor-skills/exxat-primary-nav-secondary-panel/SKILL.md +2 -4
- package/consumer-extras/handbook/HANDBOOK.md +185 -0
- package/consumer-extras/handbook/glossary.md +57 -0
- package/consumer-extras/handbook/reference-implementations.md +126 -0
- package/consumer-extras/handbook/voice-and-tone.md +262 -0
- package/consumer-extras/patterns/command-menu-pattern.md +1 -1
- package/consumer-extras/patterns/consumer-upgrade-checklist.md +0 -20
- package/consumer-extras/patterns/data-views-pattern.md +17 -54
- package/consumer-extras/patterns/shell-surface-elevation-pattern.md +3 -5
- package/dist/components/data-table/filter-date-calendar.d.ts +10 -0
- package/dist/components/data-table/filter-date-calendar.js +280 -0
- package/dist/components/data-table/filter-date-calendar.js.map +1 -0
- package/dist/components/data-table/filter-text-value-input.d.ts +15 -0
- package/dist/components/data-table/filter-text-value-input.js +561 -0
- package/dist/components/data-table/filter-text-value-input.js.map +1 -0
- package/dist/components/data-table/index.d.ts +45 -0
- package/dist/components/data-table/index.js +3085 -0
- package/dist/components/data-table/index.js.map +1 -0
- package/dist/components/data-table/pagination.d.ts +28 -0
- package/dist/components/data-table/pagination.js +3264 -0
- package/dist/components/data-table/pagination.js.map +1 -0
- package/dist/components/data-table/types.d.ts +84 -0
- package/dist/components/data-table/types.js +3 -0
- package/dist/components/data-table/types.js.map +1 -0
- package/dist/components/data-table/use-table-state.d.ts +116 -0
- package/dist/components/data-table/use-table-state.js +670 -0
- package/dist/components/data-table/use-table-state.js.map +1 -0
- package/dist/components/data-views/board-card-primitives.d.ts +22 -0
- package/dist/components/data-views/board-card-primitives.js +84 -0
- package/dist/components/data-views/board-card-primitives.js.map +1 -0
- package/dist/components/data-views/data-row-list.d.ts +33 -0
- package/dist/components/data-views/data-row-list.js +106 -0
- package/dist/components/data-views/data-row-list.js.map +1 -0
- package/dist/components/data-views/finder-panel-view.d.ts +54 -0
- package/dist/components/data-views/finder-panel-view.js +388 -0
- package/dist/components/data-views/finder-panel-view.js.map +1 -0
- package/dist/components/data-views/folder-grid-view.d.ts +22 -0
- package/dist/components/data-views/folder-grid-view.js +58 -0
- package/dist/components/data-views/folder-grid-view.js.map +1 -0
- package/dist/components/data-views/hub-table.d.ts +167 -0
- package/dist/components/data-views/hub-table.js +5561 -0
- package/dist/components/data-views/hub-table.js.map +1 -0
- package/dist/components/data-views/index.d.ts +27 -0
- package/dist/components/data-views/index.js +6575 -0
- package/dist/components/data-views/index.js.map +1 -0
- package/dist/components/data-views/list-page-board-card.d.ts +72 -0
- package/dist/components/data-views/list-page-board-card.js +264 -0
- package/dist/components/data-views/list-page-board-card.js.map +1 -0
- package/dist/components/data-views/list-page-board-template.d.ts +24 -0
- package/dist/components/data-views/list-page-board-template.js +137 -0
- package/dist/components/data-views/list-page-board-template.js.map +1 -0
- package/dist/components/data-views/list-page-connected-view-body.d.ts +19 -0
- package/dist/components/data-views/list-page-connected-view-body.js +116 -0
- package/dist/components/data-views/list-page-connected-view-body.js.map +1 -0
- package/dist/components/data-views/list-page-split-details-placeholder.d.ts +14 -0
- package/dist/components/data-views/list-page-split-details-placeholder.js +38 -0
- package/dist/components/data-views/list-page-split-details-placeholder.js.map +1 -0
- package/dist/components/data-views/list-page-split-hub-chrome.d.ts +17 -0
- package/dist/components/data-views/list-page-split-hub-chrome.js +54 -0
- package/dist/components/data-views/list-page-split-hub-chrome.js.map +1 -0
- package/dist/components/data-views/list-page-split-hub-tokens.d.ts +12 -0
- package/dist/components/data-views/list-page-split-hub-tokens.js +8 -0
- package/dist/components/data-views/list-page-split-hub-tokens.js.map +1 -0
- package/dist/components/data-views/list-page-tree-column-header.d.ts +15 -0
- package/dist/components/data-views/list-page-tree-column-header.js +22 -0
- package/dist/components/data-views/list-page-tree-column-header.js.map +1 -0
- package/dist/components/data-views/list-page-tree-panel-shell.d.ts +25 -0
- package/dist/components/data-views/list-page-tree-panel-shell.js +146 -0
- package/dist/components/data-views/list-page-tree-panel-shell.js.map +1 -0
- package/dist/components/data-views/os-folder-glyph.d.ts +35 -0
- package/dist/components/data-views/os-folder-glyph.js +104 -0
- package/dist/components/data-views/os-folder-glyph.js.map +1 -0
- package/dist/components/data-views/outline-tree-menu.d.ts +36 -0
- package/dist/components/data-views/outline-tree-menu.js +131 -0
- package/dist/components/data-views/outline-tree-menu.js.map +1 -0
- package/dist/components/table-properties/column-row.d.ts +22 -0
- package/dist/components/table-properties/column-row.js +153 -0
- package/dist/components/table-properties/column-row.js.map +1 -0
- package/dist/components/table-properties/draggable-list.d.ts +24 -0
- package/dist/components/table-properties/draggable-list.js +53 -0
- package/dist/components/table-properties/draggable-list.js.map +1 -0
- package/dist/components/table-properties/drawer-button.d.ts +110 -0
- package/dist/components/table-properties/drawer-button.js +2748 -0
- package/dist/components/table-properties/drawer-button.js.map +1 -0
- package/dist/components/table-properties/drawer.d.ts +100 -0
- package/dist/components/table-properties/drawer.js +2595 -0
- package/dist/components/table-properties/drawer.js.map +1 -0
- package/dist/components/table-properties/filter-card.d.ts +24 -0
- package/dist/components/table-properties/filter-card.js +854 -0
- package/dist/components/table-properties/filter-card.js.map +1 -0
- package/dist/components/table-properties/index.d.ts +14 -0
- package/dist/components/table-properties/index.js +2768 -0
- package/dist/components/table-properties/index.js.map +1 -0
- package/dist/components/table-properties/sort-card.d.ts +20 -0
- package/dist/components/table-properties/sort-card.js +102 -0
- package/dist/components/table-properties/sort-card.js.map +1 -0
- package/dist/components/templates/dedicated-search-landing-template.d.ts +21 -0
- package/dist/components/templates/dedicated-search-landing-template.js +254 -0
- package/dist/components/templates/dedicated-search-landing-template.js.map +1 -0
- package/dist/components/templates/dedicated-search-results-template.d.ts +15 -0
- package/dist/components/templates/dedicated-search-results-template.js +16 -0
- package/dist/components/templates/dedicated-search-results-template.js.map +1 -0
- package/dist/components/templates/index.d.ts +9 -0
- package/dist/components/templates/index.js +2720 -0
- package/dist/components/templates/index.js.map +1 -0
- package/dist/components/templates/list-page.d.ts +83 -0
- package/dist/components/templates/list-page.js +2433 -0
- package/dist/components/templates/list-page.js.map +1 -0
- package/dist/components/templates/nested-secondary-panel-shell.d.ts +20 -0
- package/dist/components/templates/nested-secondary-panel-shell.js +54 -0
- package/dist/components/templates/nested-secondary-panel-shell.js.map +1 -0
- package/dist/components/ui/accordion.d.ts +10 -0
- package/dist/components/ui/accordion.js +74 -0
- package/dist/components/ui/accordion.js.map +1 -0
- package/dist/components/ui/alert-dialog.d.ts +37 -0
- package/dist/components/ui/alert-dialog.js +201 -0
- package/dist/components/ui/alert-dialog.js.map +1 -0
- package/dist/components/ui/avatar.d.ts +84 -0
- package/dist/components/ui/avatar.js +328 -0
- package/dist/components/ui/avatar.js.map +1 -0
- package/dist/components/ui/badge.d.ts +13 -0
- package/dist/components/ui/badge.js +49 -0
- package/dist/components/ui/badge.js.map +1 -0
- package/dist/components/ui/banner.d.ts +62 -0
- package/dist/components/ui/banner.js +364 -0
- package/dist/components/ui/banner.js.map +1 -0
- package/dist/components/ui/breadcrumb.d.ts +14 -0
- package/dist/components/ui/breadcrumb.js +114 -0
- package/dist/components/ui/breadcrumb.js.map +1 -0
- package/dist/components/ui/button.d.ts +16 -0
- package/dist/components/ui/button.js +59 -0
- package/dist/components/ui/button.js.map +1 -0
- package/dist/components/ui/calendar.d.ts +13 -0
- package/dist/components/ui/calendar.js +238 -0
- package/dist/components/ui/calendar.js.map +1 -0
- package/dist/components/ui/card.d.ts +14 -0
- package/dist/components/ui/card.js +102 -0
- package/dist/components/ui/card.js.map +1 -0
- package/dist/components/ui/chart.d.ts +58 -0
- package/dist/components/ui/chart.js +292 -0
- package/dist/components/ui/chart.js.map +1 -0
- package/dist/components/ui/checkbox.d.ts +23 -0
- package/dist/components/ui/checkbox.js +155 -0
- package/dist/components/ui/checkbox.js.map +1 -0
- package/dist/components/ui/coach-mark.d.ts +27 -0
- package/dist/components/ui/coach-mark.js +306 -0
- package/dist/components/ui/coach-mark.js.map +1 -0
- package/dist/components/ui/collapsible.d.ts +8 -0
- package/dist/components/ui/collapsible.js +35 -0
- package/dist/components/ui/collapsible.js.map +1 -0
- package/dist/components/ui/command.d.ts +36 -0
- package/dist/components/ui/command.js +274 -0
- package/dist/components/ui/command.js.map +1 -0
- package/dist/components/ui/context-menu.d.ts +32 -0
- package/dist/components/ui/context-menu.js +245 -0
- package/dist/components/ui/context-menu.js.map +1 -0
- package/dist/components/ui/date-picker-field.d.ts +38 -0
- package/dist/components/ui/date-picker-field.js +550 -0
- package/dist/components/ui/date-picker-field.js.map +1 -0
- package/dist/components/ui/dialog.d.ts +22 -0
- package/dist/components/ui/dialog.js +200 -0
- package/dist/components/ui/dialog.js.map +1 -0
- package/dist/components/ui/dot-pattern.d.ts +21 -0
- package/dist/components/ui/dot-pattern.js +139 -0
- package/dist/components/ui/dot-pattern.js.map +1 -0
- package/dist/components/ui/drag-handle-grip.d.ts +10 -0
- package/dist/components/ui/drag-handle-grip.js +15 -0
- package/dist/components/ui/drag-handle-grip.js.map +1 -0
- package/dist/components/ui/drawer.d.ts +16 -0
- package/dist/components/ui/drawer.js +125 -0
- package/dist/components/ui/drawer.js.map +1 -0
- package/dist/components/ui/dropdown-menu.d.ts +45 -0
- package/dist/components/ui/dropdown-menu.js +353 -0
- package/dist/components/ui/dropdown-menu.js.map +1 -0
- package/dist/components/ui/export-drawer.d.ts +11 -0
- package/dist/components/ui/export-drawer.js +1658 -0
- package/dist/components/ui/export-drawer.js.map +1 -0
- package/dist/components/ui/field.d.ts +30 -0
- package/dist/components/ui/field.js +249 -0
- package/dist/components/ui/field.js.map +1 -0
- package/dist/components/ui/form.d.ts +28 -0
- package/dist/components/ui/form.js +110 -0
- package/dist/components/ui/form.js.map +1 -0
- package/dist/components/ui/hover-card.d.ts +9 -0
- package/dist/components/ui/hover-card.js +43 -0
- package/dist/components/ui/hover-card.js.map +1 -0
- package/dist/components/ui/input-group.d.ts +20 -0
- package/dist/components/ui/input-group.js +219 -0
- package/dist/components/ui/input-group.js.map +1 -0
- package/dist/components/ui/input-mask.d.ts +39 -0
- package/dist/components/ui/input-mask.js +118 -0
- package/dist/components/ui/input-mask.js.map +1 -0
- package/dist/components/ui/input.d.ts +5 -0
- package/dist/components/ui/input.js +30 -0
- package/dist/components/ui/input.js.map +1 -0
- package/dist/components/ui/kbd.d.ts +20 -0
- package/dist/components/ui/kbd.js +45 -0
- package/dist/components/ui/kbd.js.map +1 -0
- package/dist/components/ui/key-metrics-context.d.ts +19 -0
- package/dist/components/ui/key-metrics-context.js +26 -0
- package/dist/components/ui/key-metrics-context.js.map +1 -0
- package/dist/components/ui/key-metrics.d.ts +131 -0
- package/dist/components/ui/key-metrics.js +1015 -0
- package/dist/components/ui/key-metrics.js.map +1 -0
- package/dist/components/ui/label.d.ts +6 -0
- package/dist/components/ui/label.js +28 -0
- package/dist/components/ui/label.js.map +1 -0
- package/dist/components/ui/list-page-view-frame.d.ts +22 -0
- package/dist/components/ui/list-page-view-frame.js +24 -0
- package/dist/components/ui/list-page-view-frame.js.map +1 -0
- package/dist/components/ui/page-header.d.ts +51 -0
- package/dist/components/ui/page-header.js +372 -0
- package/dist/components/ui/page-header.js.map +1 -0
- package/dist/components/ui/payment-card-fields.d.ts +10 -0
- package/dist/components/ui/payment-card-fields.js +80 -0
- package/dist/components/ui/payment-card-fields.js.map +1 -0
- package/dist/components/ui/popover.d.ts +10 -0
- package/dist/components/ui/popover.js +47 -0
- package/dist/components/ui/popover.js.map +1 -0
- package/dist/components/ui/radio-group.d.ts +29 -0
- package/dist/components/ui/radio-group.js +190 -0
- package/dist/components/ui/radio-group.js.map +1 -0
- package/dist/components/ui/resizable.d.ts +16 -0
- package/dist/components/ui/resizable.js +51 -0
- package/dist/components/ui/resizable.js.map +1 -0
- package/dist/components/ui/scroll-area.d.ts +8 -0
- package/dist/components/ui/scroll-area.js +66 -0
- package/dist/components/ui/scroll-area.js.map +1 -0
- package/dist/components/ui/select.d.ts +18 -0
- package/dist/components/ui/select.js +186 -0
- package/dist/components/ui/select.js.map +1 -0
- package/dist/components/ui/selection-tile-grid.d.ts +52 -0
- package/dist/components/ui/selection-tile-grid.js +347 -0
- package/dist/components/ui/selection-tile-grid.js.map +1 -0
- package/dist/components/ui/separator.d.ts +7 -0
- package/dist/components/ui/separator.js +33 -0
- package/dist/components/ui/separator.js.map +1 -0
- package/dist/components/ui/sheet.d.ts +18 -0
- package/dist/components/ui/sheet.js +181 -0
- package/dist/components/ui/sheet.js.map +1 -0
- package/dist/components/ui/sidebar.d.ts +94 -0
- package/dist/components/ui/sidebar.js +805 -0
- package/dist/components/ui/sidebar.js.map +1 -0
- package/dist/components/ui/skeleton.d.ts +5 -0
- package/dist/components/ui/skeleton.js +22 -0
- package/dist/components/ui/skeleton.js.map +1 -0
- package/dist/components/ui/slider.d.ts +7 -0
- package/dist/components/ui/slider.js +66 -0
- package/dist/components/ui/slider.js.map +1 -0
- package/dist/components/ui/sonner.d.ts +6 -0
- package/dist/components/ui/sonner.js +38 -0
- package/dist/components/ui/sonner.js.map +1 -0
- package/dist/components/ui/status-badge.d.ts +38 -0
- package/dist/components/ui/status-badge.js +77 -0
- package/dist/components/ui/status-badge.js.map +1 -0
- package/dist/components/ui/table.d.ts +13 -0
- package/dist/components/ui/table.js +115 -0
- package/dist/components/ui/table.js.map +1 -0
- package/dist/components/ui/tabs.d.ts +15 -0
- package/dist/components/ui/tabs.js +93 -0
- package/dist/components/ui/tabs.js.map +1 -0
- package/dist/components/ui/textarea.d.ts +6 -0
- package/dist/components/ui/textarea.js +25 -0
- package/dist/components/ui/textarea.js.map +1 -0
- package/dist/components/ui/tip.d.ts +12 -0
- package/dist/components/ui/tip.js +61 -0
- package/dist/components/ui/tip.js.map +1 -0
- package/dist/components/ui/toggle-group.d.ts +14 -0
- package/dist/components/ui/toggle-group.js +104 -0
- package/dist/components/ui/toggle-group.js.map +1 -0
- package/dist/components/ui/toggle-switch.d.ts +10 -0
- package/dist/components/ui/toggle-switch.js +33 -0
- package/dist/components/ui/toggle-switch.js.map +1 -0
- package/dist/components/ui/toggle.d.ts +13 -0
- package/dist/components/ui/toggle.js +51 -0
- package/dist/components/ui/toggle.js.map +1 -0
- package/dist/components/ui/tooltip.d.ts +10 -0
- package/dist/components/ui/tooltip.js +68 -0
- package/dist/components/ui/tooltip.js.map +1 -0
- package/dist/components/ui/view-segmented-control.d.ts +31 -0
- package/dist/components/ui/view-segmented-control.js +167 -0
- package/dist/components/ui/view-segmented-control.js.map +1 -0
- package/dist/data-list-view-registry-CyBoBML4.d.ts +73 -0
- package/dist/hooks/use-app-theme.d.ts +24 -0
- package/dist/hooks/use-app-theme.js +286 -0
- package/dist/hooks/use-app-theme.js.map +1 -0
- package/dist/hooks/use-coach-mark.d.ts +86 -0
- package/dist/hooks/use-coach-mark.js +218 -0
- package/dist/hooks/use-coach-mark.js.map +1 -0
- package/dist/hooks/use-mobile.d.ts +3 -0
- package/dist/hooks/use-mobile.js +29 -0
- package/dist/hooks/use-mobile.js.map +1 -0
- package/dist/hooks/use-mod-key-label.d.ts +6 -0
- package/dist/hooks/use-mod-key-label.js +25 -0
- package/dist/hooks/use-mod-key-label.js.map +1 -0
- package/dist/index.d.ts +120 -0
- package/dist/index.js +13324 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/compose-refs.d.ts +6 -0
- package/dist/lib/compose-refs.js +17 -0
- package/dist/lib/compose-refs.js.map +1 -0
- package/dist/lib/conditional-rule-match.d.ts +30 -0
- package/dist/lib/conditional-rule-match.js +66 -0
- package/dist/lib/conditional-rule-match.js.map +1 -0
- package/dist/lib/data-list-display-options.d.ts +26 -0
- package/dist/lib/data-list-display-options.js +14 -0
- package/dist/lib/data-list-display-options.js.map +1 -0
- package/dist/lib/data-list-view-registry.d.ts +2 -0
- package/dist/lib/data-list-view-registry.js +102 -0
- package/dist/lib/data-list-view-registry.js.map +1 -0
- package/dist/lib/data-list-view-surface.d.ts +2 -0
- package/dist/lib/data-list-view-surface.js +80 -0
- package/dist/lib/data-list-view-surface.js.map +1 -0
- package/dist/lib/data-list-view.d.ts +21 -0
- package/dist/lib/data-list-view.js +25 -0
- package/dist/lib/data-list-view.js.map +1 -0
- package/dist/lib/date-filter.d.ts +22 -0
- package/dist/lib/date-filter.js +61 -0
- package/dist/lib/date-filter.js.map +1 -0
- package/dist/lib/dev-log.d.ts +8 -0
- package/dist/lib/dev-log.js +10 -0
- package/dist/lib/dev-log.js.map +1 -0
- package/dist/lib/dropdown-menu-surface.d.ts +14 -0
- package/dist/lib/dropdown-menu-surface.js +6 -0
- package/dist/lib/dropdown-menu-surface.js.map +1 -0
- package/dist/lib/editable-target.d.ts +12 -0
- package/dist/lib/editable-target.js +12 -0
- package/dist/lib/editable-target.js.map +1 -0
- package/dist/lib/list-page-table-properties.d.ts +35 -0
- package/dist/lib/list-page-table-properties.js +81 -0
- package/dist/lib/list-page-table-properties.js.map +1 -0
- package/dist/lib/raf-throttle.d.ts +23 -0
- package/dist/lib/raf-throttle.js +27 -0
- package/dist/lib/raf-throttle.js.map +1 -0
- package/dist/lib/row-height.d.ts +16 -0
- package/dist/lib/row-height.js +10 -0
- package/dist/lib/row-height.js.map +1 -0
- package/dist/lib/table-properties-types.d.ts +83 -0
- package/dist/lib/table-properties-types.js +19 -0
- package/dist/lib/table-properties-types.js.map +1 -0
- package/dist/lib/utils.d.ts +5 -0
- package/dist/lib/utils.js +11 -0
- package/dist/lib/utils.js.map +1 -0
- package/package.json +83 -19
- package/src/components/data-table/filter-date-calendar.tsx +38 -0
- package/src/components/data-table/filter-text-value-input.tsx +77 -0
- package/src/components/data-table/index.tsx +1678 -0
- package/src/components/data-table/pagination.tsx +255 -0
- package/src/components/data-table/types.ts +96 -0
- package/src/components/data-table/use-table-state.ts +767 -0
- package/src/components/data-views/board-card-primitives.tsx +93 -0
- package/src/components/data-views/data-row-list.tsx +183 -0
- package/src/components/data-views/finder-panel-view.tsx +405 -0
- package/src/components/data-views/folder-grid-view.tsx +86 -0
- package/src/components/data-views/hub-table.tsx +498 -0
- package/src/components/data-views/index.ts +28 -0
- package/src/components/data-views/list-page-board-card.tsx +192 -0
- package/src/components/data-views/list-page-board-template.tsx +122 -0
- package/src/components/data-views/list-page-connected-view-body.tsx +66 -0
- package/src/components/data-views/list-page-split-details-placeholder.tsx +39 -0
- package/src/components/data-views/list-page-split-hub-chrome.tsx +60 -0
- package/src/components/data-views/list-page-split-hub-tokens.ts +16 -0
- package/src/components/data-views/list-page-tree-column-header.tsx +31 -0
- package/src/components/data-views/list-page-tree-panel-shell.tsx +91 -0
- package/src/components/data-views/os-folder-glyph.tsx +141 -0
- package/src/components/data-views/outline-tree-menu.tsx +157 -0
- package/src/components/table-properties/column-row.tsx +90 -0
- package/src/components/table-properties/draggable-list.ts +54 -0
- package/src/components/table-properties/drawer-button.tsx +300 -0
- package/src/components/table-properties/drawer.tsx +1148 -0
- package/src/components/table-properties/filter-card.tsx +251 -0
- package/src/components/table-properties/index.ts +36 -0
- package/src/components/table-properties/sort-card.tsx +63 -0
- package/src/components/templates/dedicated-search-landing-template.tsx +124 -0
- package/src/components/templates/dedicated-search-results-template.tsx +19 -0
- package/src/components/templates/index.ts +33 -0
- package/src/components/templates/list-page.tsx +602 -0
- package/src/components/templates/nested-secondary-panel-shell.tsx +70 -0
- package/src/components/ui/accordion.tsx +92 -0
- package/src/components/ui/alert-dialog.tsx +221 -0
- package/src/components/ui/avatar.tsx +13 -2
- package/src/components/ui/banner.tsx +2 -2
- package/src/components/ui/button.tsx +4 -4
- package/src/components/ui/calendar.tsx +1 -1
- package/src/components/ui/coach-mark.tsx +1 -1
- package/src/components/ui/context-menu.tsx +291 -0
- package/src/components/ui/date-picker-field.tsx +2 -2
- package/src/components/ui/dot-pattern.tsx +183 -0
- package/src/components/ui/export-drawer.tsx +375 -0
- package/src/components/ui/hover-card.tsx +66 -0
- package/src/components/ui/key-metrics-context.tsx +78 -0
- package/src/components/ui/key-metrics.tsx +1133 -0
- package/src/components/ui/list-page-view-frame.tsx +64 -0
- package/src/components/ui/page-header.tsx +244 -0
- package/src/components/ui/payment-card-fields.tsx +2 -2
- package/src/components/ui/resizable.tsx +68 -0
- package/src/components/ui/scroll-area.tsx +72 -0
- package/src/components/ui/selection-tile-grid.tsx +9 -2
- package/src/components/ui/sidebar.tsx +84 -12
- package/src/components/ui/slider.tsx +83 -0
- package/src/globals.css +2201 -7
- package/src/globals.d.ts +20 -0
- package/src/index.ts +68 -1
- package/src/lib/conditional-rule-match.ts +119 -0
- package/src/lib/data-list-display-options.ts +35 -0
- package/src/lib/data-list-view-registry.ts +104 -0
- package/src/lib/data-list-view-surface.ts +83 -0
- package/src/lib/data-list-view.ts +47 -0
- package/src/lib/dev-log.ts +10 -0
- package/src/lib/editable-target.ts +20 -0
- package/src/lib/list-page-table-properties.ts +48 -0
- package/src/lib/raf-throttle.ts +45 -0
- package/src/lib/row-height.ts +19 -0
- package/src/lib/table-properties-types.ts +98 -0
- package/template/.cursor/rules/exxat-command-menu.mdc +1 -1
- package/template/.cursor/rules/exxat-dashboard-view-charts.mdc +3 -3
- package/template/.cursor/rules/exxat-data-tables.mdc +1 -1
- package/template/.cursor/rules/exxat-ds-agents.mdc +2 -2
- package/template/.cursor/rules/exxat-kbd-shortcuts.mdc +2 -2
- package/template/.cursor/rules/exxat-table-properties-drawer.mdc +1 -1
- package/template/AGENTS.md +104 -78
- package/template/app/(app)/dashboard/loading.tsx +15 -3
- package/template/app/(app)/dashboard/page.tsx +14 -2
- package/template/app/(app)/examples/page.tsx +0 -2
- package/template/app/(app)/layout.tsx +17 -4
- package/template/app/(app)/loading.tsx +18 -1
- package/template/app/(app)/question-bank/find/page.tsx +1 -2
- package/template/app/(app)/question-bank/layout.tsx +1 -1
- package/template/app/(app)/question-bank/library/page.tsx +1 -2
- package/template/app/(app)/question-bank/list/page.tsx +1 -2
- package/template/app/(app)/question-bank/new/page.tsx +15 -20
- package/template/app/(app)/question-bank/page.tsx +1 -2
- package/template/app/(app)/settings/page.tsx +5 -4
- package/template/app/globals.css +14 -16
- package/template/components/ask-leo-sidebar.tsx +5 -1
- package/template/components/brand-color-picker.tsx +2 -2
- package/template/components/charts-overview.tsx +1 -1
- package/template/components/compliance-board-view.tsx +142 -0
- package/template/components/compliance-client.tsx +92 -0
- package/template/components/compliance-page-header.tsx +89 -0
- package/template/components/compliance-table.tsx +468 -0
- package/template/components/dashboard-report-charts.tsx +1 -1
- package/template/components/dashboard-tabs.tsx +1 -1
- package/template/components/data-table/filter-date-calendar.tsx +1 -38
- package/template/components/data-table/filter-text-value-input.tsx +1 -77
- package/template/components/data-table/index.tsx +1 -1634
- package/template/components/data-table/pagination.tsx +1 -255
- package/template/components/data-table/types.ts +1 -94
- package/template/components/data-table/use-table-state.test.ts +420 -0
- package/template/components/data-table/use-table-state.ts +1 -758
- package/template/components/data-view-dashboard-charts-compliance.tsx +963 -0
- package/template/components/data-view-dashboard-charts-team.tsx +971 -0
- package/template/components/data-view-dashboard-charts.tsx +1503 -0
- package/template/components/data-views/board-card-primitives.tsx +1 -93
- package/template/components/data-views/data-row-list.tsx +1 -183
- package/template/components/data-views/finder-panel-view.tsx +1 -405
- package/template/components/data-views/folder-grid-view.tsx +1 -86
- package/template/components/data-views/hub-table.tsx +1 -0
- package/template/components/data-views/index.ts +50 -37
- package/template/components/data-views/list-page-board-card.tsx +1 -192
- package/template/components/data-views/list-page-board-template.tsx +1 -122
- package/template/components/data-views/list-page-connected-view-body.tsx +1 -66
- package/template/components/data-views/list-page-split-details-placeholder.tsx +1 -39
- package/template/components/data-views/list-page-split-hub-chrome.tsx +1 -68
- package/template/components/data-views/list-page-split-hub-tokens.ts +1 -16
- package/template/components/data-views/list-page-tree-column-header.tsx +1 -31
- package/template/components/data-views/list-page-tree-panel-shell.tsx +1 -91
- package/template/components/data-views/list-page-view-frame.tsx +5 -53
- package/template/components/data-views/os-folder-glyph.tsx +1 -129
- package/template/components/data-views/outline-tree-menu.tsx +1 -157
- package/template/components/export-drawer.test.tsx +71 -0
- package/template/components/export-drawer.tsx +1 -375
- package/template/components/exxat-product-logo.tsx +5 -5
- package/template/components/hub-tree-panel-view.tsx +2 -2
- package/template/components/invite-collaborators-drawer.tsx +3 -3
- package/template/components/key-metrics-ask-leo-bridge.tsx +40 -0
- package/template/components/key-metrics.tsx +1 -1063
- package/template/components/leo-insight-indicator.tsx +2 -2
- package/template/components/new-placement-back-btn.tsx +28 -0
- package/template/components/new-placement-form.tsx +942 -0
- package/template/components/new-question-composer.tsx +456 -408
- package/template/components/onboarding/index.ts +9 -0
- package/template/components/onboarding/onboarding-01.tsx +1 -1
- package/template/components/onboarding/onboarding-02.tsx +1 -1
- package/template/components/onboarding/onboarding-03.tsx +1 -1
- package/template/components/onboarding/onboarding-04.tsx +1 -1
- package/template/components/page-header.tsx +8 -226
- package/template/components/placement-board-card.tsx +250 -0
- package/template/components/placement-detail.tsx +438 -0
- package/template/components/placements-board-view.tsx +397 -0
- package/template/components/placements-client.tsx +220 -0
- package/template/components/placements-list-view.tsx +124 -0
- package/template/components/placements-page-header.tsx +166 -0
- package/template/components/placements-table-cells.test.tsx +22 -0
- package/template/components/placements-table-cells.tsx +173 -0
- package/template/components/placements-table-columns.tsx +210 -0
- package/template/components/placements-table.tsx +934 -0
- package/template/components/product-switcher.tsx +3 -4
- package/template/components/product-wordmark.tsx +2 -1
- package/template/components/question-bank-client.tsx +5 -5
- package/template/components/question-bank-hub-client.tsx +1 -1
- package/template/components/question-bank-new-folder-sheet.tsx +2 -2
- package/template/components/question-bank-secondary-nav.tsx +3 -3
- package/template/components/question-bank-table.tsx +541 -431
- package/template/components/rotations-empty-state.tsx +50 -0
- package/template/components/rotations-panel-activator.tsx +8 -0
- package/template/components/settings-appearance-card.tsx +3 -4
- package/template/components/settings-client.tsx +15 -59
- package/template/components/settings-form-row.tsx +4 -9
- package/template/components/{app-sidebar-dynamic.tsx → sidebar/app-sidebar-dynamic.tsx} +1 -1
- package/template/components/{app-sidebar.tsx → sidebar/app-sidebar.tsx} +59 -74
- package/template/components/sidebar/index.ts +16 -0
- package/template/components/{secondary-nav.tsx → sidebar/secondary-nav.tsx} +2 -2
- package/template/components/{secondary-panel.tsx → sidebar/secondary-panel.tsx} +50 -7
- package/template/components/{sidebar-auto-collapse.tsx → sidebar/sidebar-auto-collapse.tsx} +6 -2
- package/template/components/{sidebar-shell.tsx → sidebar/sidebar-shell.tsx} +1 -1
- package/template/components/site-header.tsx +1 -1
- package/template/components/sites-board-view.tsx +67 -0
- package/template/components/sites-client.tsx +154 -0
- package/template/components/sites-table.tsx +249 -0
- package/template/components/table-properties/column-row.tsx +1 -90
- package/template/components/table-properties/draggable-list.ts +1 -49
- package/template/components/table-properties/drawer-button.tsx +1 -262
- package/template/components/table-properties/drawer.tsx +1 -1166
- package/template/components/table-properties/filter-card.tsx +1 -251
- package/template/components/table-properties/sort-card.tsx +1 -59
- package/template/components/table-properties/types.ts +28 -71
- package/template/components/team-board-view.tsx +122 -0
- package/template/components/team-client.tsx +100 -0
- package/template/components/team-page-header.tsx +92 -0
- package/template/components/team-table.tsx +553 -0
- package/template/components/templates/dedicated-search-landing-template.tsx +1 -124
- package/template/components/templates/dedicated-search-results-template.tsx +1 -19
- package/template/components/templates/list-page.tsx +1 -608
- package/template/components/templates/nested-secondary-panel-shell.tsx +1 -63
- package/template/components/templates/new-focus-template.tsx +659 -0
- package/template/components/templates/secondary-panel-hub-template.tsx +1 -1
- package/template/components/ui/accordion.tsx +1 -0
- package/template/components/ui/alert-dialog.tsx +1 -0
- package/template/components/ui/context-menu.tsx +1 -0
- package/template/components/ui/dot-pattern.tsx +1 -183
- package/template/components/ui/hover-card.tsx +1 -0
- package/template/components/ui/resizable.tsx +1 -68
- package/template/components/ui/scroll-area.tsx +1 -0
- package/template/components/ui/slider.tsx +1 -0
- package/template/docs/blueprints/README.md +86 -0
- package/template/docs/blueprints/_template.md +91 -0
- package/template/docs/blueprints/board-card.md +123 -0
- package/template/docs/blueprints/data-table.md +139 -0
- package/template/docs/blueprints/key-metrics.md +128 -0
- package/template/docs/blueprints/list-page-template.md +123 -0
- package/template/docs/blueprints/page-header.md +130 -0
- package/template/docs/command-menu-pattern.md +1 -1
- package/template/docs/component-selection-guide.md +224 -0
- package/template/docs/components-audit-2026-05.md +158 -0
- package/template/docs/data-views-pattern.md +17 -54
- package/template/docs/drawer-vs-dialog-pattern.md +1 -3
- package/template/docs/migrations/0001-brand-deep-alias-stabilization.md +95 -0
- package/template/docs/migrations/0002-exxat-token-namespace.md +154 -0
- package/template/docs/migrations/0003-globals-css-canonical.md +110 -0
- package/template/docs/migrations/README.md +100 -0
- package/template/docs/migrations/_template.md +64 -0
- package/template/docs/shell-surface-elevation-pattern.md +3 -5
- package/template/docs/token-taxonomy.md +416 -0
- package/template/eslint.config.mjs +27 -0
- package/template/hooks/use-secondary-panel-hub-nav.ts +1 -1
- package/template/lib/command-menu-config.ts +0 -1
- package/template/lib/command-menu-search-data.ts +27 -11
- package/template/lib/compliance-supported-views.ts +10 -0
- package/template/lib/conditional-rule-match.ts +6 -97
- package/template/lib/data-list-display-options.ts +1 -49
- package/template/lib/data-list-view-registry.ts +1 -104
- package/template/lib/data-list-view-surface.ts +1 -83
- package/template/lib/data-list-view.ts +1 -47
- package/template/lib/data-view-dashboard-placements-layout.ts +215 -0
- package/template/lib/data-view-dashboard-storage.ts +35 -38
- package/template/lib/dev-log.ts +1 -8
- package/template/lib/editable-target.ts +1 -10
- package/template/lib/list-page-table-properties.ts +1 -48
- package/template/lib/list-status-badges.ts +97 -4
- package/template/lib/mock/compliance-kpi.ts +61 -0
- package/template/lib/mock/compliance.ts +146 -0
- package/template/lib/mock/navigation.tsx +0 -9
- package/template/lib/mock/placements-kpi.ts +134 -0
- package/template/lib/mock/placements.ts +176 -0
- package/template/lib/mock/sites-directory.ts +16 -0
- package/template/lib/mock/sites-kpi.ts +25 -0
- package/template/lib/mock/team-kpi.ts +60 -0
- package/template/lib/mock/team.ts +118 -0
- package/template/lib/placement-board-card-layout.ts +79 -0
- package/template/lib/placements-supported-views.ts +12 -0
- package/template/lib/question-bank-supported-views.ts +0 -1
- package/template/lib/raf-throttle.ts +1 -45
- package/template/lib/row-height.ts +4 -10
- package/template/lib/sidebar-state-cookie.ts +11 -2
- package/template/lib/sites-supported-views.ts +10 -0
- package/template/lib/table-state-lifecycle.ts +2 -2
- package/template/lib/team-supported-views.ts +10 -0
- package/template/package.json +1 -0
- package/template/tests/setup.ts +25 -0
- package/consumer-extras/AGENTS.md +0 -76
- package/consumer-extras/cursor-skills/exxat-consumer-app/SKILL.md +0 -37
- package/consumer-extras/cursor-skills/exxat-focused-workflow-page/SKILL.md +0 -57
- package/consumer-extras/patterns/consumer-app-pattern.md +0 -39
- package/consumer-extras/patterns/focused-workflow-page-pattern.md +0 -84
- package/src/components/ui/button-group.tsx +0 -81
- package/src/theme.css +0 -16
- package/src/tokens/README.md +0 -15
- package/src/tokens/base.css +0 -337
- package/src/tokens/high-contrast.css +0 -1195
- package/src/tokens/layers.css +0 -224
- package/src/tokens/tailwind-bridge.css +0 -118
- package/src/tokens/themes.css +0 -201
- package/template/app/(app)/data-list/layout.tsx +0 -43
- package/template/app/(app)/data-list/page.tsx +0 -10
- package/template/app/(app)/examples/focused-workflow/page.tsx +0 -5
- package/template/components/app-route-loading.tsx +0 -14
- package/template/components/dashboard-onboarding-gallery.tsx +0 -13
- package/template/components/dashboard-onboarding.tsx +0 -21
- package/template/components/data-views/list-page-calendar-view.tsx +0 -593
- package/template/components/data-views/list-page-folder-columns-panel.tsx +0 -345
- package/template/components/examples/focused-workflow-showcase.tsx +0 -183
- package/template/components/list-hub-board-view.tsx +0 -68
- package/template/components/list-hub-client.tsx +0 -186
- package/template/components/list-hub-list-view.tsx +0 -36
- package/template/components/list-hub-panel-activator.tsx +0 -8
- package/template/components/list-hub-secondary-nav.tsx +0 -121
- package/template/components/list-hub-table.tsx +0 -336
- package/template/components/question-bank-folder-columns-panel.tsx +0 -104
- package/template/components/question-bank-list-view.tsx +0 -53
- package/template/components/secondary-panel/nav-link-rows.tsx +0 -83
- package/template/components/secondary-panels/list-hub-panel.tsx +0 -39
- package/template/components/secondary-panels/question-bank-panel.tsx +0 -39
- package/template/components/secondary-panels/registry.tsx +0 -15
- package/template/components/section-cards.tsx +0 -106
- package/template/components/templates/focused-workflow-layouts.tsx +0 -448
- package/template/components/templates/focused-workflow-page-template.tsx +0 -69
- package/template/components/templates/page-loading-shell.tsx +0 -262
- package/template/components/ui/button-group.tsx +0 -1
- package/template/docs/consumer-app-pattern.md +0 -39
- package/template/docs/focused-workflow-page-pattern.md +0 -84
- package/template/lib/list-hub-nav.ts +0 -121
- package/template/lib/mock/list-hub-directory.ts +0 -27
- package/template/lib/mock/list-hub-kpi.ts +0 -27
- package/template/lib/page-loading-variant.ts +0 -40
- /package/template/components/{getting-started.tsx → onboarding/getting-started.tsx} +0 -0
- /package/template/components/{nav-documents.tsx → sidebar/nav-documents.tsx} +0 -0
- /package/template/components/{nav-main.tsx → sidebar/nav-main.tsx} +0 -0
- /package/template/components/{nav-secondary.tsx → sidebar/nav-secondary.tsx} +0 -0
- /package/template/components/{nav-user.tsx → sidebar/nav-user.tsx} +0 -0
- /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/
|
|
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 {
|
|
6
|
+
import { ALL_PLACEMENTS } from "@/lib/mock/placements"
|
|
7
7
|
|
|
8
8
|
function sampleRowSearchItems(): CommandMenuItem[] {
|
|
9
|
-
return
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
.
|
|
16
|
-
|
|
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
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
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"
|