@qijenchen/design-system 0.1.0-beta.10
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/README.md +163 -0
- package/dist/components/Accordion/accordion.d.ts +37 -0
- package/dist/components/Accordion/accordion.d.ts.map +1 -0
- package/dist/components/Accordion/accordion.js +78 -0
- package/dist/components/Accordion/accordion.js.map +1 -0
- package/dist/components/Alert/alert.d.ts +47 -0
- package/dist/components/Alert/alert.d.ts.map +1 -0
- package/dist/components/Alert/alert.js +132 -0
- package/dist/components/Alert/alert.js.map +1 -0
- package/dist/components/AppShell/_demo-helpers.d.ts +49 -0
- package/dist/components/AppShell/_demo-helpers.d.ts.map +1 -0
- package/dist/components/AppShell/app-shell.d.ts +76 -0
- package/dist/components/AppShell/app-shell.d.ts.map +1 -0
- package/dist/components/AppShell/app-shell.js +214 -0
- package/dist/components/AppShell/app-shell.js.map +1 -0
- package/dist/components/AspectRatio/aspect-ratio.d.ts +40 -0
- package/dist/components/AspectRatio/aspect-ratio.d.ts.map +1 -0
- package/dist/components/AspectRatio/aspect-ratio.js +23 -0
- package/dist/components/AspectRatio/aspect-ratio.js.map +1 -0
- package/dist/components/Avatar/avatar.d.ts +85 -0
- package/dist/components/Avatar/avatar.d.ts.map +1 -0
- package/dist/components/Avatar/avatar.js +195 -0
- package/dist/components/Avatar/avatar.js.map +1 -0
- package/dist/components/Badge/badge.d.ts +43 -0
- package/dist/components/Badge/badge.d.ts.map +1 -0
- package/dist/components/Badge/badge.js +69 -0
- package/dist/components/Badge/badge.js.map +1 -0
- package/dist/components/Breadcrumb/breadcrumb.d.ts +163 -0
- package/dist/components/Breadcrumb/breadcrumb.d.ts.map +1 -0
- package/dist/components/Breadcrumb/breadcrumb.js +300 -0
- package/dist/components/Breadcrumb/breadcrumb.js.map +1 -0
- package/dist/components/BulkActionBar/bulk-action-bar.d.ts +46 -0
- package/dist/components/BulkActionBar/bulk-action-bar.d.ts.map +1 -0
- package/dist/components/BulkActionBar/bulk-action-bar.js +78 -0
- package/dist/components/BulkActionBar/bulk-action-bar.js.map +1 -0
- package/dist/components/Button/button-group.d.ts +49 -0
- package/dist/components/Button/button-group.d.ts.map +1 -0
- package/dist/components/Button/button-group.js +46 -0
- package/dist/components/Button/button-group.js.map +1 -0
- package/dist/components/Button/button.d.ts +203 -0
- package/dist/components/Button/button.d.ts.map +1 -0
- package/dist/components/Button/button.js +309 -0
- package/dist/components/Button/button.js.map +1 -0
- package/dist/components/Calendar/calendar.d.ts +81 -0
- package/dist/components/Calendar/calendar.d.ts.map +1 -0
- package/dist/components/Calendar/calendar.js +282 -0
- package/dist/components/Calendar/calendar.js.map +1 -0
- package/dist/components/Carousel/carousel.d.ts +61 -0
- package/dist/components/Carousel/carousel.d.ts.map +1 -0
- package/dist/components/Carousel/carousel.js +276 -0
- package/dist/components/Carousel/carousel.js.map +1 -0
- package/dist/components/Chart/chart.d.ts +94 -0
- package/dist/components/Chart/chart.d.ts.map +1 -0
- package/dist/components/Chart/chart.js +233 -0
- package/dist/components/Chart/chart.js.map +1 -0
- package/dist/components/Checkbox/checkbox-group.d.ts +58 -0
- package/dist/components/Checkbox/checkbox-group.d.ts.map +1 -0
- package/dist/components/Checkbox/checkbox-group.js +28 -0
- package/dist/components/Checkbox/checkbox-group.js.map +1 -0
- package/dist/components/Checkbox/checkbox.d.ts +73 -0
- package/dist/components/Checkbox/checkbox.d.ts.map +1 -0
- package/dist/components/Checkbox/checkbox.js +125 -0
- package/dist/components/Checkbox/checkbox.js.map +1 -0
- package/dist/components/Chip/chip.d.ts +54 -0
- package/dist/components/Chip/chip.d.ts.map +1 -0
- package/dist/components/Chip/chip.js +224 -0
- package/dist/components/Chip/chip.js.map +1 -0
- package/dist/components/CircularProgress/circular-progress.d.ts +40 -0
- package/dist/components/CircularProgress/circular-progress.d.ts.map +1 -0
- package/dist/components/CircularProgress/circular-progress.js +118 -0
- package/dist/components/CircularProgress/circular-progress.js.map +1 -0
- package/dist/components/Coachmark/coachmark.d.ts +100 -0
- package/dist/components/Coachmark/coachmark.d.ts.map +1 -0
- package/dist/components/Coachmark/coachmark.js +107 -0
- package/dist/components/Coachmark/coachmark.js.map +1 -0
- package/dist/components/Combobox/combobox.d.ts +150 -0
- package/dist/components/Combobox/combobox.d.ts.map +1 -0
- package/dist/components/Combobox/combobox.js +595 -0
- package/dist/components/Combobox/combobox.js.map +1 -0
- package/dist/components/Command/command.d.ts +106 -0
- package/dist/components/Command/command.d.ts.map +1 -0
- package/dist/components/Command/command.js +123 -0
- package/dist/components/Command/command.js.map +1 -0
- package/dist/components/DataTable/active-editor-controller.d.ts +66 -0
- package/dist/components/DataTable/active-editor-controller.d.ts.map +1 -0
- package/dist/components/DataTable/cell-registry.d.ts +37 -0
- package/dist/components/DataTable/cell-registry.d.ts.map +1 -0
- package/dist/components/DataTable/cell-registry.js +377 -0
- package/dist/components/DataTable/cell-registry.js.map +1 -0
- package/dist/components/DataTable/column-types.d.ts +145 -0
- package/dist/components/DataTable/column-types.d.ts.map +1 -0
- package/dist/components/DataTable/column-types.js +17 -0
- package/dist/components/DataTable/column-types.js.map +1 -0
- package/dist/components/DataTable/data-table-column-visibility-panel.d.ts +49 -0
- package/dist/components/DataTable/data-table-column-visibility-panel.d.ts.map +1 -0
- package/dist/components/DataTable/data-table-filter-panel.d.ts +30 -0
- package/dist/components/DataTable/data-table-filter-panel.d.ts.map +1 -0
- package/dist/components/DataTable/data-table-interaction-layer.d.ts +78 -0
- package/dist/components/DataTable/data-table-interaction-layer.d.ts.map +1 -0
- package/dist/components/DataTable/data-table-interaction-layer.js +220 -0
- package/dist/components/DataTable/data-table-interaction-layer.js.map +1 -0
- package/dist/components/DataTable/data-table-sort-manager.d.ts +19 -0
- package/dist/components/DataTable/data-table-sort-manager.d.ts.map +1 -0
- package/dist/components/DataTable/data-table.d.ts +181 -0
- package/dist/components/DataTable/data-table.d.ts.map +1 -0
- package/dist/components/DataTable/data-table.js +1851 -0
- package/dist/components/DataTable/data-table.js.map +1 -0
- package/dist/components/DataTable/filter-operators.d.ts +116 -0
- package/dist/components/DataTable/filter-operators.d.ts.map +1 -0
- package/dist/components/DataTable/filter-tree.d.ts +66 -0
- package/dist/components/DataTable/filter-tree.d.ts.map +1 -0
- package/dist/components/DataTable/lib/column-meta.d.ts +49 -0
- package/dist/components/DataTable/lib/column-meta.d.ts.map +1 -0
- package/dist/components/DateGrid/date-grid.d.ts +61 -0
- package/dist/components/DateGrid/date-grid.d.ts.map +1 -0
- package/dist/components/DateGrid/date-grid.js +168 -0
- package/dist/components/DateGrid/date-grid.js.map +1 -0
- package/dist/components/DatePicker/date-picker.d.ts +119 -0
- package/dist/components/DatePicker/date-picker.d.ts.map +1 -0
- package/dist/components/DatePicker/date-picker.js +743 -0
- package/dist/components/DatePicker/date-picker.js.map +1 -0
- package/dist/components/DescriptionList/description-list.d.ts +60 -0
- package/dist/components/DescriptionList/description-list.d.ts.map +1 -0
- package/dist/components/DescriptionList/description-list.js +77 -0
- package/dist/components/DescriptionList/description-list.js.map +1 -0
- package/dist/components/Dialog/dialog.d.ts +54 -0
- package/dist/components/Dialog/dialog.d.ts.map +1 -0
- package/dist/components/Dialog/dialog.js +151 -0
- package/dist/components/Dialog/dialog.js.map +1 -0
- package/dist/components/DropdownMenu/dropdown-menu.d.ts +111 -0
- package/dist/components/DropdownMenu/dropdown-menu.d.ts.map +1 -0
- package/dist/components/DropdownMenu/dropdown-menu.js +288 -0
- package/dist/components/DropdownMenu/dropdown-menu.js.map +1 -0
- package/dist/components/Empty/empty.d.ts +40 -0
- package/dist/components/Empty/empty.d.ts.map +1 -0
- package/dist/components/Empty/empty.js +66 -0
- package/dist/components/Empty/empty.js.map +1 -0
- package/dist/components/Field/field-context.d.ts +77 -0
- package/dist/components/Field/field-context.d.ts.map +1 -0
- package/dist/components/Field/field-context.js +37 -0
- package/dist/components/Field/field-context.js.map +1 -0
- package/dist/components/Field/field-types.d.ts +5 -0
- package/dist/components/Field/field-types.d.ts.map +1 -0
- package/dist/components/Field/field-types.js +13 -0
- package/dist/components/Field/field-types.js.map +1 -0
- package/dist/components/Field/field-wrapper.d.ts +17 -0
- package/dist/components/Field/field-wrapper.d.ts.map +1 -0
- package/dist/components/Field/field-wrapper.js +252 -0
- package/dist/components/Field/field-wrapper.js.map +1 -0
- package/dist/components/Field/field.d.ts +127 -0
- package/dist/components/Field/field.d.ts.map +1 -0
- package/dist/components/Field/field.js +295 -0
- package/dist/components/Field/field.js.map +1 -0
- package/dist/components/FieldControlGroup/field-control-group.d.ts +74 -0
- package/dist/components/FieldControlGroup/field-control-group.d.ts.map +1 -0
- package/dist/components/FieldControlGroup/field-control-group.js +62 -0
- package/dist/components/FieldControlGroup/field-control-group.js.map +1 -0
- package/dist/components/FileItem/file-item.d.ts +44 -0
- package/dist/components/FileItem/file-item.d.ts.map +1 -0
- package/dist/components/FileItem/file-item.js +202 -0
- package/dist/components/FileItem/file-item.js.map +1 -0
- package/dist/components/FileUpload/file-upload.d.ts +97 -0
- package/dist/components/FileUpload/file-upload.d.ts.map +1 -0
- package/dist/components/FileUpload/file-upload.js +231 -0
- package/dist/components/FileUpload/file-upload.js.map +1 -0
- package/dist/components/FileViewer/file-viewer-types.d.ts +73 -0
- package/dist/components/FileViewer/file-viewer-types.d.ts.map +1 -0
- package/dist/components/FileViewer/file-viewer.d.ts +82 -0
- package/dist/components/FileViewer/file-viewer.d.ts.map +1 -0
- package/dist/components/FileViewer/file-viewer.js +752 -0
- package/dist/components/FileViewer/file-viewer.js.map +1 -0
- package/dist/components/FileViewer/image-renderer.d.ts +9 -0
- package/dist/components/FileViewer/image-renderer.d.ts.map +1 -0
- package/dist/components/FileViewer/image-renderer.js +165 -0
- package/dist/components/FileViewer/image-renderer.js.map +1 -0
- package/dist/components/HoverCard/hover-card.d.ts +30 -0
- package/dist/components/HoverCard/hover-card.d.ts.map +1 -0
- package/dist/components/HoverCard/hover-card.js +61 -0
- package/dist/components/HoverCard/hover-card.js.map +1 -0
- package/dist/components/Input/input.d.ts +72 -0
- package/dist/components/Input/input.d.ts.map +1 -0
- package/dist/components/Input/input.js +148 -0
- package/dist/components/Input/input.js.map +1 -0
- package/dist/components/LinkInput/link-input.d.ts +46 -0
- package/dist/components/LinkInput/link-input.d.ts.map +1 -0
- package/dist/components/LinkInput/link-input.js +215 -0
- package/dist/components/LinkInput/link-input.js.map +1 -0
- package/dist/components/Menu/menu-item.d.ts +83 -0
- package/dist/components/Menu/menu-item.d.ts.map +1 -0
- package/dist/components/Menu/menu-item.js +209 -0
- package/dist/components/Menu/menu-item.js.map +1 -0
- package/dist/components/NameCard/name-card.d.ts +85 -0
- package/dist/components/NameCard/name-card.d.ts.map +1 -0
- package/dist/components/NameCard/name-card.js +153 -0
- package/dist/components/NameCard/name-card.js.map +1 -0
- package/dist/components/Notice/notice.d.ts +69 -0
- package/dist/components/Notice/notice.d.ts.map +1 -0
- package/dist/components/Notice/notice.js +121 -0
- package/dist/components/Notice/notice.js.map +1 -0
- package/dist/components/NumberInput/number-input.d.ts +57 -0
- package/dist/components/NumberInput/number-input.d.ts.map +1 -0
- package/dist/components/NumberInput/number-input.js +131 -0
- package/dist/components/NumberInput/number-input.js.map +1 -0
- package/dist/components/OverflowIndicator/overflow-indicator.d.ts +23 -0
- package/dist/components/OverflowIndicator/overflow-indicator.d.ts.map +1 -0
- package/dist/components/OverflowIndicator/overflow-indicator.js +111 -0
- package/dist/components/OverflowIndicator/overflow-indicator.js.map +1 -0
- package/dist/components/PeoplePicker/avatar-stack-overflow.d.ts +57 -0
- package/dist/components/PeoplePicker/avatar-stack-overflow.d.ts.map +1 -0
- package/dist/components/PeoplePicker/avatar-stack-overflow.js +35 -0
- package/dist/components/PeoplePicker/avatar-stack-overflow.js.map +1 -0
- package/dist/components/PeoplePicker/people-picker-helpers.d.ts +7 -0
- package/dist/components/PeoplePicker/people-picker-helpers.d.ts.map +1 -0
- package/dist/components/PeoplePicker/people-picker-helpers.js +25 -0
- package/dist/components/PeoplePicker/people-picker-helpers.js.map +1 -0
- package/dist/components/PeoplePicker/people-picker.d.ts +77 -0
- package/dist/components/PeoplePicker/people-picker.d.ts.map +1 -0
- package/dist/components/PeoplePicker/people-picker.js +263 -0
- package/dist/components/PeoplePicker/people-picker.js.map +1 -0
- package/dist/components/PeoplePicker/person-display.d.ts +66 -0
- package/dist/components/PeoplePicker/person-display.d.ts.map +1 -0
- package/dist/components/PeoplePicker/person-display.js +203 -0
- package/dist/components/PeoplePicker/person-display.js.map +1 -0
- package/dist/components/Popover/popover.d.ts +50 -0
- package/dist/components/Popover/popover.d.ts.map +1 -0
- package/dist/components/Popover/popover.js +113 -0
- package/dist/components/Popover/popover.js.map +1 -0
- package/dist/components/ProgressBar/progress-bar.d.ts +37 -0
- package/dist/components/ProgressBar/progress-bar.d.ts.map +1 -0
- package/dist/components/ProgressBar/progress-bar.js +86 -0
- package/dist/components/ProgressBar/progress-bar.js.map +1 -0
- package/dist/components/RadioGroup/radio-group.d.ts +78 -0
- package/dist/components/RadioGroup/radio-group.d.ts.map +1 -0
- package/dist/components/RadioGroup/radio-group.js +153 -0
- package/dist/components/RadioGroup/radio-group.js.map +1 -0
- package/dist/components/Rating/rating.d.ts +46 -0
- package/dist/components/Rating/rating.d.ts.map +1 -0
- package/dist/components/Rating/rating.js +179 -0
- package/dist/components/Rating/rating.js.map +1 -0
- package/dist/components/ScrollArea/scroll-area.d.ts +45 -0
- package/dist/components/ScrollArea/scroll-area.d.ts.map +1 -0
- package/dist/components/ScrollArea/scroll-area.js +65 -0
- package/dist/components/ScrollArea/scroll-area.js.map +1 -0
- package/dist/components/SegmentedControl/segmented-control.d.ts +102 -0
- package/dist/components/SegmentedControl/segmented-control.d.ts.map +1 -0
- package/dist/components/SegmentedControl/segmented-control.js +171 -0
- package/dist/components/SegmentedControl/segmented-control.js.map +1 -0
- package/dist/components/Select/select.d.ts +102 -0
- package/dist/components/Select/select.d.ts.map +1 -0
- package/dist/components/Select/select.js +435 -0
- package/dist/components/Select/select.js.map +1 -0
- package/dist/components/SelectMenu/select-menu.d.ts +103 -0
- package/dist/components/SelectMenu/select-menu.d.ts.map +1 -0
- package/dist/components/SelectMenu/select-menu.js +239 -0
- package/dist/components/SelectMenu/select-menu.js.map +1 -0
- package/dist/components/SelectionControl/selection-item.d.ts +69 -0
- package/dist/components/SelectionControl/selection-item.d.ts.map +1 -0
- package/dist/components/SelectionControl/selection-item.js +142 -0
- package/dist/components/SelectionControl/selection-item.js.map +1 -0
- package/dist/components/Separator/separator.d.ts +17 -0
- package/dist/components/Separator/separator.d.ts.map +1 -0
- package/dist/components/Separator/separator.js +39 -0
- package/dist/components/Separator/separator.js.map +1 -0
- package/dist/components/Sheet/sheet.d.ts +56 -0
- package/dist/components/Sheet/sheet.d.ts.map +1 -0
- package/dist/components/Sheet/sheet.js +145 -0
- package/dist/components/Sheet/sheet.js.map +1 -0
- package/dist/components/Sidebar/sidebar.d.ts +195 -0
- package/dist/components/Sidebar/sidebar.d.ts.map +1 -0
- package/dist/components/Sidebar/sidebar.js +826 -0
- package/dist/components/Sidebar/sidebar.js.map +1 -0
- package/dist/components/Skeleton/skeleton.d.ts +16 -0
- package/dist/components/Skeleton/skeleton.d.ts.map +1 -0
- package/dist/components/Skeleton/skeleton.js +30 -0
- package/dist/components/Skeleton/skeleton.js.map +1 -0
- package/dist/components/Slider/slider.d.ts +48 -0
- package/dist/components/Slider/slider.d.ts.map +1 -0
- package/dist/components/Slider/slider.js +108 -0
- package/dist/components/Slider/slider.js.map +1 -0
- package/dist/components/Steps/steps.d.ts +71 -0
- package/dist/components/Steps/steps.d.ts.map +1 -0
- package/dist/components/Steps/steps.js +583 -0
- package/dist/components/Steps/steps.js.map +1 -0
- package/dist/components/Switch/switch.d.ts +112 -0
- package/dist/components/Switch/switch.d.ts.map +1 -0
- package/dist/components/Switch/switch.js +179 -0
- package/dist/components/Switch/switch.js.map +1 -0
- package/dist/components/Tabs/tabs.d.ts +104 -0
- package/dist/components/Tabs/tabs.d.ts.map +1 -0
- package/dist/components/Tabs/tabs.js +316 -0
- package/dist/components/Tabs/tabs.js.map +1 -0
- package/dist/components/Tag/tag.d.ts +86 -0
- package/dist/components/Tag/tag.d.ts.map +1 -0
- package/dist/components/Tag/tag.js +172 -0
- package/dist/components/Tag/tag.js.map +1 -0
- package/dist/components/Textarea/textarea.d.ts +74 -0
- package/dist/components/Textarea/textarea.d.ts.map +1 -0
- package/dist/components/Textarea/textarea.js +224 -0
- package/dist/components/Textarea/textarea.js.map +1 -0
- package/dist/components/TimePicker/time-columns.d.ts +46 -0
- package/dist/components/TimePicker/time-columns.d.ts.map +1 -0
- package/dist/components/TimePicker/time-columns.js +173 -0
- package/dist/components/TimePicker/time-columns.js.map +1 -0
- package/dist/components/TimePicker/time-picker.d.ts +94 -0
- package/dist/components/TimePicker/time-picker.d.ts.map +1 -0
- package/dist/components/TimePicker/time-picker.js +253 -0
- package/dist/components/TimePicker/time-picker.js.map +1 -0
- package/dist/components/Toast/toast.d.ts +61 -0
- package/dist/components/Toast/toast.d.ts.map +1 -0
- package/dist/components/Toast/toast.js +76 -0
- package/dist/components/Toast/toast.js.map +1 -0
- package/dist/components/Tooltip/tooltip.d.ts +20 -0
- package/dist/components/Tooltip/tooltip.d.ts.map +1 -0
- package/dist/components/Tooltip/tooltip.js +53 -0
- package/dist/components/Tooltip/tooltip.js.map +1 -0
- package/dist/components/TreeView/tree-view.d.ts +166 -0
- package/dist/components/TreeView/tree-view.d.ts.map +1 -0
- package/dist/components/TreeView/tree-view.js +617 -0
- package/dist/components/TreeView/tree-view.js.map +1 -0
- package/dist/hooks/use-controllable.d.ts +16 -0
- package/dist/hooks/use-controllable.d.ts.map +1 -0
- package/dist/hooks/use-controllable.js +26 -0
- package/dist/hooks/use-controllable.js.map +1 -0
- package/dist/hooks/use-is-narrow-viewport.d.ts +2 -0
- package/dist/hooks/use-is-narrow-viewport.d.ts.map +1 -0
- package/dist/hooks/use-is-narrow-viewport.js +19 -0
- package/dist/hooks/use-is-narrow-viewport.js.map +1 -0
- package/dist/hooks/use-is-touch-device.d.ts +8 -0
- package/dist/hooks/use-is-touch-device.d.ts.map +1 -0
- package/dist/hooks/use-is-touch-device.js +16 -0
- package/dist/hooks/use-is-touch-device.js.map +1 -0
- package/dist/hooks/use-overflow-items.d.ts +124 -0
- package/dist/hooks/use-overflow-items.d.ts.map +1 -0
- package/dist/hooks/use-overflow-items.js +97 -0
- package/dist/hooks/use-overflow-items.js.map +1 -0
- package/dist/index.d.ts +74 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +371 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/drag-visual.d.ts +158 -0
- package/dist/lib/drag-visual.d.ts.map +1 -0
- package/dist/lib/drag-visual.js +96 -0
- package/dist/lib/drag-visual.js.map +1 -0
- package/dist/lib/i18n/i18n-context.d.ts +105 -0
- package/dist/lib/i18n/i18n-context.d.ts.map +1 -0
- package/dist/lib/multi-select-ordering.d.ts +54 -0
- package/dist/lib/multi-select-ordering.d.ts.map +1 -0
- package/dist/lib/multi-select-ordering.js +13 -0
- package/dist/lib/multi-select-ordering.js.map +1 -0
- package/dist/lib/utils.d.ts +12 -0
- package/dist/lib/utils.d.ts.map +1 -0
- package/dist/lib/utils.js +79 -0
- package/dist/lib/utils.js.map +1 -0
- package/dist/patterns/element-anatomy/item-anatomy.d.ts +370 -0
- package/dist/patterns/element-anatomy/item-anatomy.d.ts.map +1 -0
- package/dist/patterns/element-anatomy/item-anatomy.js +272 -0
- package/dist/patterns/element-anatomy/item-anatomy.js.map +1 -0
- package/dist/patterns/header-canonical/chrome-header.d.ts +80 -0
- package/dist/patterns/header-canonical/chrome-header.d.ts.map +1 -0
- package/dist/patterns/header-canonical/chrome-header.js +75 -0
- package/dist/patterns/header-canonical/chrome-header.js.map +1 -0
- package/dist/patterns/horizontal-overflow/horizontal-overflow.d.ts +101 -0
- package/dist/patterns/horizontal-overflow/horizontal-overflow.d.ts.map +1 -0
- package/dist/patterns/horizontal-overflow/horizontal-overflow.js +105 -0
- package/dist/patterns/horizontal-overflow/horizontal-overflow.js.map +1 -0
- package/dist/patterns/overlay-surface/overlay-surface.d.ts +28 -0
- package/dist/patterns/overlay-surface/overlay-surface.d.ts.map +1 -0
- package/dist/patterns/overlay-surface/overlay-surface.js +85 -0
- package/dist/patterns/overlay-surface/overlay-surface.js.map +1 -0
- package/dist/patterns/resize-handle/resize-handle.d.ts +102 -0
- package/dist/patterns/resize-handle/resize-handle.d.ts.map +1 -0
- package/dist/patterns/resize-handle/resize-handle.js +74 -0
- package/dist/patterns/resize-handle/resize-handle.js.map +1 -0
- package/dist/react-day-picker.css +457 -0
- package/dist/stories-helpers/anatomy/anatomy-utils.d.ts +40 -0
- package/dist/stories-helpers/anatomy/anatomy-utils.d.ts.map +1 -0
- package/dist/tokens/elevation/overlay-geometry.d.ts +12 -0
- package/dist/tokens/elevation/overlay-geometry.d.ts.map +1 -0
- package/dist/tokens/elevation/overlay-geometry.js +7 -0
- package/dist/tokens/elevation/overlay-geometry.js.map +1 -0
- package/dist/tokens/motion/motion.d.ts +15 -0
- package/dist/tokens/motion/motion.d.ts.map +1 -0
- package/dist/tokens/motion/motion.js +9 -0
- package/dist/tokens/motion/motion.js.map +1 -0
- package/dist/tokens/uiSize/icon-size.d.ts +53 -0
- package/dist/tokens/uiSize/icon-size.d.ts.map +1 -0
- package/package.json +92 -0
- package/src/README.md +32 -0
- package/src/components/Accordion/accordion.tsx +104 -0
- package/src/components/Alert/alert.tsx +188 -0
- package/src/components/AppShell/_demo-helpers.tsx +198 -0
- package/src/components/AppShell/app-shell.tsx +364 -0
- package/src/components/AspectRatio/aspect-ratio.tsx +58 -0
- package/src/components/Avatar/avatar.tsx +368 -0
- package/src/components/Badge/badge.tsx +104 -0
- package/src/components/Breadcrumb/breadcrumb.tsx +619 -0
- package/src/components/BulkActionBar/bulk-action-bar.tsx +156 -0
- package/src/components/Button/button-group.tsx +96 -0
- package/src/components/Button/button.tsx +539 -0
- package/src/components/Calendar/calendar.tsx +411 -0
- package/src/components/Carousel/carousel.tsx +371 -0
- package/src/components/Chart/chart.tsx +376 -0
- package/src/components/Checkbox/checkbox-group.tsx +94 -0
- package/src/components/Checkbox/checkbox.tsx +237 -0
- package/src/components/Chip/chip.tsx +359 -0
- package/src/components/CircularProgress/circular-progress.tsx +204 -0
- package/src/components/Coachmark/coachmark.tsx +255 -0
- package/src/components/Combobox/combobox.tsx +826 -0
- package/src/components/Command/command.tsx +187 -0
- package/src/components/DataTable/active-editor-controller.ts +72 -0
- package/src/components/DataTable/cell-registry.tsx +520 -0
- package/src/components/DataTable/column-types.ts +180 -0
- package/src/components/DataTable/data-table-column-visibility-panel.tsx +261 -0
- package/src/components/DataTable/data-table-filter-panel.tsx +813 -0
- package/src/components/DataTable/data-table-interaction-layer.tsx +483 -0
- package/src/components/DataTable/data-table-sort-manager.tsx +210 -0
- package/src/components/DataTable/data-table.css +165 -0
- package/src/components/DataTable/data-table.tsx +2924 -0
- package/src/components/DataTable/filter-operators.ts +225 -0
- package/src/components/DataTable/filter-tree.ts +313 -0
- package/src/components/DataTable/lib/column-meta.ts +79 -0
- package/src/components/DateGrid/date-grid.tsx +209 -0
- package/src/components/DatePicker/date-picker.tsx +1114 -0
- package/src/components/DescriptionList/description-list.tsx +141 -0
- package/src/components/Dialog/dialog.tsx +267 -0
- package/src/components/DropdownMenu/dropdown-menu.tsx +475 -0
- package/src/components/Empty/empty.tsx +108 -0
- package/src/components/Field/field-context.ts +136 -0
- package/src/components/Field/field-types.ts +52 -0
- package/src/components/Field/field-wrapper.tsx +348 -0
- package/src/components/Field/field.tsx +535 -0
- package/src/components/FieldControlGroup/field-control-group.tsx +136 -0
- package/src/components/FileItem/file-item.tsx +322 -0
- package/src/components/FileUpload/file-upload.tsx +326 -0
- package/src/components/FileViewer/file-viewer-types.ts +76 -0
- package/src/components/FileViewer/file-viewer.tsx +1065 -0
- package/src/components/FileViewer/image-renderer.tsx +256 -0
- package/src/components/HoverCard/hover-card.tsx +79 -0
- package/src/components/Input/input.tsx +233 -0
- package/src/components/LinkInput/link-input.tsx +304 -0
- package/src/components/Menu/menu-item.tsx +334 -0
- package/src/components/NameCard/name-card.tsx +319 -0
- package/src/components/Notice/notice.tsx +196 -0
- package/src/components/NumberInput/number-input.tsx +203 -0
- package/src/components/OverflowIndicator/overflow-indicator.tsx +156 -0
- package/src/components/PeoplePicker/avatar-stack-overflow.ts +100 -0
- package/src/components/PeoplePicker/people-picker-helpers.ts +76 -0
- package/src/components/PeoplePicker/people-picker.tsx +455 -0
- package/src/components/PeoplePicker/person-display.tsx +358 -0
- package/src/components/Popover/popover.tsx +183 -0
- package/src/components/ProgressBar/progress-bar.tsx +157 -0
- package/src/components/README.md +58 -0
- package/src/components/RadioGroup/radio-group.tsx +261 -0
- package/src/components/Rating/rating.tsx +295 -0
- package/src/components/ScrollArea/scroll-area.tsx +110 -0
- package/src/components/SegmentedControl/segmented-control.tsx +304 -0
- package/src/components/Select/select.tsx +658 -0
- package/src/components/SelectMenu/select-menu.tsx +430 -0
- package/src/components/SelectionControl/selection-item.tsx +261 -0
- package/src/components/Separator/separator.tsx +48 -0
- package/src/components/Sheet/sheet.tsx +240 -0
- package/src/components/Sidebar/sidebar.tsx +1280 -0
- package/src/components/Skeleton/skeleton.tsx +35 -0
- package/src/components/Slider/slider.tsx +158 -0
- package/src/components/Steps/steps.tsx +850 -0
- package/src/components/Switch/switch.tsx +285 -0
- package/src/components/Tabs/tabs.tsx +515 -0
- package/src/components/Tag/tag.tsx +246 -0
- package/src/components/Textarea/textarea.tsx +280 -0
- package/src/components/TimePicker/time-columns.tsx +260 -0
- package/src/components/TimePicker/time-picker.tsx +419 -0
- package/src/components/Toast/toast.tsx +129 -0
- package/src/components/Tooltip/tooltip.tsx +68 -0
- package/src/components/TreeView/tree-view.tsx +1031 -0
- package/src/hooks/use-controllable.ts +40 -0
- package/src/hooks/use-is-narrow-viewport.ts +19 -0
- package/src/hooks/use-is-touch-device.ts +21 -0
- package/src/hooks/use-overflow-items.ts +256 -0
- package/src/index.ts +85 -0
- package/src/lib/README.md +82 -0
- package/src/lib/drag-visual.ts +272 -0
- package/src/lib/i18n/README.md +60 -0
- package/src/lib/i18n/i18n-context.tsx +129 -0
- package/src/lib/multi-select-ordering.ts +61 -0
- package/src/lib/utils.ts +93 -0
- package/src/patterns/README.md +67 -0
- package/src/patterns/element-anatomy/item-anatomy.tsx +744 -0
- package/src/patterns/header-canonical/chrome-header.tsx +175 -0
- package/src/patterns/header-canonical/header-canonical.css +27 -0
- package/src/patterns/horizontal-overflow/horizontal-overflow.tsx +217 -0
- package/src/patterns/overlay-surface/overlay-surface.tsx +191 -0
- package/src/patterns/resize-handle/resize-handle.tsx +188 -0
- package/src/stories-helpers/anatomy/anatomy-utils.tsx +64 -0
- package/src/styles/preset.css +31 -0
- package/src/styles/tokens.css +35 -0
- package/src/tokens/README.md +53 -0
- package/src/tokens/color/primitives.css +429 -0
- package/src/tokens/color/semantic.css +539 -0
- package/src/tokens/elevation/overlay-geometry.ts +13 -0
- package/src/tokens/layoutSpace/layoutSpace.css +36 -0
- package/src/tokens/motion/motion.css +30 -0
- package/src/tokens/motion/motion.ts +17 -0
- package/src/tokens/opacity/opacity.css +23 -0
- package/src/tokens/radius/radius.css +19 -0
- package/src/tokens/typography/typography.css +118 -0
- package/src/tokens/uiSize/icon-size.ts +52 -0
- package/src/tokens/uiSize/uiSize.css +125 -0
|
@@ -0,0 +1,744 @@
|
|
|
1
|
+
// @benchmark-unverified-blanket: file-level retraction per M22 (d) — claims herein not individually URL-cited; treat as unverified visual/usage rumor unless retrofit per-claim. Hook escape preserved.
|
|
2
|
+
// code-quality-allow: file-size — foundational SSOT for Family 1+2 row primitives + all item-anatomy helpers(ItemContent / ItemIcon / ItemPrefix / ItemSuffix / ItemInlineAction 等),拆檔會讓 primitive 跨檔 import 滿天飛
|
|
3
|
+
import * as React from "react"
|
|
4
|
+
import type { LucideIcon } from "lucide-react"
|
|
5
|
+
import { cva } from "class-variance-authority"
|
|
6
|
+
import { cn } from "@/lib/utils"
|
|
7
|
+
import { Avatar, type AvatarProps } from "@/design-system/components/Avatar/avatar"
|
|
8
|
+
import {
|
|
9
|
+
Tooltip,
|
|
10
|
+
TooltipTrigger,
|
|
11
|
+
TooltipContent,
|
|
12
|
+
} from "@/design-system/components/Tooltip/tooltip"
|
|
13
|
+
|
|
14
|
+
// ── InlineActionConfig ─────────────────────────────────────────────────────
|
|
15
|
+
// 宣告式 API:consumer 只宣告 intent,host 根據 size tier 自動渲染。
|
|
16
|
+
// 跨 Family 消費(Field family + Row family),SSOT 住 patterns/element-anatomy/
|
|
17
|
+
// 對齊 inline-action.spec.md「API 設計」節。
|
|
18
|
+
//
|
|
19
|
+
// Canonical 實作:`ItemInlineAction`(本檔內)。
|
|
20
|
+
export interface InlineActionConfig {
|
|
21
|
+
icon: LucideIcon
|
|
22
|
+
/** aria-label,同時作為 tooltip 來源 */
|
|
23
|
+
label: string
|
|
24
|
+
/**
|
|
25
|
+
* 點擊 handler。可選接收 React 事件物件——有需要時可呼叫 `stopPropagation()` 避免
|
|
26
|
+
* 事件冒泡到父層(例如 Select 清除按鈕在 popover trigger 內,不想觸發 popover open)。
|
|
27
|
+
*/
|
|
28
|
+
onClick: (e?: React.MouseEvent<HTMLButtonElement>) => void
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* Item Layout — 共用工具與常數
|
|
33
|
+
*
|
|
34
|
+
* 這個檔案不是元件,是 `item-anatomy.spec.md` 的 code-level 實作。
|
|
35
|
+
* 所有 row primitives(`MenuItem` / `TreeItem` / `SidebarMenuButton` /
|
|
36
|
+
* `SelectionItem`)**必須 import 本 module 的 ICON_SIZE 和 helpers**,
|
|
37
|
+
* 不要各自複製常數或 wrapper className——單一 source of truth,避免不一致。
|
|
38
|
+
*
|
|
39
|
+
* 詳細規則見 `item-anatomy.spec.md`。
|
|
40
|
+
*/
|
|
41
|
+
|
|
42
|
+
// ── Types ──────────────────────────────────────────────────────────────────
|
|
43
|
+
|
|
44
|
+
/** Row primitive 的 size 變體,對齊 `--field-height-*` token family */
|
|
45
|
+
export type RowSize = "sm" | "md" | "lg"
|
|
46
|
+
|
|
47
|
+
// ── Constants ──────────────────────────────────────────────────────────────
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* Icon 尺寸(px)per row size。
|
|
51
|
+
*
|
|
52
|
+
* - sm / md: 16px(標準 row icon)
|
|
53
|
+
* - lg: 20px(對齊 text-body-lg 的視覺重量)
|
|
54
|
+
*
|
|
55
|
+
* **用法**:透過 Lucide icon 的 `size` prop 直接傳入,不要用 CSS `[&>svg]:size-*`
|
|
56
|
+
* selector——當 icon 被包在 `<ItemPrefix>` 的 `h-[1lh]` wrapper 內時,
|
|
57
|
+
* `>` 直接子選擇器會失效,Lucide icon 會 fallback 到預設 24px。
|
|
58
|
+
*
|
|
59
|
+
* ```tsx
|
|
60
|
+
* import { ICON_SIZE } from "@/design-system/patterns/element-anatomy/item-anatomy"
|
|
61
|
+
*
|
|
62
|
+
* const iconPx = ICON_SIZE[size] // size: RowSize
|
|
63
|
+
* <MyLucideIcon size={iconPx} />
|
|
64
|
+
* ```
|
|
65
|
+
*/
|
|
66
|
+
export const ICON_SIZE: Record<RowSize, number> = {
|
|
67
|
+
sm: 16,
|
|
68
|
+
md: 16,
|
|
69
|
+
lg: 20,
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Avatar 尺寸(px)per row size,分 inline / block 兩種模式。
|
|
74
|
+
*
|
|
75
|
+
* - **inline**(無 description,大宗情境)→ 對齊第一行 label,用 `h-[1lh]` 容器
|
|
76
|
+
* sm: 20, md: 24, lg: 24
|
|
77
|
+
* - **block**(有 description,avatar 跨越 label + desc 兩行)→ 對齊文字塊中心
|
|
78
|
+
* sm: 32, md: 32, lg: 40
|
|
79
|
+
*
|
|
80
|
+
* **用法**:所有 row primitive 的 consumer(包含 `asChild` pattern 的 Sidebar / Tree
|
|
81
|
+
* consumer)**必須**從本 module import 這個常數,**不可硬寫 `size={24}`**。
|
|
82
|
+
* 硬寫會讓 sm 變體的 avatar 尺寸錯誤、跟 ICON_SIZE 對齊規則脫鉤。
|
|
83
|
+
*
|
|
84
|
+
* ```tsx
|
|
85
|
+
* import { AVATAR_SIZE } from "@/design-system/patterns/element-anatomy/item-anatomy"
|
|
86
|
+
*
|
|
87
|
+
* const { size } = useSidebar() // RowSize
|
|
88
|
+
* <Avatar size={AVATAR_SIZE.inline[size]} />
|
|
89
|
+
* ```
|
|
90
|
+
*
|
|
91
|
+
* Canonical 實作:`MenuItem` 的內部 `AVATAR_SIZE` 查表(即將 re-export 自此)。
|
|
92
|
+
*/
|
|
93
|
+
export const AVATAR_SIZE = {
|
|
94
|
+
inline: { sm: 20, md: 24, lg: 24 },
|
|
95
|
+
block: { sm: 32, md: 32, lg: 40 },
|
|
96
|
+
} as const satisfies Record<"inline" | "block", Record<RowSize, number>>
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* 回傳 uniform prefix slot 的 inline style(CSS variable form)。
|
|
100
|
+
*
|
|
101
|
+
* 用法:Row-primitive 頂層容器(例如 `<SidebarProvider uniformPrefix>` 的 wrapper)
|
|
102
|
+
* 把這個 style 套在 `style` prop 上(預先放在 `--mixed-prefix-slot` 候選位置,
|
|
103
|
+
* 由外層 CSS `:has()` 條件化套到 `--item-prefix-slot`),子樹內所有 `<ItemPrefix>` /
|
|
104
|
+
* `<ItemIcon>` / `<ItemAvatar>` 會自動用固定寬槽對齊。
|
|
105
|
+
*
|
|
106
|
+
* 槽寬 = `AVATAR_SIZE.inline[size]`(20/24/24 @ sm/md/lg)——以最大的 inline
|
|
107
|
+
* prefix(avatar)為基準,icon 在槽內 `justify-center`。
|
|
108
|
+
*
|
|
109
|
+
* ```tsx
|
|
110
|
+
* <ul style={getUniformPrefixSlotStyle(size)}>
|
|
111
|
+
* <ItemIcon icon={Folder} /> ← 16px icon 在 24px 槽內置中
|
|
112
|
+
* <ItemAvatar alt="GitHub" /> ← 24px logo 填滿槽
|
|
113
|
+
* </ul>
|
|
114
|
+
* ```
|
|
115
|
+
*/
|
|
116
|
+
export function getUniformPrefixSlotStyle(size: RowSize): React.CSSProperties {
|
|
117
|
+
return { "--item-prefix-slot": `${AVATAR_SIZE.inline[size]}px` } as React.CSSProperties
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
/**
|
|
121
|
+
* Inline action(suffix 裡的 hover-action icon button)hover 背景尺寸(px)。
|
|
122
|
+
* 規則:icon size + 2px(每邊 +1px,用 absolute positioning 溢出不影響排版)。
|
|
123
|
+
* 對齊 `item-anatomy.spec.md`「Inline Action 設計規格」節。
|
|
124
|
+
*/
|
|
125
|
+
// code-quality-allow: dead-export — public constant — DS API surface,consumer 可引(即使當前 internal-only)
|
|
126
|
+
export const INLINE_ACTION_HOVER_BG_SIZE: Record<RowSize, number> = {
|
|
127
|
+
sm: 18,
|
|
128
|
+
md: 18,
|
|
129
|
+
lg: 22,
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* `ROW_PADDING_BY_SIZE` — Family 1+2 row primitive 的 size → padding + typography 公式 SSOT
|
|
134
|
+
*
|
|
135
|
+
* Formula(canonical):`py = (field-height - 1lh) / 2`,對齊 1em text 到 field-height center。
|
|
136
|
+
*
|
|
137
|
+
* **消費者**:menuItemVariants / sidebarMenuButtonVariants / treeItemVariants / 未來 row primitive
|
|
138
|
+
* **Rationale**:前先 3 cva 各自寫同一公式 → drift risk(item-anatomy.spec 有明文「SidebarMenuButton
|
|
139
|
+
* 獨立實作風險」)。抽本 export 後改**一處全同步**,risk 消失(對齊 Meta-Pattern M17「SSOT 必可傳播」)。
|
|
140
|
+
*
|
|
141
|
+
* 2026-04-24 D Phase 1+2 consolidation 建立。
|
|
142
|
+
*/
|
|
143
|
+
export const ROW_PADDING_BY_SIZE: Record<RowSize, string> = {
|
|
144
|
+
sm: 'text-body leading-compact py-[calc((var(--field-height-sm)-1lh)/2)]',
|
|
145
|
+
md: 'text-body leading-compact py-[calc((var(--field-height-md)-1lh)/2)]',
|
|
146
|
+
lg: 'text-body-lg leading-compact py-[calc((var(--field-height-lg)-1lh)/2)]',
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// ── Helper components ──────────────────────────────────────────────────────
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* `<ItemPrefix>` — Row primitive 的 prefix slot wrapper。
|
|
153
|
+
*
|
|
154
|
+
* 把 prefix(icon / avatar / checkbox / indicator)包進 `h-[1lh]` 容器,
|
|
155
|
+
* 強制對齊第一行文字中線。這是 item-layout 的底層規則(詳見 spec「Prefix 垂直對齊」)。
|
|
156
|
+
*
|
|
157
|
+
* **為什麼**:
|
|
158
|
+
* - Row primitive 的 outer flex 是 `items-start`(多行 label 時 prefix 不飄移)
|
|
159
|
+
* - 單行時 prefix 透過這個 wrapper 強制對齊第一行中線,視覺效果 = `items-center`
|
|
160
|
+
* - 不同尺寸的 prefix(icon 16、Avatar 24、Checkbox 20)都能穩定置中
|
|
161
|
+
*
|
|
162
|
+
* 用法:
|
|
163
|
+
* ```tsx
|
|
164
|
+
* <div className="flex items-start gap-2">
|
|
165
|
+
* <ItemPrefix>
|
|
166
|
+
* <Avatar size={24} />
|
|
167
|
+
* </ItemPrefix>
|
|
168
|
+
* <span className="min-w-0 flex-1 truncate">{label}</span>
|
|
169
|
+
* </div>
|
|
170
|
+
* ```
|
|
171
|
+
*
|
|
172
|
+
* asChild 的 consumer(例如 `<SidebarMenuButton asChild>`)**必須自己**用這個 wrapper
|
|
173
|
+
* 包 prefix,不能省略。
|
|
174
|
+
*/
|
|
175
|
+
export const ItemPrefix = React.forwardRef<
|
|
176
|
+
HTMLSpanElement,
|
|
177
|
+
React.HTMLAttributes<HTMLSpanElement>
|
|
178
|
+
>(({ className, ...props }, ref) => (
|
|
179
|
+
<span
|
|
180
|
+
ref={ref}
|
|
181
|
+
className={cn(
|
|
182
|
+
"h-[1lh] shrink-0 flex items-center",
|
|
183
|
+
// Uniform prefix slot:讀取 `--item-prefix-slot` CSS variable,
|
|
184
|
+
// fallback `auto`(= 預設行為,不佔額外寬度、不 center)。
|
|
185
|
+
// Row-primitive 頂層容器(例如 `<SidebarProvider uniformPrefix>` 的 wrapper)
|
|
186
|
+
// 若設定這個 var,小尺寸 prefix(icon 16px)會在固定寬槽(24px @ md)內置中,
|
|
187
|
+
// 與大尺寸 prefix(app logo 24px)對齊,達成 label 齊左。
|
|
188
|
+
// 詳見 item-anatomy.spec.md「Uniform prefix slot」節。
|
|
189
|
+
"min-w-[var(--item-prefix-slot,auto)] justify-center",
|
|
190
|
+
className
|
|
191
|
+
)}
|
|
192
|
+
{...props}
|
|
193
|
+
/>
|
|
194
|
+
))
|
|
195
|
+
ItemPrefix.displayName = "ItemPrefix"
|
|
196
|
+
|
|
197
|
+
/**
|
|
198
|
+
* `itemPrefixAlignVariants` — Prefix 對齊高度的 cva SSOT(for 複雜 prefix 結構)。
|
|
199
|
+
*
|
|
200
|
+
* ── 存在的理由 ──
|
|
201
|
+
* `<ItemPrefix>` primitive 預設單一 child(icon / avatar 任一)+ inline 對齊。
|
|
202
|
+
* 但**複雜 prefix 結構**(e.g. MenuItem 可同時含 checkbox + icon/avatar 多 child)
|
|
203
|
+
* 或**block mode**(大 prefix + description 對齊塊中心)需要自建 wrapper cva。
|
|
204
|
+
* 本 cva 把 block formula SSOT 化,讓 MenuItem 等 consumer 共用,
|
|
205
|
+
* 改值一處(token or cva)→ 所有 block-prefix consumer 同步。
|
|
206
|
+
*
|
|
207
|
+
* ── Align 變體(對齊 item-anatomy.spec.md「24px 閾值對齊規則」) ──
|
|
208
|
+
* - `"inline"`:`h-[1lh]`,小 prefix(≤ 24px)對齊 label 第一行
|
|
209
|
+
* - `"block-sm"` / `"block-md"`:大 prefix + scanning desc(sm/md menu),
|
|
210
|
+
* 高度 = label(1lh) + gap(token) + desc(caption × 1.3)
|
|
211
|
+
* - `"block-lg"`:大 prefix + scanning desc(lg menu),desc 改為 body × 1.3
|
|
212
|
+
*
|
|
213
|
+
* ── Consumer ──
|
|
214
|
+
* MenuItem(inline / block-sm / block-md / block-lg)。其他 row primitive 簡單情境
|
|
215
|
+
* 直接用 `<ItemPrefix>` 就好,不需此 cva。
|
|
216
|
+
*
|
|
217
|
+
* ── SSOT 傳播 ──
|
|
218
|
+
* gap 用 `var(--item-gap-label-desc)` token;font-size 用 token-awareness
|
|
219
|
+
* (`var(--font-caption-size)` / `var(--font-body-size)`)。改 token → 公式同步。
|
|
220
|
+
*/
|
|
221
|
+
// code-quality-allow: long-function — cva variant/styles table — 拆 fn 會失去 type inference + 跨 fn 傳 config 反而難讀
|
|
222
|
+
export const itemPrefixAlignVariants = cva(
|
|
223
|
+
"flex items-center gap-2 shrink-0",
|
|
224
|
+
{
|
|
225
|
+
variants: {
|
|
226
|
+
align: {
|
|
227
|
+
inline: "h-[1lh]",
|
|
228
|
+
// sm/md desc = text-caption (12px × 1.3 = 15.6px)
|
|
229
|
+
// block-sm / block-md:scanning mode(body + caption 組合)→ 用 scanning mode 專屬 gap token
|
|
230
|
+
"block-sm":
|
|
231
|
+
"h-[calc(1lh+var(--item-gap-label-desc-scanning)+var(--font-caption-size)*1.3)]",
|
|
232
|
+
"block-md":
|
|
233
|
+
"h-[calc(1lh+var(--item-gap-label-desc-scanning)+var(--font-caption-size)*1.3)]",
|
|
234
|
+
// block-lg:scanning-lg mode(body-lg + body compact 組合)→ 用 scanning-lg mode 專屬 gap token
|
|
235
|
+
"block-lg":
|
|
236
|
+
"h-[calc(1lh+var(--item-gap-label-desc-scanning-lg)+var(--font-body-size)*1.3)]",
|
|
237
|
+
},
|
|
238
|
+
},
|
|
239
|
+
defaultVariants: {
|
|
240
|
+
align: "inline",
|
|
241
|
+
},
|
|
242
|
+
}
|
|
243
|
+
)
|
|
244
|
+
|
|
245
|
+
/**
|
|
246
|
+
* `<ItemLabel>` — Row primitive 的 label span。
|
|
247
|
+
*
|
|
248
|
+
* 預設 `min-w-0 flex-1 truncate`——單行截斷,佔滿剩餘 flex 空間。
|
|
249
|
+
* 加 `data-sidebar="menu-label"` attribute 讓 sidebar icon 模式的 CSS selector 能命中。
|
|
250
|
+
*
|
|
251
|
+
* 用法:
|
|
252
|
+
* ```tsx
|
|
253
|
+
* <ItemLabel>{label}</ItemLabel>
|
|
254
|
+
* ```
|
|
255
|
+
*/
|
|
256
|
+
export const ItemLabel = React.forwardRef<
|
|
257
|
+
HTMLSpanElement,
|
|
258
|
+
React.HTMLAttributes<HTMLSpanElement>
|
|
259
|
+
>(({ className, ...props }, ref) => (
|
|
260
|
+
<span
|
|
261
|
+
ref={ref}
|
|
262
|
+
data-sidebar="menu-label"
|
|
263
|
+
className={cn("min-w-0 flex-1 truncate", className)}
|
|
264
|
+
{...props}
|
|
265
|
+
/>
|
|
266
|
+
))
|
|
267
|
+
ItemLabel.displayName = "ItemLabel"
|
|
268
|
+
|
|
269
|
+
/**
|
|
270
|
+
* `<ItemContent>` — Row primitive 的 label + optional description 內容區(SSOT)。
|
|
271
|
+
*
|
|
272
|
+
* ── 存在的唯一理由 ──
|
|
273
|
+
* 封裝「flex-col + label + description + `mt-[var(--item-gap-label-desc)]` gap」結構,
|
|
274
|
+
* 避免 13+ 消費者各自 hard-code `mt-0.5`。改 token 一處,全 DS 同步。
|
|
275
|
+
*
|
|
276
|
+
* ── Consumer 偏離 canonical ──
|
|
277
|
+
* 消費端若有**確切合理理由**不用 `<ItemContent>` / 需自訂 label/desc 行為,必在
|
|
278
|
+
* 該元件 `spec.md` 明文寫下 rationale(對齊 item-anatomy canonical「偏離必明文」)。
|
|
279
|
+
* 合法偏離範例:
|
|
280
|
+
* - MenuItem 的 `leading-compact + text-caption` scanning-mode typography
|
|
281
|
+
* → MenuItem spec 明文 rationale + 消費 `--item-gap-label-desc` token 直接用
|
|
282
|
+
* - SelectionItem 的 control slot 跟 block formula 綁定
|
|
283
|
+
* → SelectionItem spec 明文 rationale
|
|
284
|
+
*
|
|
285
|
+
* ── Props ──
|
|
286
|
+
* - `label`:label 內容(ReactNode,必填)
|
|
287
|
+
* - `description`:description 內容(ReactNode,optional)
|
|
288
|
+
* - `mode`:typography tier(scanning vs reading,世界級對齊 Material `dense` / Carbon `size` / Ant `size`)
|
|
289
|
+
* - `"reading"`(預設):繼承 parent `text-body`(14px / 1.5 leading),舒適閱讀(Family 2 / Material body-large 派)
|
|
290
|
+
* - `"scanning"`:desc 縮為 `text-caption`(12px)+ `leading-compact`(1.3),緊湊掃視
|
|
291
|
+
* (Family 1 Menu item / Material body-medium dense 派)
|
|
292
|
+
* - `descriptionTone`:desc 顏色語意
|
|
293
|
+
* - `"secondary"`(預設):`text-fg-secondary`
|
|
294
|
+
* - `"error"`:`text-error-text`
|
|
295
|
+
* - `"muted"`:`text-fg-muted`
|
|
296
|
+
* - `descriptionWrap`:desc 多行 wrap(預設 true)/ false = truncate
|
|
297
|
+
* - `labelClassName` / `descriptionClassName`:escape hatches(明文 rationale 才用)
|
|
298
|
+
*
|
|
299
|
+
* ── World-class benchmark ──
|
|
300
|
+
* 6 家 DS(Material / Polaris / Atlassian / Apple HIG / Carbon / Ant)一致用
|
|
301
|
+
* 14/20 vs 16/24 兩擋 body 表達 scanning vs reading。API 兩派:(A)密度 prop
|
|
302
|
+
* Material `dense` / Carbon `size` / Ant `size`(本 DS 採 A 派);(B)typography
|
|
303
|
+
* token 手選 Polaris / Atlassian / Apple。
|
|
304
|
+
*/
|
|
305
|
+
|
|
306
|
+
export interface ItemContentProps extends Omit<React.HTMLAttributes<HTMLDivElement>, "title"> {
|
|
307
|
+
label: React.ReactNode
|
|
308
|
+
description?: React.ReactNode
|
|
309
|
+
/**
|
|
310
|
+
* Typography mode — **leading 行為**(跟 size 正交的另一維):
|
|
311
|
+
* - `"reading"`(預設):default leading(1.5)— Family 2 List item 舒適閱讀
|
|
312
|
+
* - `"scanning"`:`leading-compact`(1.3)+ desc 降一 tier —— Menu item / Step 掃視
|
|
313
|
+
*/
|
|
314
|
+
mode?: "reading" | "scanning"
|
|
315
|
+
/**
|
|
316
|
+
* Size tier — **label 字級**(跟 mode 正交的另一維):
|
|
317
|
+
* - `"md"`(預設):label `text-body`(14)
|
|
318
|
+
* - `"lg"`:label `text-body-lg`(16)— desc 永遠比 label 小一 tier(body)
|
|
319
|
+
*
|
|
320
|
+
* 2 × 2 = 4 typography 組合,對應 4 tokens:
|
|
321
|
+
* - md + reading → body(14/1.5) + body(14/1.5)
|
|
322
|
+
* - md + scanning → body(14/1.3) + caption(12/1.3)
|
|
323
|
+
* - lg + reading → body-lg(16/1.5) + body(14/1.5)
|
|
324
|
+
* - lg + scanning → body-lg(16/1.3) + body(14/1.3)
|
|
325
|
+
*/
|
|
326
|
+
size?: "md" | "lg"
|
|
327
|
+
descriptionTone?: "secondary" | "error" | "muted" | "disabled"
|
|
328
|
+
descriptionWrap?: boolean
|
|
329
|
+
/**
|
|
330
|
+
* Description 多行截斷(line-clamp-N)。undefined = 不 clamp(自由 wrap)。
|
|
331
|
+
* Tailwind line-clamp utilities 支援 1-6。
|
|
332
|
+
*/
|
|
333
|
+
descriptionClamp?: number
|
|
334
|
+
/**
|
|
335
|
+
* Description `break-words`(default `false`,僅在 consumer 需要時 opt-in)。
|
|
336
|
+
* 使用情境:SelectionItem / Steps 這類 description 可能塞長英文 token 需 break。
|
|
337
|
+
*/
|
|
338
|
+
descriptionBreakWords?: boolean
|
|
339
|
+
/**
|
|
340
|
+
* Label truncate (default `true`, 對齊 row-item 大宗 idiom)。
|
|
341
|
+
* Opt-out(`false`)for card-like consumers(e.g. NameCard label 允許 wrap)。
|
|
342
|
+
* World-class 對照:Material `ListItemText primaryTypographyProps.noWrap` default true。
|
|
343
|
+
*/
|
|
344
|
+
labelTruncate?: boolean
|
|
345
|
+
labelClassName?: string
|
|
346
|
+
descriptionClassName?: string
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
// code-quality-allow: long-function — foundational composite main body — 拆 sub-fn 會複雜化 local state / ref / context binding
|
|
350
|
+
export const ItemContent = React.forwardRef<HTMLDivElement, ItemContentProps>(
|
|
351
|
+
(
|
|
352
|
+
{
|
|
353
|
+
label,
|
|
354
|
+
description,
|
|
355
|
+
mode = "reading",
|
|
356
|
+
size = "md",
|
|
357
|
+
descriptionTone = "secondary",
|
|
358
|
+
descriptionWrap = true,
|
|
359
|
+
descriptionClamp,
|
|
360
|
+
descriptionBreakWords = false,
|
|
361
|
+
labelTruncate = true,
|
|
362
|
+
labelClassName,
|
|
363
|
+
descriptionClassName,
|
|
364
|
+
className,
|
|
365
|
+
...props
|
|
366
|
+
},
|
|
367
|
+
ref,
|
|
368
|
+
) => {
|
|
369
|
+
const toneClass = {
|
|
370
|
+
secondary: "text-fg-secondary",
|
|
371
|
+
error: "text-error-text",
|
|
372
|
+
muted: "text-fg-muted",
|
|
373
|
+
disabled: "text-fg-disabled",
|
|
374
|
+
}[descriptionTone]
|
|
375
|
+
|
|
376
|
+
// Typography desc class — 2 維 cross-product (mode × size):
|
|
377
|
+
// ┌───────────┬────────────────────────────┬────────────────────────────┐
|
|
378
|
+
// │ │ size="md"(body label 14) │ size="lg"(body-lg label 16)│
|
|
379
|
+
// ├───────────┼────────────────────────────┼────────────────────────────┤
|
|
380
|
+
// │ reading │ 繼承 body(14/1.5) │ body(14)default leading │
|
|
381
|
+
// │ scanning │ caption(12)+ compact │ body(14)+ compact │
|
|
382
|
+
// └───────────┴────────────────────────────┴────────────────────────────┘
|
|
383
|
+
const isLg = size === "lg"
|
|
384
|
+
const modeClass =
|
|
385
|
+
mode === "scanning"
|
|
386
|
+
? isLg
|
|
387
|
+
? "text-[length:var(--font-body-size)] leading-compact"
|
|
388
|
+
: "text-[length:var(--font-caption-size)] leading-compact"
|
|
389
|
+
: isLg
|
|
390
|
+
? "text-[length:var(--font-body-size)]"
|
|
391
|
+
: ""
|
|
392
|
+
|
|
393
|
+
const clampClass = descriptionClamp ? `line-clamp-${descriptionClamp}` : ""
|
|
394
|
+
|
|
395
|
+
return (
|
|
396
|
+
<div
|
|
397
|
+
ref={ref}
|
|
398
|
+
className={cn("flex flex-col min-w-0 flex-1", className)}
|
|
399
|
+
{...props}
|
|
400
|
+
>
|
|
401
|
+
<span className={cn(labelTruncate && "truncate", labelClassName)}>{label}</span>
|
|
402
|
+
{description && (
|
|
403
|
+
<span
|
|
404
|
+
className={cn(
|
|
405
|
+
// Typography-mode-aware gap token(2026-04-23):
|
|
406
|
+
// (mode, size)2 維正交 → 4 token 對應 4 typography 組合
|
|
407
|
+
mode === "scanning"
|
|
408
|
+
? isLg
|
|
409
|
+
? "mt-[var(--item-gap-label-desc-scanning-lg)]"
|
|
410
|
+
: "mt-[var(--item-gap-label-desc-scanning)]"
|
|
411
|
+
: isLg
|
|
412
|
+
? "mt-[var(--item-gap-label-desc-reading-lg)]"
|
|
413
|
+
: "mt-[var(--item-gap-label-desc-reading)]",
|
|
414
|
+
modeClass,
|
|
415
|
+
toneClass,
|
|
416
|
+
clampClass,
|
|
417
|
+
descriptionBreakWords && "break-words",
|
|
418
|
+
!descriptionWrap && "truncate",
|
|
419
|
+
descriptionClassName,
|
|
420
|
+
)}
|
|
421
|
+
>
|
|
422
|
+
{description}
|
|
423
|
+
</span>
|
|
424
|
+
)}
|
|
425
|
+
</div>
|
|
426
|
+
)
|
|
427
|
+
},
|
|
428
|
+
)
|
|
429
|
+
ItemContent.displayName = "ItemContent"
|
|
430
|
+
|
|
431
|
+
// ── Row size context ──────────────────────────────────────────────────────
|
|
432
|
+
//
|
|
433
|
+
// 作用:讓 row primitive(Sidebar / SelectMenu / Tree / DropdownMenu)只 propagate
|
|
434
|
+
// 一次 size,整個子樹內的 <ItemIcon> / <ItemAvatar> 都自動查到正確尺寸。
|
|
435
|
+
//
|
|
436
|
+
// **存在的唯一理由**:消除 asChild pattern 下 consumer 硬寫 `size={24}` 的可能性。
|
|
437
|
+
// 過去 consumer 必須 `useSidebar().size` 再手動傳 `AVATAR_SIZE.inline[size]`,
|
|
438
|
+
// 每個 consumer 都是一次漂移機會(曾發生:sm 欄 avatar 顯示 24 而非 20,破壞規格)。
|
|
439
|
+
// 現在 consumer 只寫 `<ItemAvatar alt="..." />`,尺寸由 context 自動決定。
|
|
440
|
+
//
|
|
441
|
+
// Row primitive 實作者的責任:在元件內部用 `<RowSizeProvider value={size}>` 包裹
|
|
442
|
+
// children(不論 asChild 或非 asChild 路徑),這樣任何 descendant 的 ItemIcon /
|
|
443
|
+
// ItemAvatar 都會拿到正確尺寸。
|
|
444
|
+
|
|
445
|
+
const RowSizeContext = React.createContext<RowSize | null>(null)
|
|
446
|
+
|
|
447
|
+
export const RowSizeProvider = RowSizeContext.Provider
|
|
448
|
+
|
|
449
|
+
/**
|
|
450
|
+
* 讀取當前 row size。在 row primitive 子樹外呼叫會回 fallback(預設 "md")。
|
|
451
|
+
* Row primitive 實作者不該呼叫這個——你們已經知道自己的 size。
|
|
452
|
+
* 這個 hook 是給 `<ItemIcon>` / `<ItemAvatar>` 和 asChild consumer 的逃生艙用。
|
|
453
|
+
*/
|
|
454
|
+
export function useRowSize(fallback: RowSize = "md"): RowSize {
|
|
455
|
+
return React.useContext(RowSizeContext) ?? fallback
|
|
456
|
+
}
|
|
457
|
+
|
|
458
|
+
// ── Prefab prefix elements ────────────────────────────────────────────────
|
|
459
|
+
//
|
|
460
|
+
// `<ItemIcon>` 和 `<ItemAvatar>` 是「prefix slot 的完成品元件」——自帶 ItemPrefix
|
|
461
|
+
// wrapper + 自動查 ICON_SIZE / AVATAR_SIZE。
|
|
462
|
+
//
|
|
463
|
+
// **asChild consumer 必須用這兩個元件,禁止手動寫 `<Avatar size={N} />`**。
|
|
464
|
+
// 禁令理由見 item-anatomy.spec.md「Avatar 尺寸選擇」節。
|
|
465
|
+
|
|
466
|
+
export interface ItemIconProps
|
|
467
|
+
extends Omit<React.SVGAttributes<SVGSVGElement>, "children"> {
|
|
468
|
+
/** Lucide icon 元件(不是 JSX element) */
|
|
469
|
+
icon: LucideIcon
|
|
470
|
+
}
|
|
471
|
+
|
|
472
|
+
/**
|
|
473
|
+
* `<ItemIcon icon={Folder} />` — 自動查 `ICON_SIZE[rowSize]`,包在 `ItemPrefix` 內。
|
|
474
|
+
* 永遠對齊第一行 label(item-layout 的 icon 永遠 inline)。
|
|
475
|
+
*/
|
|
476
|
+
export const ItemIcon = React.forwardRef<SVGSVGElement, ItemIconProps>(
|
|
477
|
+
({ icon: Icon, className, ...props }, ref) => {
|
|
478
|
+
const size = useRowSize()
|
|
479
|
+
return (
|
|
480
|
+
// data-prefix-type="icon" 讓 row-primitive 容器的 `:has()` selector 能偵測類型,
|
|
481
|
+
// 用於 `<SidebarProvider uniformPrefix>` opt-in 後的混用 auto-detection
|
|
482
|
+
<ItemPrefix data-prefix-type="icon">
|
|
483
|
+
<Icon
|
|
484
|
+
ref={ref}
|
|
485
|
+
size={ICON_SIZE[size]}
|
|
486
|
+
className={cn("shrink-0", className)}
|
|
487
|
+
aria-hidden
|
|
488
|
+
{...props}
|
|
489
|
+
/>
|
|
490
|
+
</ItemPrefix>
|
|
491
|
+
)
|
|
492
|
+
}
|
|
493
|
+
)
|
|
494
|
+
ItemIcon.displayName = "ItemIcon"
|
|
495
|
+
|
|
496
|
+
export interface ItemAvatarProps extends Omit<AvatarProps, "size"> {
|
|
497
|
+
/**
|
|
498
|
+
* 對齊模式(選填,預設 `"inline"`):
|
|
499
|
+
* - `"inline"`:視覺重量輕的 avatar,對齊第一行 label(20/24/24 @ sm/md/lg)
|
|
500
|
+
* - `"block"`:視覺重量重的 avatar,對齊 label + description 文字塊(32/32/40 @ sm/md/lg)
|
|
501
|
+
*
|
|
502
|
+
* 預設 inline——大宗情境(扁平 row、footer user、主清單)都用這個。
|
|
503
|
+
* 只有 avatar 是 item 主體(人物卡、顯著身份辨識)才用 block。
|
|
504
|
+
*
|
|
505
|
+
* 詳見 `item-anatomy.spec.md`「Avatar 尺寸選擇」節,含 24px 對齊模式閾值規則。
|
|
506
|
+
*/
|
|
507
|
+
mode?: "inline" | "block"
|
|
508
|
+
}
|
|
509
|
+
|
|
510
|
+
/**
|
|
511
|
+
* `<ItemAvatar alt="Alan" color="blue" />` — 自動查 `AVATAR_SIZE[mode][rowSize]`,
|
|
512
|
+
* 包在 `ItemPrefix` 內。
|
|
513
|
+
*
|
|
514
|
+
* asChild consumer 的標準用法——不需知道 size,不需 import AVATAR_SIZE,
|
|
515
|
+
* 不可能硬寫錯誤尺寸。
|
|
516
|
+
*/
|
|
517
|
+
// code-quality-allow: long-function — forwardRef body 含 size×color×hoverCard 三正交 prop 渲染分流,拆出 sub-fn 反增 prop drilling
|
|
518
|
+
export const ItemAvatar = React.forwardRef<HTMLDivElement, ItemAvatarProps>(
|
|
519
|
+
({ mode = "inline", ...props }, ref) => {
|
|
520
|
+
const size = useRowSize()
|
|
521
|
+
return (
|
|
522
|
+
// data-prefix-type="avatar" 讓 row-primitive 容器的 `:has()` selector 能偵測類型,
|
|
523
|
+
// 用於 `<SidebarProvider uniformPrefix>` opt-in 後的混用 auto-detection
|
|
524
|
+
<ItemPrefix data-prefix-type="avatar">
|
|
525
|
+
<Avatar ref={ref} size={AVATAR_SIZE[mode][size]} {...props} />
|
|
526
|
+
</ItemPrefix>
|
|
527
|
+
)
|
|
528
|
+
}
|
|
529
|
+
)
|
|
530
|
+
ItemAvatar.displayName = "ItemAvatar"
|
|
531
|
+
|
|
532
|
+
// ── Inline action (suffix slot) ───────────────────────────────────────────
|
|
533
|
+
//
|
|
534
|
+
// 共用 row-primitive 的 suffix inline action 渲染——消除過去 Input / NumberInput /
|
|
535
|
+
// Tag / LinkInput / Combobox 各自複製 18 行 JSX 的狀況(見 `item-anatomy.spec.md`
|
|
536
|
+
// 「Inline Action 設計規格」節)。
|
|
537
|
+
//
|
|
538
|
+
// 規格(完全對齊 item-anatomy.spec.md「Inline Action 設計規格」節):
|
|
539
|
+
// - Icon 尺寸 = ICON_SIZE[rowSize] (16/16/20)
|
|
540
|
+
// - Hover bg 尺寸 = icon + 2 (18/18/22,INLINE_ACTION_HOVER_BG_SIZE)
|
|
541
|
+
// - Hover bg 圓角 = rounded-md (sm/md)/ rounded-md (lg)
|
|
542
|
+
// - Icon 顏色:fg-muted → foreground on hover/active
|
|
543
|
+
// - 必須 <button type="button">,aria-label = label,Tooltip(同 label)
|
|
544
|
+
// - cursor-pointer
|
|
545
|
+
|
|
546
|
+
// ── Low-level:ItemInlineActionButton ──
|
|
547
|
+
//
|
|
548
|
+
// Root 是 `<button>`(不是 Tooltip wrapper),可直接塞進 Radix `asChild` pattern
|
|
549
|
+
// 例如 `<Collapsible.Trigger asChild>`, `<Popover.Trigger asChild>` 等。
|
|
550
|
+
//
|
|
551
|
+
// 自動從 `RowSizeContext` 查:
|
|
552
|
+
// - 寬高 = ICON_SIZE[size]
|
|
553
|
+
// - Hover bg 尺寸 = INLINE_ACTION_HOVER_BG_SIZE[size]
|
|
554
|
+
// - Hover bg 圓角 = rounded-md (sm/md) / rounded-md (lg)
|
|
555
|
+
// - Icon color = fg-muted → foreground on hover/active
|
|
556
|
+
//
|
|
557
|
+
// 接受任何 button props 並 spread(讓 Radix Slot 可以 merge onClick / data-state 等)
|
|
558
|
+
|
|
559
|
+
export interface ItemInlineActionButtonProps
|
|
560
|
+
extends Omit<React.ButtonHTMLAttributes<HTMLButtonElement>, "children"> {
|
|
561
|
+
/** Lucide icon 元件 */
|
|
562
|
+
icon: LucideIcon
|
|
563
|
+
/** 可選:額外加在 icon 上的 className(例如 rotate transition) */
|
|
564
|
+
iconClassName?: string
|
|
565
|
+
/**
|
|
566
|
+
* Size 明確覆寫(用於 fields / Tag 等**不在 `RowSizeProvider` 樹內**的 host):
|
|
567
|
+
* 傳了就以此為準,否則從 `RowSizeContext` 讀(fallback "md")。
|
|
568
|
+
*
|
|
569
|
+
* 使用時機:
|
|
570
|
+
* - Row primitive 子樹(Sidebar / TreeView / SelectMenu / DropdownMenu)→ **不傳**,吃 context
|
|
571
|
+
* - Field / Tag(自有 size 系統,沒包 RowSizeProvider)→ **必傳** `size={hostSize}`
|
|
572
|
+
*/
|
|
573
|
+
size?: RowSize
|
|
574
|
+
/**
|
|
575
|
+
* Hover / active bg className override(chromatic host 專用,如 Tag solid variant)。
|
|
576
|
+
*
|
|
577
|
+
* 預設 `group-hover/action:bg-neutral-hover group-active/action:bg-neutral-active`。
|
|
578
|
+
* 提供後**完全取代** hover + active 雙態 className,但**不影響** rest 態(永遠 bg-transparent)
|
|
579
|
+
* 與 overlay-trigger 態(`group-data-[state=open]:bg-neutral-selected`)。
|
|
580
|
+
*
|
|
581
|
+
* Consumer 須**同時**包含 hover + active 兩態 selector,例:
|
|
582
|
+
* `'group-hover/action:bg-[var(--my-hover)] group-active/action:bg-[var(--my-active)]'`
|
|
583
|
+
*
|
|
584
|
+
* Use case:Tag solid(blue/green/red 等)需 hover bg 跟 host 色相一致(非 neutral)。
|
|
585
|
+
*/
|
|
586
|
+
hoverBgClassName?: string
|
|
587
|
+
/**
|
|
588
|
+
* 標記本 button 是 **overlay trigger**(DropdownMenu / Popover / Tooltip 透過 `asChild` 包覆)。
|
|
589
|
+
*
|
|
590
|
+
* `true` 時:Radix overlay open 期間(`data-state="open"`)button 維持 host hover 樣式
|
|
591
|
+
* — 視覺鎖,讓 user 追溯 floating panel 來源(對齊 shadcn / Radix Themes / Material)。
|
|
592
|
+
*
|
|
593
|
+
* `false` 時(default):無 data-state=open 樣式 — 適用 in-place 互動如 `Collapsible.Trigger`
|
|
594
|
+
* (展開內容接在 trigger 下方,不需追溯)、drag handle、dismiss X 等。
|
|
595
|
+
*
|
|
596
|
+
* Canonical 詳 `inline-action.spec.md`「Overlay trigger canonical」段。
|
|
597
|
+
*
|
|
598
|
+
* @default false
|
|
599
|
+
*/
|
|
600
|
+
overlayTrigger?: boolean
|
|
601
|
+
}
|
|
602
|
+
|
|
603
|
+
export const ItemInlineActionButton = React.forwardRef<
|
|
604
|
+
HTMLButtonElement,
|
|
605
|
+
ItemInlineActionButtonProps
|
|
606
|
+
>(({ icon: Icon, className, iconClassName, style, type = "button", size: sizeProp, hoverBgClassName, overlayTrigger = false, ...rest }, ref) => {
|
|
607
|
+
const contextSize = useRowSize()
|
|
608
|
+
const size = sizeProp ?? contextSize
|
|
609
|
+
const iconPx = ICON_SIZE[size]
|
|
610
|
+
const hoverBgPx = INLINE_ACTION_HOVER_BG_SIZE[size]
|
|
611
|
+
return (
|
|
612
|
+
<button
|
|
613
|
+
ref={ref}
|
|
614
|
+
type={type}
|
|
615
|
+
className={cn(
|
|
616
|
+
"group/action relative grid place-content-center shrink-0 cursor-pointer",
|
|
617
|
+
"text-fg-muted hover:text-foreground active:text-foreground transition-colors",
|
|
618
|
+
// Overlay trigger active state — 只在 consumer 顯式宣告 overlayTrigger=true 時生效
|
|
619
|
+
// (canonical 2026-05-05:Collapsible / drag / dismiss 等 in-place 互動 ≠ overlay,
|
|
620
|
+
// 不應 leak 視覺 lock。詳 inline-action.spec.md「Overlay trigger canonical」)
|
|
621
|
+
overlayTrigger && "data-[state=open]:text-foreground",
|
|
622
|
+
"focus-visible:outline-2 focus-visible:outline-ring",
|
|
623
|
+
className
|
|
624
|
+
)}
|
|
625
|
+
style={{ width: iconPx, height: iconPx, ...style }}
|
|
626
|
+
{...rest}
|
|
627
|
+
>
|
|
628
|
+
<span
|
|
629
|
+
aria-hidden
|
|
630
|
+
className={cn(
|
|
631
|
+
"absolute pointer-events-none",
|
|
632
|
+
"rounded-md",
|
|
633
|
+
"bg-transparent",
|
|
634
|
+
hoverBgClassName ?? "group-hover/action:bg-neutral-hover group-active/action:bg-neutral-active",
|
|
635
|
+
// Overlay 開啟 = 維持 host hover bg(只在 overlayTrigger=true 時生效)
|
|
636
|
+
overlayTrigger && "group-data-[state=open]/action:bg-neutral-hover",
|
|
637
|
+
"transition-colors"
|
|
638
|
+
)}
|
|
639
|
+
style={{
|
|
640
|
+
width: hoverBgPx,
|
|
641
|
+
height: hoverBgPx,
|
|
642
|
+
top: "50%",
|
|
643
|
+
left: "50%",
|
|
644
|
+
transform: "translate(-50%, -50%)",
|
|
645
|
+
}}
|
|
646
|
+
/>
|
|
647
|
+
<Icon size={iconPx} className={cn("relative", iconClassName)} aria-hidden />
|
|
648
|
+
</button>
|
|
649
|
+
)
|
|
650
|
+
})
|
|
651
|
+
ItemInlineActionButton.displayName = "ItemInlineActionButton"
|
|
652
|
+
|
|
653
|
+
// ── High-level:ItemInlineAction ──
|
|
654
|
+
//
|
|
655
|
+
// 宣告式 API:consumer 給 `{ icon, label, onClick }`,元件自動處理 Tooltip +
|
|
656
|
+
// aria-label。適用於 row primitive 的 suffix slot(SidebarMenuButton inlineActions 等)。
|
|
657
|
+
//
|
|
658
|
+
// 內部用 `ItemInlineActionButton`——兩層共用同一套視覺規格、同一個 RowSizeContext 查表。
|
|
659
|
+
// 需要接 Radix asChild(例如 collapsible trigger)時改用 `ItemInlineActionButton`,
|
|
660
|
+
// 那層沒有 Tooltip 包裝,Root 就是 button,可直接塞進 Radix Slot。
|
|
661
|
+
|
|
662
|
+
export interface ItemInlineActionProps {
|
|
663
|
+
action: InlineActionConfig
|
|
664
|
+
/** Size 覆寫(見 `ItemInlineActionButtonProps.size`)——Field Controls / Tag 必傳 */
|
|
665
|
+
size?: RowSize
|
|
666
|
+
}
|
|
667
|
+
|
|
668
|
+
export const ItemInlineAction = React.forwardRef<
|
|
669
|
+
HTMLButtonElement,
|
|
670
|
+
ItemInlineActionProps
|
|
671
|
+
>(({ action, size }, ref) => (
|
|
672
|
+
<Tooltip>
|
|
673
|
+
<TooltipTrigger asChild>
|
|
674
|
+
<ItemInlineActionButton
|
|
675
|
+
ref={ref}
|
|
676
|
+
icon={action.icon}
|
|
677
|
+
aria-label={action.label}
|
|
678
|
+
onClick={action.onClick}
|
|
679
|
+
size={size}
|
|
680
|
+
/>
|
|
681
|
+
</TooltipTrigger>
|
|
682
|
+
<TooltipContent>{action.label}</TooltipContent>
|
|
683
|
+
</Tooltip>
|
|
684
|
+
))
|
|
685
|
+
ItemInlineAction.displayName = "ItemInlineAction"
|
|
686
|
+
|
|
687
|
+
// ── Item suffix slot ──────────────────────────────────────────────────────
|
|
688
|
+
//
|
|
689
|
+
// `<ItemSuffix>` — row primitive 的 suffix 容器。對齊 item-layout spec:
|
|
690
|
+
// - `h-[1lh]`:suffix 永遠對齊第一行 label(跟 prefix 解耦)
|
|
691
|
+
// - `ml-auto`:靠右
|
|
692
|
+
// - `gap-2`:多個 inline action 之間的標準間距(item-anatomy.spec.md「Inline Action 設計規格」節)
|
|
693
|
+
// - `hoverReveal` opt-in:opacity 0→1 on 父層 row hover/focus-visible
|
|
694
|
+
//
|
|
695
|
+
// `hoverGroup`(2026-05-05 v8 SSOT 升級):row primitive group selector 參數化。
|
|
696
|
+
// 先前 `/menu-item` hardcode → TreeView(`/tree-item`)/ FileItem(`/row`)/
|
|
697
|
+
// DataTable(`/row`)無法消費,被迫自刻 suffix slot(M1 違反)。
|
|
698
|
+
// 參數化後 4 個 row primitive 都走同一條 SSOT。Tailwind JIT 需 static class,
|
|
699
|
+
// 故用 lookup table 而非動態字串(避免 JIT scan 失敗)。
|
|
700
|
+
|
|
701
|
+
const SUFFIX_HOVER_REVEAL_BY_GROUP = {
|
|
702
|
+
"menu-item":
|
|
703
|
+
"opacity-0 group-hover/menu-item:opacity-100 group-has-[:focus-visible]/menu-item:opacity-100 transition-opacity duration-150",
|
|
704
|
+
"tree-item":
|
|
705
|
+
"opacity-0 group-hover/tree-item:opacity-100 group-has-[:focus-visible]/tree-item:opacity-100 transition-opacity duration-150",
|
|
706
|
+
row: "opacity-0 group-hover/row:opacity-100 group-has-[:focus-visible]/row:opacity-100 transition-opacity duration-150",
|
|
707
|
+
} as const
|
|
708
|
+
|
|
709
|
+
export type ItemSuffixHoverGroup = keyof typeof SUFFIX_HOVER_REVEAL_BY_GROUP
|
|
710
|
+
|
|
711
|
+
export interface ItemSuffixProps extends React.HTMLAttributes<HTMLSpanElement> {
|
|
712
|
+
/**
|
|
713
|
+
* Hover-reveal:預設隱藏,父層 row hover / keyboard focus-visible 時才淡入。
|
|
714
|
+
* 對齊 TreeView / SidebarMenuButton inline action 行為。預設 false(永遠顯示,如 Badge)。
|
|
715
|
+
*
|
|
716
|
+
* 用 `group-has-[:focus-visible]` 而非 `group-focus-within`——後者會被 mouse click 觸發,
|
|
717
|
+
* 導致 click 後 suffix 永久顯示直到焦點移走。focus-visible 只在鍵盤 tab 時啟動。
|
|
718
|
+
*/
|
|
719
|
+
hoverReveal?: boolean
|
|
720
|
+
/**
|
|
721
|
+
* 父層 row 的 group selector(consumer 必加 `group/<name>` 到 row wrapper)。
|
|
722
|
+
* 預設 `'menu-item'`(對齊 MenuItem / Sidebar)。其他 row primitive:
|
|
723
|
+
* `'tree-item'`(TreeView)/ `'row'`(DataTable / FileItem)。
|
|
724
|
+
*
|
|
725
|
+
* 加新 row primitive(且需 hover-reveal)時:在 `SUFFIX_HOVER_REVEAL_BY_GROUP`
|
|
726
|
+
* 加 entry,Tailwind JIT 才能 scan 到 static class。
|
|
727
|
+
*/
|
|
728
|
+
hoverGroup?: ItemSuffixHoverGroup
|
|
729
|
+
}
|
|
730
|
+
|
|
731
|
+
export const ItemSuffix = React.forwardRef<HTMLSpanElement, ItemSuffixProps>(
|
|
732
|
+
({ className, hoverReveal = false, hoverGroup = "menu-item", ...props }, ref) => (
|
|
733
|
+
<span
|
|
734
|
+
ref={ref}
|
|
735
|
+
className={cn(
|
|
736
|
+
"h-[1lh] shrink-0 ml-auto flex items-center gap-2",
|
|
737
|
+
hoverReveal && SUFFIX_HOVER_REVEAL_BY_GROUP[hoverGroup],
|
|
738
|
+
className
|
|
739
|
+
)}
|
|
740
|
+
{...props}
|
|
741
|
+
/>
|
|
742
|
+
)
|
|
743
|
+
)
|
|
744
|
+
ItemSuffix.displayName = "ItemSuffix"
|