@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,377 @@
|
|
|
1
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
2
|
+
import * as React from "react";
|
|
3
|
+
import { Pencil } from "lucide-react";
|
|
4
|
+
import { cn } from "../../lib/utils.js";
|
|
5
|
+
import { Input } from "../Input/input.js";
|
|
6
|
+
import { Textarea } from "../Textarea/textarea.js";
|
|
7
|
+
import { NumberInput } from "../NumberInput/number-input.js";
|
|
8
|
+
import { Select } from "../Select/select.js";
|
|
9
|
+
import { Combobox } from "../Combobox/combobox.js";
|
|
10
|
+
import { DatePicker as DatePickerWithRange } from "../DatePicker/date-picker.js";
|
|
11
|
+
import { TimePicker } from "../TimePicker/time-picker.js";
|
|
12
|
+
import { PeoplePicker } from "../PeoplePicker/people-picker.js";
|
|
13
|
+
import { LinkInput } from "../LinkInput/link-input.js";
|
|
14
|
+
import { Checkbox } from "../Checkbox/checkbox.js";
|
|
15
|
+
import { Button } from "../Button/button.js";
|
|
16
|
+
import { FieldSurfaceProvider } from "../Field/field-context.js";
|
|
17
|
+
function makeKeyHandler(onCommit, onCancel, parseValue) {
|
|
18
|
+
return (e) => {
|
|
19
|
+
if (e.key === "Escape") {
|
|
20
|
+
e.preventDefault();
|
|
21
|
+
onCancel == null ? void 0 : onCancel();
|
|
22
|
+
}
|
|
23
|
+
if (e.key === "Enter") {
|
|
24
|
+
e.preventDefault();
|
|
25
|
+
const raw = e.target.value;
|
|
26
|
+
onCommit == null ? void 0 : onCommit(raw);
|
|
27
|
+
}
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
const sizeForInput = (size) => size;
|
|
31
|
+
const displayOrDisabled = (isDisabled) => isDisabled ? "disabled" : "display";
|
|
32
|
+
function StringCell({ value, meta, mode, size, isDisabled, autoRowHeight, onCommit, onCancel, onDraft }) {
|
|
33
|
+
const maxLines = meta == null ? void 0 : meta.maxLines;
|
|
34
|
+
const clampClass = maxLines && autoRowHeight ? `line-clamp-[${maxLines}]` : void 0;
|
|
35
|
+
const v = value != null ? String(value) : "";
|
|
36
|
+
if (mode === "display") {
|
|
37
|
+
return autoRowHeight ? /* @__PURE__ */ jsx(Textarea, { variant: "naked", mode: displayOrDisabled(isDisabled), value: v, className: clampClass }) : /* @__PURE__ */ jsx(Input, { variant: "naked", mode: displayOrDisabled(isDisabled), value: v });
|
|
38
|
+
}
|
|
39
|
+
if (autoRowHeight) {
|
|
40
|
+
const newlineRows = (v.match(/\n/g) || []).length + 1;
|
|
41
|
+
const wrapRows = Math.ceil(v.length / 40);
|
|
42
|
+
const estimateRows = Math.min(10, Math.max(1, newlineRows, wrapRows));
|
|
43
|
+
return /* @__PURE__ */ jsx(
|
|
44
|
+
Textarea,
|
|
45
|
+
{
|
|
46
|
+
autoFocus: true,
|
|
47
|
+
variant: "naked",
|
|
48
|
+
size: sizeForInput(size),
|
|
49
|
+
rows: estimateRows,
|
|
50
|
+
defaultValue: v,
|
|
51
|
+
style: { fieldSizing: "content" },
|
|
52
|
+
onChange: (e) => onDraft == null ? void 0 : onDraft(e.target.value),
|
|
53
|
+
onBlur: (e) => onCommit == null ? void 0 : onCommit(e.target.value),
|
|
54
|
+
onKeyDown: (e) => {
|
|
55
|
+
if (e.key === "Escape") {
|
|
56
|
+
e.preventDefault();
|
|
57
|
+
onCancel == null ? void 0 : onCancel();
|
|
58
|
+
}
|
|
59
|
+
if (e.key === "Enter" && (e.metaKey || e.ctrlKey)) {
|
|
60
|
+
e.preventDefault();
|
|
61
|
+
onCommit == null ? void 0 : onCommit(e.target.value);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
);
|
|
66
|
+
}
|
|
67
|
+
return /* @__PURE__ */ jsx(
|
|
68
|
+
Input,
|
|
69
|
+
{
|
|
70
|
+
autoFocus: true,
|
|
71
|
+
variant: "naked",
|
|
72
|
+
size: sizeForInput(size),
|
|
73
|
+
defaultValue: v,
|
|
74
|
+
onChange: (e) => onDraft == null ? void 0 : onDraft(e.target.value),
|
|
75
|
+
onBlur: (e) => onCommit == null ? void 0 : onCommit(e.target.value),
|
|
76
|
+
onKeyDown: makeKeyHandler(onCommit, onCancel)
|
|
77
|
+
}
|
|
78
|
+
);
|
|
79
|
+
}
|
|
80
|
+
function NumberCell({ value, meta, mode, size, isDisabled, onCommit, onCancel, onDraft }) {
|
|
81
|
+
const isCurrency = (meta == null ? void 0 : meta.type) === "currency";
|
|
82
|
+
const prefix = isCurrency ? (meta == null ? void 0 : meta.prefix) ?? "$" : meta == null ? void 0 : meta.prefix;
|
|
83
|
+
if (mode === "display") {
|
|
84
|
+
return /* @__PURE__ */ jsx(
|
|
85
|
+
NumberInput,
|
|
86
|
+
{
|
|
87
|
+
variant: "naked",
|
|
88
|
+
mode: displayOrDisabled(isDisabled),
|
|
89
|
+
value,
|
|
90
|
+
prefix,
|
|
91
|
+
suffix: meta == null ? void 0 : meta.suffix,
|
|
92
|
+
precision: meta == null ? void 0 : meta.precision,
|
|
93
|
+
locale: meta == null ? void 0 : meta.locale
|
|
94
|
+
}
|
|
95
|
+
);
|
|
96
|
+
}
|
|
97
|
+
const initial = typeof value === "number" ? value : null;
|
|
98
|
+
const [localValue, setLocalValue] = React.useState(initial);
|
|
99
|
+
return /* @__PURE__ */ jsx(
|
|
100
|
+
NumberInput,
|
|
101
|
+
{
|
|
102
|
+
autoFocus: true,
|
|
103
|
+
variant: "naked",
|
|
104
|
+
size: sizeForInput(size),
|
|
105
|
+
value: localValue,
|
|
106
|
+
onChange: (v) => {
|
|
107
|
+
setLocalValue(v);
|
|
108
|
+
onDraft == null ? void 0 : onDraft(v);
|
|
109
|
+
},
|
|
110
|
+
prefix,
|
|
111
|
+
suffix: meta == null ? void 0 : meta.suffix,
|
|
112
|
+
precision: meta == null ? void 0 : meta.precision,
|
|
113
|
+
onBlur: () => onCommit == null ? void 0 : onCommit(localValue),
|
|
114
|
+
onKeyDown: (e) => {
|
|
115
|
+
if (e.key === "Escape") {
|
|
116
|
+
e.preventDefault();
|
|
117
|
+
onCancel == null ? void 0 : onCancel();
|
|
118
|
+
}
|
|
119
|
+
if (e.key === "Enter") {
|
|
120
|
+
e.preventDefault();
|
|
121
|
+
onCommit == null ? void 0 : onCommit(localValue);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
);
|
|
126
|
+
}
|
|
127
|
+
const dismissOnClose = (onCancel) => (open) => {
|
|
128
|
+
if (!open) onCancel == null ? void 0 : onCancel();
|
|
129
|
+
};
|
|
130
|
+
function DateCell({ value, meta, mode, size, isDisabled, isEditable, onCommit, onCancel }) {
|
|
131
|
+
if (mode === "display") {
|
|
132
|
+
return /* @__PURE__ */ jsx(
|
|
133
|
+
DatePickerWithRange,
|
|
134
|
+
{
|
|
135
|
+
variant: "naked",
|
|
136
|
+
mode: displayOrDisabled(isDisabled),
|
|
137
|
+
value,
|
|
138
|
+
size,
|
|
139
|
+
formatOptions: meta == null ? void 0 : meta.formatOptions,
|
|
140
|
+
locale: meta == null ? void 0 : meta.locale,
|
|
141
|
+
showDisplayEndIcon: isEditable === true
|
|
142
|
+
},
|
|
143
|
+
"display"
|
|
144
|
+
);
|
|
145
|
+
}
|
|
146
|
+
return /* @__PURE__ */ jsx(
|
|
147
|
+
DatePickerWithRange,
|
|
148
|
+
{
|
|
149
|
+
autoFocus: true,
|
|
150
|
+
variant: "naked",
|
|
151
|
+
size: sizeForInput(size),
|
|
152
|
+
value: typeof value === "string" ? value : null,
|
|
153
|
+
showTime: (meta == null ? void 0 : meta.includeTime) === true,
|
|
154
|
+
onChange: (v) => onCommit == null ? void 0 : onCommit(v),
|
|
155
|
+
defaultOpen: true,
|
|
156
|
+
onOpenChange: dismissOnClose(onCancel)
|
|
157
|
+
},
|
|
158
|
+
"edit"
|
|
159
|
+
);
|
|
160
|
+
}
|
|
161
|
+
function TimeCell({ value, meta, mode, size, isDisabled, isEditable, onCommit, onCancel }) {
|
|
162
|
+
if (mode === "display") {
|
|
163
|
+
return /* @__PURE__ */ jsx(
|
|
164
|
+
TimePicker,
|
|
165
|
+
{
|
|
166
|
+
variant: "naked",
|
|
167
|
+
mode: displayOrDisabled(isDisabled),
|
|
168
|
+
value,
|
|
169
|
+
size,
|
|
170
|
+
formatOptions: meta == null ? void 0 : meta.formatOptions,
|
|
171
|
+
locale: meta == null ? void 0 : meta.locale,
|
|
172
|
+
showDisplayEndIcon: isEditable === true
|
|
173
|
+
},
|
|
174
|
+
"display"
|
|
175
|
+
);
|
|
176
|
+
}
|
|
177
|
+
return /* @__PURE__ */ jsx(
|
|
178
|
+
TimePicker,
|
|
179
|
+
{
|
|
180
|
+
variant: "naked",
|
|
181
|
+
size: sizeForInput(size),
|
|
182
|
+
value: typeof value === "string" ? value : null,
|
|
183
|
+
showSeconds: (meta == null ? void 0 : meta.showSeconds) === true,
|
|
184
|
+
minuteStep: meta == null ? void 0 : meta.minuteStep,
|
|
185
|
+
secondStep: meta == null ? void 0 : meta.secondStep,
|
|
186
|
+
onChange: (v) => onCommit == null ? void 0 : onCommit(v),
|
|
187
|
+
defaultOpen: true,
|
|
188
|
+
onOpenChange: dismissOnClose(onCancel)
|
|
189
|
+
},
|
|
190
|
+
"edit"
|
|
191
|
+
);
|
|
192
|
+
}
|
|
193
|
+
function SelectCell({ value, meta, mode, size, isDisabled, isEditable, onCommit, onCancel }) {
|
|
194
|
+
const displayMode = (meta == null ? void 0 : meta.display) ?? "plain";
|
|
195
|
+
if (mode === "display") {
|
|
196
|
+
return /* @__PURE__ */ jsx(
|
|
197
|
+
Select,
|
|
198
|
+
{
|
|
199
|
+
variant: "naked",
|
|
200
|
+
mode: displayOrDisabled(isDisabled),
|
|
201
|
+
value,
|
|
202
|
+
options: (meta == null ? void 0 : meta.options) ?? [],
|
|
203
|
+
size,
|
|
204
|
+
display: displayMode,
|
|
205
|
+
showDisplayEndIcon: isEditable === true
|
|
206
|
+
},
|
|
207
|
+
"display"
|
|
208
|
+
);
|
|
209
|
+
}
|
|
210
|
+
return /* @__PURE__ */ jsx(
|
|
211
|
+
Select,
|
|
212
|
+
{
|
|
213
|
+
autoFocus: true,
|
|
214
|
+
variant: "naked",
|
|
215
|
+
size: sizeForInput(size),
|
|
216
|
+
options: (meta == null ? void 0 : meta.options) ?? [],
|
|
217
|
+
value,
|
|
218
|
+
onChange: (v) => onCommit == null ? void 0 : onCommit(v),
|
|
219
|
+
searchable: (meta == null ? void 0 : meta.searchable) === true,
|
|
220
|
+
display: displayMode,
|
|
221
|
+
defaultOpen: true,
|
|
222
|
+
onOpenChange: dismissOnClose(onCancel)
|
|
223
|
+
},
|
|
224
|
+
"edit"
|
|
225
|
+
);
|
|
226
|
+
}
|
|
227
|
+
function MultiSelectCell({ value, meta, mode, size, isDisabled, autoRowHeight, isEditable, onCommitLive, onCancel }) {
|
|
228
|
+
const wrap = autoRowHeight && (meta == null ? void 0 : meta.wrap) === true;
|
|
229
|
+
if (mode === "display") {
|
|
230
|
+
return /* @__PURE__ */ jsx(
|
|
231
|
+
Combobox,
|
|
232
|
+
{
|
|
233
|
+
variant: "naked",
|
|
234
|
+
mode: displayOrDisabled(isDisabled),
|
|
235
|
+
value: value ?? [],
|
|
236
|
+
options: (meta == null ? void 0 : meta.options) ?? [],
|
|
237
|
+
wrap,
|
|
238
|
+
size,
|
|
239
|
+
showDisplayEndIcon: isEditable === true
|
|
240
|
+
},
|
|
241
|
+
"display"
|
|
242
|
+
);
|
|
243
|
+
}
|
|
244
|
+
return /* @__PURE__ */ jsx(
|
|
245
|
+
Combobox,
|
|
246
|
+
{
|
|
247
|
+
variant: "naked",
|
|
248
|
+
size: sizeForInput(size),
|
|
249
|
+
options: (meta == null ? void 0 : meta.options) ?? [],
|
|
250
|
+
value: Array.isArray(value) ? value : [],
|
|
251
|
+
onChange: (v) => onCommitLive == null ? void 0 : onCommitLive(v),
|
|
252
|
+
defaultOpen: true,
|
|
253
|
+
onOpenChange: dismissOnClose(onCancel)
|
|
254
|
+
},
|
|
255
|
+
"edit"
|
|
256
|
+
);
|
|
257
|
+
}
|
|
258
|
+
function PersonCell({ value, mode, size, isDisabled, isEditable, onCommit, onCancel, meta }) {
|
|
259
|
+
if (mode === "display") {
|
|
260
|
+
return /* @__PURE__ */ jsx(PeoplePicker, { variant: "naked", mode: displayOrDisabled(isDisabled), value, size, showDisplayEndIcon: isEditable === true }, "display");
|
|
261
|
+
}
|
|
262
|
+
return /* @__PURE__ */ jsx(
|
|
263
|
+
PeoplePicker,
|
|
264
|
+
{
|
|
265
|
+
variant: "naked",
|
|
266
|
+
size: sizeForInput(size),
|
|
267
|
+
value,
|
|
268
|
+
people: (meta == null ? void 0 : meta.people) ?? [],
|
|
269
|
+
onChange: (next) => onCommit == null ? void 0 : onCommit(next[0] ?? null),
|
|
270
|
+
defaultOpen: true,
|
|
271
|
+
onOpenChange: dismissOnClose(onCancel)
|
|
272
|
+
},
|
|
273
|
+
"edit"
|
|
274
|
+
);
|
|
275
|
+
}
|
|
276
|
+
function MultiPersonCell({ value, mode, size, isDisabled, isEditable, onCommitLive, onCancel, meta }) {
|
|
277
|
+
if (mode === "display") {
|
|
278
|
+
return /* @__PURE__ */ jsx(PeoplePicker, { variant: "naked", mode: displayOrDisabled(isDisabled), value: value ?? [], size, showDisplayEndIcon: isEditable === true }, "display");
|
|
279
|
+
}
|
|
280
|
+
return /* @__PURE__ */ jsx(
|
|
281
|
+
PeoplePicker,
|
|
282
|
+
{
|
|
283
|
+
variant: "naked",
|
|
284
|
+
size: sizeForInput(size),
|
|
285
|
+
value: Array.isArray(value) ? value : [],
|
|
286
|
+
people: (meta == null ? void 0 : meta.people) ?? [],
|
|
287
|
+
onChange: (next) => onCommitLive == null ? void 0 : onCommitLive(next),
|
|
288
|
+
defaultOpen: true,
|
|
289
|
+
onOpenChange: dismissOnClose(onCancel)
|
|
290
|
+
},
|
|
291
|
+
"edit"
|
|
292
|
+
);
|
|
293
|
+
}
|
|
294
|
+
function BooleanCell({ value, mode, meta, size, isEditable, isDisabled, onCommit }) {
|
|
295
|
+
if (mode === "display" && (!isEditable || isDisabled)) {
|
|
296
|
+
return /* @__PURE__ */ jsx(Checkbox, { variant: "naked", mode: displayOrDisabled(isDisabled), checked: value === true });
|
|
297
|
+
}
|
|
298
|
+
return /* @__PURE__ */ jsx(
|
|
299
|
+
Checkbox,
|
|
300
|
+
{
|
|
301
|
+
size: size === "lg" ? "lg" : "md",
|
|
302
|
+
checked: value === true,
|
|
303
|
+
onCheckedChange: (checked) => onCommit == null ? void 0 : onCommit(checked === true),
|
|
304
|
+
"aria-label": (meta == null ? void 0 : meta.ariaLabel) ?? "切換"
|
|
305
|
+
}
|
|
306
|
+
);
|
|
307
|
+
}
|
|
308
|
+
function UrlCell({ value, meta, mode, size, isDisabled, isEditable, onRequestEdit, onCommit, onCancel }) {
|
|
309
|
+
if (mode === "display") {
|
|
310
|
+
const display = /* @__PURE__ */ jsx(LinkInput, { variant: "naked", mode: displayOrDisabled(isDisabled), value, label: meta == null ? void 0 : meta.linkLabel, size, showDisplayEndIcon: true });
|
|
311
|
+
if (!isEditable || isDisabled) return display;
|
|
312
|
+
return /* @__PURE__ */ jsxs("span", { className: "group/cell relative flex items-center w-full", children: [
|
|
313
|
+
" ",
|
|
314
|
+
/* @__PURE__ */ jsx("span", { className: "flex-1 min-w-0", children: display }),
|
|
315
|
+
/* @__PURE__ */ jsx(
|
|
316
|
+
Button,
|
|
317
|
+
{
|
|
318
|
+
variant: "tertiary",
|
|
319
|
+
size: "xs",
|
|
320
|
+
iconOnly: true,
|
|
321
|
+
startIcon: Pencil,
|
|
322
|
+
"aria-label": "編輯連結",
|
|
323
|
+
className: cn("ml-1 opacity-0 group-hover/cell:opacity-100 transition-opacity"),
|
|
324
|
+
onClick: (e) => {
|
|
325
|
+
e.stopPropagation();
|
|
326
|
+
onRequestEdit == null ? void 0 : onRequestEdit();
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
)
|
|
330
|
+
] });
|
|
331
|
+
}
|
|
332
|
+
return /* @__PURE__ */ jsx(
|
|
333
|
+
Input,
|
|
334
|
+
{
|
|
335
|
+
autoFocus: true,
|
|
336
|
+
variant: "naked",
|
|
337
|
+
size: sizeForInput(size),
|
|
338
|
+
defaultValue: value != null ? String(value) : "",
|
|
339
|
+
onBlur: (e) => onCommit == null ? void 0 : onCommit(e.target.value),
|
|
340
|
+
onKeyDown: makeKeyHandler(onCommit, onCancel)
|
|
341
|
+
}
|
|
342
|
+
);
|
|
343
|
+
}
|
|
344
|
+
const cellRegistry = {
|
|
345
|
+
string: StringCell,
|
|
346
|
+
number: NumberCell,
|
|
347
|
+
currency: NumberCell,
|
|
348
|
+
// 共用 NumberCell — currency-ness 走 meta.type 判 prefix='$'
|
|
349
|
+
date: DateCell,
|
|
350
|
+
time: TimeCell,
|
|
351
|
+
select: SelectCell,
|
|
352
|
+
multiSelect: MultiSelectCell,
|
|
353
|
+
person: PersonCell,
|
|
354
|
+
multiPerson: MultiPersonCell,
|
|
355
|
+
boolean: BooleanCell,
|
|
356
|
+
url: UrlCell
|
|
357
|
+
};
|
|
358
|
+
const cellWithSurfaceCache = /* @__PURE__ */ new Map();
|
|
359
|
+
function buildCellWithSurface(Inner, key) {
|
|
360
|
+
const CellWithSurface = React.memo(function CellWithSurface2(props) {
|
|
361
|
+
return /* @__PURE__ */ jsx(FieldSurfaceProvider, { surface: "table-cell", children: /* @__PURE__ */ jsx(Inner, { ...props }) });
|
|
362
|
+
});
|
|
363
|
+
CellWithSurface.displayName = `CellWithSurface(${key})`;
|
|
364
|
+
return CellWithSurface;
|
|
365
|
+
}
|
|
366
|
+
for (const type of Object.keys(cellRegistry)) {
|
|
367
|
+
cellWithSurfaceCache.set(type, buildCellWithSurface(cellRegistry[type], type));
|
|
368
|
+
}
|
|
369
|
+
cellWithSurfaceCache.set("_default_", buildCellWithSurface(StringCell, "StringCell-fallback"));
|
|
370
|
+
function resolveCellComponent(type) {
|
|
371
|
+
return cellWithSurfaceCache.get(type ?? "_default_") ?? cellWithSurfaceCache.get("_default_");
|
|
372
|
+
}
|
|
373
|
+
export {
|
|
374
|
+
cellRegistry,
|
|
375
|
+
resolveCellComponent
|
|
376
|
+
};
|
|
377
|
+
//# sourceMappingURL=cell-registry.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cell-registry.js","sources":["../../../src/components/DataTable/cell-registry.tsx"],"sourcesContent":["// @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.\n// code-quality-allow: file-size — Cell Registry 含 10 cell-type components(string/number/date/time/select/multiSelect/person/multiPerson/boolean/url)+ shared helpers,split-into-files 會破壞 type-keyed registry SSOT canonical\n// DataTable Cell Registry — type-keyed SSOT for cell rendering(Phase C 2026-05-05)\n//\n// 對齊 M17 SSOT consolidation + audit recommendation:\n// 原 `renderTypedValue` switch + `EditableCellContent` switch 兩條平行 type-switch 已 collapse\n// 為**一張 type → cell component** registry。每個 cell component 同時處理 display / edit mode,\n// 靠底層 Field control 的 `mode` prop 切換。\n//\n// 設計原則:\n// - 每個 cell component 接同一組 props(`CellComponentProps`)\n// - 用 `variant=\"naked\"` — DataTable cell-as-input substrate(對齊 Field B1 chrome=bare)\n// - 消費 full Field 家族 primitive(無 stub)\n// - 不再用 `meta._editable` 私有 flag — `isEditable` 直接顯式入參(消除 M1 hack)\n//\n// World-class 對照(@benchmark-unverified):AG Grid cellRendererSelector / Material X-Grid\n// `valueGetter + renderCell` / Notion property type registry。\n\nimport * as React from 'react'\nimport type { ComponentType } from 'react'\nimport { Pencil } from 'lucide-react'\nimport { cn } from '@/lib/utils'\nimport type { ColumnType } from './column-types'\nimport { Input } from '@/design-system/components/Input/input'\nimport { Textarea } from '@/design-system/components/Textarea/textarea'\nimport { NumberInput } from '@/design-system/components/NumberInput/number-input'\nimport { Select } from '@/design-system/components/Select/select'\nimport { Combobox } from '@/design-system/components/Combobox/combobox'\nimport { DatePicker } from '@/design-system/components/DatePicker/date-picker'\nimport { TimePicker } from '@/design-system/components/TimePicker/time-picker'\nimport { PeoplePicker } from '@/design-system/components/PeoplePicker/people-picker'\nimport { LinkInput } from '@/design-system/components/LinkInput/link-input'\nimport { Checkbox } from '@/design-system/components/Checkbox/checkbox'\nimport { Button } from '@/design-system/components/Button/button'\nimport type { PersonValue } from '@/design-system/components/PeoplePicker/person-display'\nimport { FieldSurfaceProvider } from '@/design-system/components/Field/field-context'\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type CellMode = 'display' | 'edit'\nexport type CellSize = 'sm' | 'md' | 'lg'\n\nexport interface CellComponentProps {\n // any-allow: free-form column value(consumer-defined,跨 type 共用 signature)\n value: any\n // any-allow: free-form consumer meta bag(prefix / options / formatOptions / locale / linkLabel 等)\n meta: Record<string, any>\n mode: CellMode\n size: CellSize\n autoRowHeight: boolean\n /** 該 cell 是否可編。replaces 舊 `meta._editable` 私有 flag(Phase C M1 hack 移除)。 */\n isEditable?: boolean\n /** 2026-05-13:cell 是否 disabled(state overlay,orthogonal to display/edit lifecycle)。\n * per codex Q3 verdict:不擴 CellMode='disabled',加 prop。各 Cell function 收 true 時\n * 傳 `mode='disabled'` 給 inner Field control,各 Field 內部走具體 disabled token(非 wrapper opacity)。 */\n isDisabled?: boolean\n /** Cell 進 edit mode → 提交新值(blur / Enter / option select 都觸發)— 提交後**自動 exit edit**。\n * 適用 single-shot commit:string / number / select(single)/ person(single)/ date / time / boolean / url。 */\n onCommit?: (next: unknown) => void\n /** Live commit — 提交新值但 **不 exit edit**(popover 持續開)。\n * 適用 multi-select 類:multiSelect / multiPerson — user 連續勾選,直到點外面才關。\n * 對齊 Notion / Linear / Airtable canonical:multi-pick popover 不在每次 toggle 後關閉。 */\n onCommitLive?: (next: unknown) => void\n /** Esc 取消編輯,不 commit。 */\n onCancel?: () => void\n /** URL cell 專用:hover 顯示的 Pencil 鈕 → 進 edit mode(read mode 保留 link click 語意)。 */\n onRequestEdit?: () => void\n /** Per-keystroke draft propagation(2026-05-10 Phase 7 D.3 portal Field virtualizer unmount preserve draft):\n * Edit mode 內部 input onChange/onValueChange 每 keystroke 呼叫 onDraft,讓 lifted draft state(in\n * data-table.tsx)持有 user 編輯中字。Cell DOM unmount(virtualizer scroll out)時 draft 在\n * parent state 不丟;mount-back 時 portal Field value 從 draft 取,user 字保留。\n * 非 portal mode(inline edit)不傳此 prop,各 Cell 走原 uncontrolled defaultValue 路徑。 */\n onDraft?: (next: unknown) => void\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\n/** 鍵盤 commit / cancel — string / number cell edit mode 共用 */\nfunction makeKeyHandler(\n onCommit?: (v: unknown) => void,\n onCancel?: () => void,\n parseValue?: (raw: string) => unknown,\n) {\n return (e: React.KeyboardEvent<HTMLInputElement>) => {\n if (e.key === 'Escape') { e.preventDefault(); onCancel?.() }\n if (e.key === 'Enter') {\n e.preventDefault()\n const raw = (e.target as HTMLInputElement).value\n onCommit?.(parseValue ? parseValue(raw) : raw)\n }\n }\n}\n\nconst sizeForInput = (size: CellSize): CellSize => size\n\n/** 2026-05-13 Q3 helper(per codex Q3 verdict):cell display + isDisabled → Field mode='disabled'。\n * Cell display lifecycle 不擴 CellMode='disabled',而是各 Cell 在 display branch 翻譯 isDisabled\n * → Field mode='disabled' prop。inner Field 內部走具體 disabled token(text-fg-disabled / bg-disabled 等),\n * 非 wrapper blanket opacity-disabled 逃生艙(per color.spec.md:729)。 */\nconst displayOrDisabled = (isDisabled?: boolean): 'display' | 'disabled' =>\n isDisabled ? 'disabled' : 'display'\n\n// ── Cell Components ──────────────────────────────────────────────────────────\n\nfunction StringCell({ value, meta, mode, size, isDisabled, autoRowHeight, onCommit, onCancel, onDraft }: CellComponentProps) {\n // 2026-05-14 I9 fix(per codex+Layer A 共識):meta.maxLines opt-in line-clamp。\n // display autoRow 用 Tailwind arbitrary line-clamp 支援 N rows;edit textarea field-sizing\n // 已 auto-grow to content,natural match clamp。\n // 2026-05-16 Round 5 audit Dim 27 fix:narrow type 取代 `as any` cast。\n const maxLines: number | undefined = (meta as { maxLines?: number } | undefined)?.maxLines\n const clampClass = maxLines && autoRowHeight ? `line-clamp-[${maxLines}]` : undefined\n // string type canonical(2026-05-05 v2 user 校正:input space ≥ display space):\n // - autoRowHeight: Textarea(display + edit)— display wrap text 撐高 row,edit textarea\n // 多行輸入、`!h-full` 填 cell。對齊 Notion long-text cell canonical。\n // - fixed: Input(display + edit)— 單行 truncate display,單行 input edit;Field naked intrinsic\n // 高 = cell 高 = h-field-md,文字位置 display↔edit 完全一致。對齊 AG Grid / Material X-Grid。\n // - autoRowHeight 是 table 框架決定(consumer 不需 per-column 設 meta.wrap)。\n // - 互動(Textarea):Esc cancel / Cmd|Ctrl+Enter commit / blur commit;Enter 保留換行\n // - 互動(Input):Esc cancel / Enter commit / blur commit\n const v = value != null ? String(value) : ''\n if (mode === 'display') {\n return autoRowHeight\n ? <Textarea variant=\"naked\" mode={displayOrDisabled(isDisabled)} value={v} className={clampClass} />\n : <Input variant=\"naked\" mode={displayOrDisabled(isDisabled)} value={v} />\n }\n if (autoRowHeight) {\n // 2026-05-14 I8 fix(per codex verdict + user 抓「edit cell shrink」):\n // 原 `wrapRows = value.length / 40` 字元估算不準(對應實際 column width 不同\n // → cell 進 edit shrink)。改 CSS `field-sizing: content`(Chrome 123+ / FF 122+ /\n // Safari 17+)讓 textarea 自動 grow to content,匹配 display wrap 真實高度。\n // Fallback rows 仍保留給舊 browser(rows attr 在 field-sizing 支援時被覆蓋)。\n const newlineRows = (v.match(/\\n/g) || []).length + 1\n const wrapRows = Math.ceil(v.length / 40)\n const estimateRows = Math.min(10, Math.max(1, newlineRows, wrapRows))\n return (\n <Textarea\n autoFocus\n variant=\"naked\"\n size={sizeForInput(size)}\n rows={estimateRows}\n defaultValue={v}\n // any-allow: CSS `field-sizing` 屬性 Chrome 123+/FF 122+/Safari 17+ 支援但 TypeScript lib.dom\n // 尚未加 type;narrow 到 CSSProperties 仍需 cast,保留 single-site any 較 type aug 簡潔。\n style={{ fieldSizing: 'content' } as React.CSSProperties}\n onChange={(e) => onDraft?.((e.target as HTMLTextAreaElement).value)}\n onBlur={(e) => onCommit?.((e.target as HTMLTextAreaElement).value)}\n onKeyDown={(e) => {\n if (e.key === 'Escape') { e.preventDefault(); onCancel?.() }\n if (e.key === 'Enter' && (e.metaKey || e.ctrlKey)) {\n e.preventDefault()\n onCommit?.((e.target as HTMLTextAreaElement).value)\n }\n }}\n />\n )\n }\n return (\n <Input\n autoFocus\n variant=\"naked\"\n size={sizeForInput(size)}\n defaultValue={v}\n onChange={(e) => onDraft?.(e.target.value)}\n onBlur={(e) => onCommit?.(e.target.value)}\n onKeyDown={makeKeyHandler(onCommit, onCancel)}\n />\n )\n}\n\nfunction NumberCell({ value, meta, mode, size, isDisabled, onCommit, onCancel, onDraft }: CellComponentProps) {\n // currency 透過 columnType-aware prefix:type='currency' → 預設 '$'(可 override)\n const isCurrency = meta?.type === 'currency'\n const prefix = isCurrency ? (meta?.prefix ?? '$') : meta?.prefix\n if (mode === 'display') {\n return (\n <NumberInput\n variant=\"naked\"\n mode={displayOrDisabled(isDisabled)}\n value={value as number | null}\n prefix={prefix}\n suffix={meta?.suffix}\n precision={meta?.precision}\n locale={meta?.locale}\n />\n )\n }\n // Edit mode value pre-fill canonical(2026-05-05):NumberInput edit 強制 controlled\n // (`value={value ?? ''}`)— 若 NumberCell 以 `defaultValue` 傳入,NumberInput value=undefined → ''\n // empty。對齊 cell-as-input「edit mode 自動帶入 display 值」(對齊 Notion / Airtable 共識),\n // 改用 local state controlled。User typing → setLocalValue;blur/Enter → onCommit(localValue)。\n const initial = typeof value === 'number' ? value : null\n const [localValue, setLocalValue] = React.useState<number | null>(initial)\n return (\n <NumberInput\n autoFocus\n variant=\"naked\"\n size={sizeForInput(size)}\n value={localValue}\n onChange={(v) => { setLocalValue(v); onDraft?.(v) }}\n prefix={prefix}\n suffix={meta?.suffix}\n precision={meta?.precision}\n onBlur={() => onCommit?.(localValue)}\n onKeyDown={(e) => {\n if (e.key === 'Escape') { e.preventDefault(); onCancel?.() }\n if (e.key === 'Enter') { e.preventDefault(); onCommit?.(localValue) }\n }}\n />\n )\n}\n\n// Cell-as-input dismiss canonical(2026-05-05):defaultOpen=true 開始 → user click 外 popover 關\n// → 元件 fire onOpenChange(false) → cell call onCancel exit edit。否則 cell 卡 edit mode 不可 re-trigger\n// (對齊 Airtable / Notion canonical:click 外即關)。\nconst dismissOnClose = (onCancel?: () => void) => (open: boolean) => { if (!open) onCancel?.() }\n\n// Mode-keyed remount canonical(2026-05-05):display↔edit 切換時,因 React reconciliation 同 type 同\n// position 會重用 instance,導致 `useState(defaultOpen)` 只在首次 mount 跑(那時 mode='display'\n// defaultOpen 沒給→預設 false)。後續 mode='edit' 即使傳 defaultOpen=true 也無效。\n// Fix:`key={mode}` 強制 React unmount + remount,每次切 mode 都重跑 useState init。\n// 對齊 Notion / Airtable cell-as-input「display 跟 edit 是不同 mount cycle」語義。\n\nfunction DateCell({ value, meta, mode, size, isDisabled, isEditable, onCommit, onCancel }: CellComponentProps) {\n if (mode === 'display') {\n return (\n <DatePicker\n key=\"display\"\n variant=\"naked\"\n mode={displayOrDisabled(isDisabled)}\n value={value as string | null}\n size={size}\n formatOptions={meta?.formatOptions}\n locale={meta?.locale}\n // Indicator(calendar icon)= editable affordance(2026-05-10 user 糾正)。\n // Non-editable cell 不該顯 picker indicator(誤導 read-only 為 editable)。\n // 對齊 UrlCell L394 / BooleanCell L368 既有 isEditable conditional pattern。\n showDisplayEndIcon={isEditable === true}\n />\n )\n }\n return (\n <DatePicker\n key=\"edit\"\n autoFocus\n variant=\"naked\"\n size={sizeForInput(size)}\n value={typeof value === 'string' ? value : null}\n showTime={meta?.includeTime === true}\n onChange={(v) => onCommit?.(v)}\n defaultOpen\n onOpenChange={dismissOnClose(onCancel)}\n />\n )\n}\n\nfunction TimeCell({ value, meta, mode, size, isDisabled, isEditable, onCommit, onCancel }: CellComponentProps) {\n if (mode === 'display') {\n return (\n <TimePicker\n key=\"display\"\n variant=\"naked\"\n mode={displayOrDisabled(isDisabled)}\n value={value as string | null}\n size={size}\n formatOptions={meta?.formatOptions}\n locale={meta?.locale}\n showDisplayEndIcon={isEditable === true} // 2026-05-10:non-editable 不顯 indicator\n />\n )\n }\n return (\n <TimePicker\n key=\"edit\"\n variant=\"naked\"\n size={sizeForInput(size)}\n value={typeof value === 'string' ? value : null}\n showSeconds={meta?.showSeconds === true}\n minuteStep={meta?.minuteStep}\n secondStep={meta?.secondStep}\n onChange={(v) => onCommit?.(v)}\n defaultOpen\n onOpenChange={dismissOnClose(onCancel)}\n />\n )\n}\n\nfunction SelectCell({ value, meta, mode, size, isDisabled, isEditable, onCommit, onCancel }: CellComponentProps) {\n // Display canonical(2026-05-05):cell IS variant,default plain text(no Tag pill 疊在 cell border 內)。\n // Consumer 可在 column meta.display='tag' opt-in 內容導向的 Tag 視覺(category 含色彩標籤等)。\n // 對齊 JTable / AG Grid「renderer/editor 視覺一致」canonical。\n const displayMode = (meta?.display as 'plain' | 'tag' | undefined) ?? 'plain'\n if (mode === 'display') {\n return (\n <Select\n key=\"display\"\n variant=\"naked\"\n mode={displayOrDisabled(isDisabled)}\n value={value as string | null}\n options={meta?.options ?? []}\n size={size}\n display={displayMode}\n showDisplayEndIcon={isEditable === true} // 2026-05-10:non-editable 不顯 chevron indicator\n />\n )\n }\n return (\n <Select\n key=\"edit\"\n autoFocus\n variant=\"naked\"\n size={sizeForInput(size)}\n options={meta?.options ?? []}\n value={value as string | null | undefined}\n onChange={(v) => onCommit?.(v)}\n // B7(2026-05-05):cell 編輯時支援 inline search,沿用 Select.searchable 機制(對齊 cell-as-input\n // 「沿用既有輸入框互動」原則)。Default false,consumer 在 meta.searchable 開啟。\n searchable={meta?.searchable === true}\n display={displayMode}\n defaultOpen\n onOpenChange={dismissOnClose(onCancel)}\n />\n )\n}\n\nfunction MultiSelectCell({ value, meta, mode, size, isDisabled, autoRowHeight, isEditable, onCommitLive, onCancel }: CellComponentProps) {\n const wrap = autoRowHeight && meta?.wrap === true\n if (mode === 'display') {\n return (\n <Combobox\n key=\"display\"\n variant=\"naked\"\n mode={displayOrDisabled(isDisabled)}\n value={(value as string[] | null) ?? []}\n options={meta?.options ?? []}\n wrap={wrap}\n size={size}\n showDisplayEndIcon={isEditable === true} // 2026-05-10:non-editable 不顯 chevron indicator\n />\n )\n }\n // Multi 用 onCommitLive(commit 但不 exit edit)— 每勾一項即時生效,popover 持續開\n // 直到點外面;onOpenChange(false) → onCancel exit edit。對齊 Notion / Linear / Airtable canonical。\n return (\n <Combobox\n key=\"edit\"\n variant=\"naked\"\n size={sizeForInput(size)}\n options={meta?.options ?? []}\n value={Array.isArray(value) ? (value as string[]) : []}\n onChange={(v) => onCommitLive?.(v)}\n defaultOpen\n onOpenChange={dismissOnClose(onCancel)}\n />\n )\n}\n\nfunction PersonCell({ value, mode, size, isDisabled, isEditable, onCommit, onCancel, meta }: CellComponentProps) {\n if (mode === 'display') {\n // 2026-05-10:non-editable 不顯 chevron indicator(對齊 UrlCell isEditable conditional pattern)\n return <PeoplePicker key=\"display\" variant=\"naked\" mode={displayOrDisabled(isDisabled)} value={value as PersonValue | null} size={size} showDisplayEndIcon={isEditable === true} />\n }\n return (\n <PeoplePicker\n key=\"edit\"\n variant=\"naked\"\n size={sizeForInput(size)}\n value={value as PersonValue | null}\n people={meta?.people ?? []}\n // PeoplePicker onChange 永遠 emit array(API contract);single mode commit 取首位\n onChange={(next) => onCommit?.(next[0] ?? null)}\n defaultOpen\n onOpenChange={dismissOnClose(onCancel)}\n />\n )\n}\n\nfunction MultiPersonCell({ value, mode, size, isDisabled, isEditable, onCommitLive, onCancel, meta }: CellComponentProps) {\n if (mode === 'display') {\n // 2026-05-10:non-editable 不顯 chevron indicator\n return <PeoplePicker key=\"display\" variant=\"naked\" mode={displayOrDisabled(isDisabled)} value={(value as PersonValue[]) ?? []} size={size} showDisplayEndIcon={isEditable === true} />\n }\n // Multi 用 onCommitLive(commit 但不 exit edit)— 每勾一人即時生效,popover 持續開\n // 直到點外面;onOpenChange(false) → onCancel exit edit。對齊 multiSelect canonical。\n return (\n <PeoplePicker\n key=\"edit\"\n variant=\"naked\"\n size={sizeForInput(size)}\n value={Array.isArray(value) ? (value as PersonValue[]) : []}\n people={meta?.people ?? []}\n onChange={(next) => onCommitLive?.(next)}\n defaultOpen\n onOpenChange={dismissOnClose(onCancel)}\n />\n )\n}\n\nfunction BooleanCell({ value, mode, meta, size, isEditable, isDisabled, onCommit }: CellComponentProps) {\n // boolean 不分 read/edit mode — display 渲 mode='display' 純展示;editable 時直接 toggle Checkbox\n // 2026-05-13 codex V1 fix:editable=true + disabled=true 之前 fall through to live Checkbox,\n // onCheckedChange 仍 fire(violate disabled contract)。Fix:`!isEditable || isDisabled` →\n // 走 display branch,Checkbox 拿 disabled mode + 不接 onCheckedChange。\n if (mode === 'display' && (!isEditable || isDisabled)) {\n return <Checkbox variant=\"naked\" mode={displayOrDisabled(isDisabled)} checked={value === true} />\n }\n return (\n <Checkbox\n size={size === 'lg' ? 'lg' : 'md'}\n checked={value === true}\n onCheckedChange={(checked) => onCommit?.(checked === true)}\n aria-label={meta?.ariaLabel ?? '切換'}\n />\n )\n}\n\n/**\n * UrlCell — Phase C drift fix:\n * 舊 EditableCellContent edit mode 對 url 走 plain `<Input>`(失去 URL 驗證 + auto-link)。\n * 現改用 `<LinkInput>` edit mode → 保留 URL parse / hostname 顯示一致性 + 鍵盤 commit / cancel。\n * read mode 仍 `<LinkInput mode={displayOrDisabled(isDisabled)}>` = 一致 SSOT。\n * editable 互動:hover 時右側出 Pencil 鈕 → 進 edit(保留 link click 語意,對齊原 spec)。\n */\nfunction UrlCell({ value, meta, mode, size, isDisabled, isEditable, onRequestEdit, onCommit, onCancel }: CellComponentProps) {\n if (mode === 'display') {\n // showDisplayEndIcon ← D path Phase 2(2026-05-08):Field naked wrapper 包 anchor,與 Input edit 同 chrome\n const display = (\n <LinkInput variant=\"naked\" mode={displayOrDisabled(isDisabled)} value={value as string | null} label={meta?.linkLabel} size={size} showDisplayEndIcon />\n )\n // 2026-05-13 codex V1 fix:disabled URL 不顯 Pencil affordance(parent onRequestEdit 已被攔但 UI 仍誤導)\n if (!isEditable || isDisabled) return display\n // editable read mode:hover Pencil 鈕(對齊 spec 第十二段「url:read = 連結 + Pencil」)\n return (\n <span className=\"group/cell relative flex items-center w-full\"> {/* @naked-row-mode-allow: URL hover-Pencil 是 inline action 不是 value content,items-center 鎖 Pencil 對齊行高第一行(autoRow 跟 fixed 皆同視覺正確) */}\n <span className=\"flex-1 min-w-0\">{display}</span>\n <Button\n variant=\"tertiary\"\n size=\"xs\"\n iconOnly\n startIcon={Pencil}\n aria-label=\"編輯連結\"\n className={cn('ml-1 opacity-0 group-hover/cell:opacity-100 transition-opacity')}\n onClick={(e) => {\n e.stopPropagation()\n onRequestEdit?.()\n }}\n />\n </span>\n )\n }\n // edit mode value pre-fill canonical(2026-05-05):LinkInput edit `value` prop 強制 controlled\n // (line 113 `useState(value ?? '')`)+ `showLink = !editing && hasValidValue` 預設顯 link 不顯 input\n // → cell-as-input editing 場景需要 input 直接 focus 編輯。改用 plain `<Input>`(uncontrolled\n // `defaultValue` 正確 pre-fill,Input.tsx `value={value}` 是 undefined → uncontrolled 走 defaultValue)。\n // URL 驗證等 deferred 到 commit phase(consumer 可在 onCommit 時 validate)。\n return (\n <Input\n autoFocus\n variant=\"naked\"\n size={sizeForInput(size)}\n defaultValue={value != null ? String(value) : ''}\n onBlur={(e) => onCommit?.(e.target.value)}\n onKeyDown={makeKeyHandler(onCommit, onCancel)}\n />\n )\n}\n\n// ── Registry ────────────────────────────────────────────────────────────────\n//\n// type → cell component。新增 columnType 必同步註冊一條(否則 fallback 到 string)。\n\nexport const cellRegistry: Record<ColumnType, ComponentType<CellComponentProps>> = {\n string: StringCell,\n number: NumberCell,\n currency: NumberCell, // 共用 NumberCell — currency-ness 走 meta.type 判 prefix='$'\n date: DateCell,\n time: TimeCell,\n select: SelectCell,\n multiSelect: MultiSelectCell,\n person: PersonCell,\n multiPerson: MultiPersonCell,\n boolean: BooleanCell,\n url: UrlCell,\n}\n\n/** Resolve cell component by type;default = StringCell(consumer 沒設 type 的 fallback)。\n * 2026-05-12 Stream C Cluster B fix:wrap with FieldSurfaceProvider `surface='table-cell'`\n * 讓所有 Cell 內的 Field family controls 透過 `useFieldSurface()` 取得「我在 cell 裡」context,\n * 取代散落的 `variant === 'naked'` cell-detection heuristic + per-prop hardcoded padding。\n *\n * **2026-05-13 (a) perf fix(user 拍板 + codex V1 verdict + Layer A grep root cause)**:\n * 原 factory pattern 每次 call 在 function body 內宣告新 `CellWithSurface` FC closure → 每 scroll\n * × 每 visible cell 都 return 新 FC reference → React 認 component type 變,**整 subtree mount/\n * unmount cascade**(Field + ItemPrefix/Suffix + Avatar / Tag / PersonDisplay)。\n * Fix:每 ColumnType **module-level 預建** wrapped FC + `React.memo`,resolve 走 cached map,\n * identity stable across scroll → memo 真生效 + subtree 不 mount/unmount。\n * Cite world-class:AG Grid「cell renderer per-type stable reference」/ MUI X DataGrid「memoized\n * subcomponents」/ Glide Data Grid「DOM virtualization 加解掛 = bottleneck」。 */\nconst cellWithSurfaceCache = new Map<ColumnType | '_default_', ComponentType<CellComponentProps>>()\n\nfunction buildCellWithSurface(Inner: ComponentType<CellComponentProps>, key: string): ComponentType<CellComponentProps> {\n const CellWithSurface = React.memo(function CellWithSurface(props: CellComponentProps) {\n return (\n <FieldSurfaceProvider surface=\"table-cell\">\n <Inner {...props} />\n </FieldSurfaceProvider>\n )\n })\n ;(CellWithSurface as { displayName?: string }).displayName = `CellWithSurface(${key})`\n return CellWithSurface as ComponentType<CellComponentProps>\n}\n\n// Pre-build per-type cached wrapped components(module-level,one-time init)\nfor (const type of Object.keys(cellRegistry) as ColumnType[]) {\n cellWithSurfaceCache.set(type, buildCellWithSurface(cellRegistry[type], type))\n}\ncellWithSurfaceCache.set('_default_', buildCellWithSurface(StringCell, 'StringCell-fallback'))\n\nexport function resolveCellComponent(type?: ColumnType): ComponentType<CellComponentProps> {\n return cellWithSurfaceCache.get(type ?? '_default_') ?? cellWithSurfaceCache.get('_default_')!\n}\n"],"names":["DatePicker","CellWithSurface"],"mappings":"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"}
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
import type { RowData } from '@tanstack/react-table';
|
|
2
|
+
export declare const columnTypes: readonly ["string", "number", "currency", "date", "time", "select", "multiSelect", "person", "multiPerson", "boolean", "url"];
|
|
3
|
+
export type ColumnType = (typeof columnTypes)[number];
|
|
4
|
+
export interface ColumnTypeConfig {
|
|
5
|
+
/** Default horizontal alignment */
|
|
6
|
+
align: 'left' | 'right' | 'center';
|
|
7
|
+
}
|
|
8
|
+
/** Default config per column type */
|
|
9
|
+
export declare const columnTypeDefaults: Record<ColumnType, ColumnTypeConfig>;
|
|
10
|
+
declare module '@tanstack/react-table' {
|
|
11
|
+
interface ColumnMeta<TData extends RowData, TValue> {
|
|
12
|
+
/**
|
|
13
|
+
* Column data type — determines default alignment, rendering, sorting, filtering.
|
|
14
|
+
*
|
|
15
|
+
* **Filterable column 必須設此 prop**(filter UI 只列有 `type` 的 column,
|
|
16
|
+
* 對齊 Notion / Airtable / Linear:每 property 強制有 type)。
|
|
17
|
+
*/
|
|
18
|
+
type?: ColumnType;
|
|
19
|
+
/** Override default alignment from column type */
|
|
20
|
+
align?: 'left' | 'right' | 'center';
|
|
21
|
+
/** Allow text wrapping (only effective when autoRowHeight is true) */
|
|
22
|
+
wrap?: boolean;
|
|
23
|
+
/**
|
|
24
|
+
* Max visible lines before ellipsis(2026-05-14 I9 per codex verdict + user 拍板)。
|
|
25
|
+
* Only effective when `autoRowHeight + wrap=true`。Display 用 `line-clamp-N`,edit textarea
|
|
26
|
+
* 高度 match clamp。對齊 Notion 「Truncate」row toggle / AG Grid `cellClassRules` 上限 row height。
|
|
27
|
+
* Default = undefined(無 clamp,完整 wrap)。
|
|
28
|
+
*/
|
|
29
|
+
maxLines?: number;
|
|
30
|
+
/**
|
|
31
|
+
* Column 寬度(px)— DS canonical 命名(2026-05-06 v14.3)。
|
|
32
|
+
*
|
|
33
|
+
* **為何不用 TanStack 的 `size`** — DS 內 `size` 既定意為元件 density string
|
|
34
|
+
* `'sm' | 'md' | 'lg'`(Field / Button / Input 等 49+ 處 use case)。Column 寬
|
|
35
|
+
* 是 px 數字,跟 density 概念衝突 → 走 DS-internal 命名 `width`/`minWidth`/`maxWidth`
|
|
36
|
+
* (對齊 CSS 原生 + AG Grid 的命名共識)。
|
|
37
|
+
*
|
|
38
|
+
* 內部 pre-process 會把 `meta.width` copy 到 TanStack 的 root `size`,確保 column
|
|
39
|
+
* resize feature(`enableColumnResize=true` 拖拉 + columnSizing state)正常運作。
|
|
40
|
+
*
|
|
41
|
+
* - **No resize mode**(default): `width` = column 寬度 reserve(cell ≥ width,
|
|
42
|
+
* flex 可 grow,**不可 shrink** 低於 width)。對齊 user 預期「我設多寬就多寬」。
|
|
43
|
+
* - **Resize mode**(`enableColumnResize=true`):`width` = 初始寬度,user 可拖拉
|
|
44
|
+
* 到 `minWidth` 為止。
|
|
45
|
+
*/
|
|
46
|
+
width?: number;
|
|
47
|
+
/**
|
|
48
|
+
* Column 最小寬度(px)— resize 模式下的拖拉下限。No resize 模式不使用(`width` 即下限)。
|
|
49
|
+
* Default(resize mode 沒明示時)= 80px(`MIN_COLUMN_WIDTH`)。
|
|
50
|
+
*/
|
|
51
|
+
minWidth?: number;
|
|
52
|
+
/**
|
|
53
|
+
* Column 最大寬度(px)— resize 模式下的拖拉上限。預設無上限。
|
|
54
|
+
*/
|
|
55
|
+
maxWidth?: number;
|
|
56
|
+
/**
|
|
57
|
+
* Column 是否可被 user 拖拉 resize(opt-out per col)。Default `true`(when `enableColumnResize` 也 true)。
|
|
58
|
+
*
|
|
59
|
+
* Set `false` 表「此 col 寬度由內容決定不允許 resize」(2026-05-10 加,per user
|
|
60
|
+
* 「操作列這種內容決定寬度的欄位應該不允許 resize」)。對齊 AG Grid `colDef.resizable` /
|
|
61
|
+
* Material X-DataGrid 同 API。
|
|
62
|
+
*
|
|
63
|
+
* System cols(__select__ / __drag__ / __actions__)自動 false(內部 hard-code,
|
|
64
|
+
* consumer 不需設)— 永遠固定寬。
|
|
65
|
+
*
|
|
66
|
+
* 典型 use case:custom 圖示欄、status indicator 欄、密度太緊不該 user 動的欄。
|
|
67
|
+
*/
|
|
68
|
+
resizable?: boolean;
|
|
69
|
+
/**
|
|
70
|
+
* Explicit opt-out from filter UI(預設 accessor column 有 type 即 filterable)。
|
|
71
|
+
*
|
|
72
|
+
* 用於:有 `type` 但不想在 filter UI 出現的 accessor column
|
|
73
|
+
* (例如:internal sorting key、composite display 用的 hidden column)。
|
|
74
|
+
*/
|
|
75
|
+
filterable?: boolean;
|
|
76
|
+
/** Number/currency formatting options */
|
|
77
|
+
prefix?: string;
|
|
78
|
+
suffix?: string;
|
|
79
|
+
precision?: number;
|
|
80
|
+
locale?: string;
|
|
81
|
+
/** Select/multiSelect options — maps value to display label */
|
|
82
|
+
options?: Array<{
|
|
83
|
+
value: string;
|
|
84
|
+
label: string;
|
|
85
|
+
}>;
|
|
86
|
+
/** Date: Intl.DateTimeFormat options */
|
|
87
|
+
formatOptions?: Intl.DateTimeFormatOptions;
|
|
88
|
+
/**
|
|
89
|
+
* Date: 是否含時間部分(datetime mode)。對齊 Notion idiom — 不另設 datetime column type。
|
|
90
|
+
*
|
|
91
|
+
* - `false`(default):cell 顯示與 filter 比對僅日期(day-level 精度)
|
|
92
|
+
* - `true`:cell 渲 date+time;filter 比對走 ms 精度(避開 Airtable 著名地雷)
|
|
93
|
+
*
|
|
94
|
+
* 在 advanced filter 中,`date` columnType 配 `includeTime=true` 時,
|
|
95
|
+
* `date_*` ValueShape 自動 promote 到 `datetime_*`,渲 `<DatePicker showTime>` /
|
|
96
|
+
* `<DatePickerRange showTime>`(詳 `filter-operators.ts` `getValueShape`)。
|
|
97
|
+
*/
|
|
98
|
+
includeTime?: boolean;
|
|
99
|
+
/** Link: 自訂顯示文字(不設則自動從 URL 提取 hostname) */
|
|
100
|
+
linkLabel?: string;
|
|
101
|
+
/**
|
|
102
|
+
* Inline edit:column 是否可編輯。
|
|
103
|
+
* - `true`:可編
|
|
104
|
+
* - `false`(default):唯讀
|
|
105
|
+
* - `(row) => boolean`:per-row 動態決定(e.g. row.status !== 'archived' 才能編)
|
|
106
|
+
*
|
|
107
|
+
* 互動 per type(對齊 Notion / Airtable):
|
|
108
|
+
* - string / number / currency:click cell → inline `<Input>` autoFocus + selected
|
|
109
|
+
* - date / select / multiSelect / person / multiPerson:click cell → 進 edit mode 的 Field control(用戶按 trigger 開 picker)
|
|
110
|
+
* - boolean:不分 read/edit mode,直接 `<Checkbox>` 點即 toggle + commit
|
|
111
|
+
* - url:read = 連結;**hover cell** 右側出 Pencil 按鈕,click 才進 `<Input>` edit mode(保留 click 連結語意)
|
|
112
|
+
*
|
|
113
|
+
* Esc cancel / blur or Enter commit。Commit 觸發 `onCellCommit`。
|
|
114
|
+
*/
|
|
115
|
+
editable?: boolean | ((row: TData) => boolean);
|
|
116
|
+
/**
|
|
117
|
+
* Inline disabled state(2026-05-13 Stream C Cluster B Q3 ship,per codex Q3 verdict):
|
|
118
|
+
* cell 不可操作 state(orthogonal to editable readonly)。
|
|
119
|
+
* - `true`:cell 永遠 disabled
|
|
120
|
+
* - `false` / undef(default):cell normal
|
|
121
|
+
* - `(row) => boolean`:per-row 動態(e.g. row.archived 或 row.locked 才 disabled)
|
|
122
|
+
*
|
|
123
|
+
* 視覺(per `color.spec.md` `--bg-disabled` component-state token):
|
|
124
|
+
* - TD 外殼:`bg-disabled` + `cursor-not-allowed` + 抑制 hover ring
|
|
125
|
+
* - Cell content(Field family):`mode='disabled'` → 各 Field type 內部走具體 disabled token
|
|
126
|
+
* - edit entry 抑制:disabled cell 點擊不進 edit(對齊 `editable && !disabled` invariant)
|
|
127
|
+
*
|
|
128
|
+
* 跟 `editable=false` 區分:editable=false = readonly(可看不可編);disabled = 不可操作 state。
|
|
129
|
+
* 對齊 MUI X `isCellEditable` + cellClassName / AG Grid `cellClassRules` / Notion locked properties。
|
|
130
|
+
*/
|
|
131
|
+
disabled?: boolean | ((row: TData) => boolean);
|
|
132
|
+
/**
|
|
133
|
+
* Locked column — column reorder 不可拖,Notion 「primary column」 pattern。
|
|
134
|
+
* 對齊 SKU / ID 等不可移欄位。
|
|
135
|
+
*/
|
|
136
|
+
locked?: boolean;
|
|
137
|
+
/** Person/multiPerson edit mode: people pool for picker(2026-05-05 v4 type-augmentation)。 */
|
|
138
|
+
people?: Array<{
|
|
139
|
+
name: string;
|
|
140
|
+
avatarUrl?: string;
|
|
141
|
+
description?: string;
|
|
142
|
+
}>;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
//# sourceMappingURL=column-types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"column-types.d.ts","sourceRoot":"","sources":["../../../src/components/DataTable/column-types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,uBAAuB,CAAA;AASpD,eAAO,MAAM,WAAW,+HAYd,CAAA;AAEV,MAAM,MAAM,UAAU,GAAG,CAAC,OAAO,WAAW,CAAC,CAAC,MAAM,CAAC,CAAA;AAIrD,MAAM,WAAW,gBAAgB;IAC/B,mCAAmC;IACnC,KAAK,EAAE,MAAM,GAAG,OAAO,GAAG,QAAQ,CAAA;CAInC;AAED,qCAAqC;AACrC,eAAO,MAAM,kBAAkB,EAAE,MAAM,CAAC,UAAU,EAAE,gBAAgB,CAYnE,CAAA;AAID,OAAO,QAAQ,uBAAuB,CAAC;IACrC,UAAU,UAAU,CAAC,KAAK,SAAS,OAAO,EAAE,MAAM;QAChD;;;;;WAKG;QACH,IAAI,CAAC,EAAE,UAAU,CAAA;QACjB,kDAAkD;QAClD,KAAK,CAAC,EAAE,MAAM,GAAG,OAAO,GAAG,QAAQ,CAAA;QACnC,sEAAsE;QACtE,IAAI,CAAC,EAAE,OAAO,CAAA;QACd;;;;;WAKG;QACH,QAAQ,CAAC,EAAE,MAAM,CAAA;QACjB;;;;;;;;;;;;;;;WAeG;QACH,KAAK,CAAC,EAAE,MAAM,CAAA;QACd;;;WAGG;QACH,QAAQ,CAAC,EAAE,MAAM,CAAA;QACjB;;WAEG;QACH,QAAQ,CAAC,EAAE,MAAM,CAAA;QACjB;;;;;;;;;;;WAWG;QACH,SAAS,CAAC,EAAE,OAAO,CAAA;QACnB;;;;;WAKG;QACH,UAAU,CAAC,EAAE,OAAO,CAAA;QACpB,yCAAyC;QACzC,MAAM,CAAC,EAAE,MAAM,CAAA;QACf,MAAM,CAAC,EAAE,MAAM,CAAA;QACf,SAAS,CAAC,EAAE,MAAM,CAAA;QAClB,MAAM,CAAC,EAAE,MAAM,CAAA;QACf,+DAA+D;QAC/D,OAAO,CAAC,EAAE,KAAK,CAAC;YAAE,KAAK,EAAE,MAAM,CAAC;YAAC,KAAK,EAAE,MAAM,CAAA;SAAE,CAAC,CAAA;QACjD,wCAAwC;QACxC,aAAa,CAAC,EAAE,IAAI,CAAC,qBAAqB,CAAA;QAC1C;;;;;;;;;WASG;QACH,WAAW,CAAC,EAAE,OAAO,CAAA;QACrB,2CAA2C;QAC3C,SAAS,CAAC,EAAE,MAAM,CAAA;QAClB;;;;;;;;;;;;;WAaG;QACH,QAAQ,CAAC,EAAE,OAAO,GAAG,CAAC,CAAC,GAAG,EAAE,KAAK,KAAK,OAAO,CAAC,CAAA;QAC9C;;;;;;;;;;;;;;WAcG;QACH,QAAQ,CAAC,EAAE,OAAO,GAAG,CAAC,CAAC,GAAG,EAAE,KAAK,KAAK,OAAO,CAAC,CAAA;QAC9C;;;WAGG;QACH,MAAM,CAAC,EAAE,OAAO,CAAA;QAChB,6FAA6F;QAC7F,MAAM,CAAC,EAAE,KAAK,CAAC;YAAE,IAAI,EAAE,MAAM,CAAC;YAAC,SAAS,CAAC,EAAE,MAAM,CAAC;YAAC,WAAW,CAAC,EAAE,MAAM,CAAA;SAAE,CAAC,CAAA;KAC3E;CACF"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
const columnTypeDefaults = {
|
|
2
|
+
string: { align: "left" },
|
|
3
|
+
number: { align: "right" },
|
|
4
|
+
currency: { align: "right" },
|
|
5
|
+
date: { align: "left" },
|
|
6
|
+
time: { align: "left" },
|
|
7
|
+
select: { align: "left" },
|
|
8
|
+
multiSelect: { align: "left" },
|
|
9
|
+
person: { align: "left" },
|
|
10
|
+
multiPerson: { align: "left" },
|
|
11
|
+
boolean: { align: "left" },
|
|
12
|
+
url: { align: "left" }
|
|
13
|
+
};
|
|
14
|
+
export {
|
|
15
|
+
columnTypeDefaults
|
|
16
|
+
};
|
|
17
|
+
//# sourceMappingURL=column-types.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"column-types.js","sources":["../../../src/components/DataTable/column-types.ts"],"sourcesContent":["import type { RowData } from '@tanstack/react-table'\n\n// ── Column Types ─────────────────────────────────────────────────────────────\n\n// ── Column Types ─────────────────────────────────────────────────────────────\n//\n// 命名原則:描述**資料型別**本身,不是視覺變體。命名要避開撞 Button `variant` 值。\n// `string` / `url` 是世界級 DS(Atlassian / Notion / Ant Table)的資料型別用語,\n// 跟 Button 的視覺變體 `text` / `link` 在 consumer 心智不會混淆。\nexport const columnTypes = [\n 'string', // 前身為 'text',因撞 Button variant=\"text\"(文字樣式按鈕)改名\n 'number',\n 'currency',\n 'date',\n 'time', // Phase C(2026-05-05):time-only column,渲 `<TimePicker>`\n 'select',\n 'multiSelect',\n 'person',\n 'multiPerson',\n 'boolean',\n 'url', // 前身為 'link',因撞 Button variant=\"link\"(連結樣式按鈕)改名\n] as const\n\nexport type ColumnType = (typeof columnTypes)[number]\n\n// ── Column Type Config ───────────────────────────────────────────────────────\n\nexport interface ColumnTypeConfig {\n /** Default horizontal alignment */\n align: 'left' | 'right' | 'center'\n // L3: sortingFn\n // L4: filterVariant, filterFn\n // L5: cellRenderer, cellEditor\n}\n\n/** Default config per column type */\nexport const columnTypeDefaults: Record<ColumnType, ColumnTypeConfig> = {\n string: { align: 'left' },\n number: { align: 'right' },\n currency: { align: 'right' },\n date: { align: 'left' },\n time: { align: 'left' },\n select: { align: 'left' },\n multiSelect: { align: 'left' },\n person: { align: 'left' },\n multiPerson: { align: 'left' },\n boolean: { align: 'left' },\n url: { align: 'left' },\n}\n\n// ── Extend TanStack Table ColumnMeta ─────────────────────────────────────────\n\ndeclare module '@tanstack/react-table' {\n interface ColumnMeta<TData extends RowData, TValue> {\n /**\n * Column data type — determines default alignment, rendering, sorting, filtering.\n *\n * **Filterable column 必須設此 prop**(filter UI 只列有 `type` 的 column,\n * 對齊 Notion / Airtable / Linear:每 property 強制有 type)。\n */\n type?: ColumnType\n /** Override default alignment from column type */\n align?: 'left' | 'right' | 'center'\n /** Allow text wrapping (only effective when autoRowHeight is true) */\n wrap?: boolean\n /**\n * Max visible lines before ellipsis(2026-05-14 I9 per codex verdict + user 拍板)。\n * Only effective when `autoRowHeight + wrap=true`。Display 用 `line-clamp-N`,edit textarea\n * 高度 match clamp。對齊 Notion 「Truncate」row toggle / AG Grid `cellClassRules` 上限 row height。\n * Default = undefined(無 clamp,完整 wrap)。\n */\n maxLines?: number\n /**\n * Column 寬度(px)— DS canonical 命名(2026-05-06 v14.3)。\n *\n * **為何不用 TanStack 的 `size`** — DS 內 `size` 既定意為元件 density string\n * `'sm' | 'md' | 'lg'`(Field / Button / Input 等 49+ 處 use case)。Column 寬\n * 是 px 數字,跟 density 概念衝突 → 走 DS-internal 命名 `width`/`minWidth`/`maxWidth`\n * (對齊 CSS 原生 + AG Grid 的命名共識)。\n *\n * 內部 pre-process 會把 `meta.width` copy 到 TanStack 的 root `size`,確保 column\n * resize feature(`enableColumnResize=true` 拖拉 + columnSizing state)正常運作。\n *\n * - **No resize mode**(default): `width` = column 寬度 reserve(cell ≥ width,\n * flex 可 grow,**不可 shrink** 低於 width)。對齊 user 預期「我設多寬就多寬」。\n * - **Resize mode**(`enableColumnResize=true`):`width` = 初始寬度,user 可拖拉\n * 到 `minWidth` 為止。\n */\n width?: number\n /**\n * Column 最小寬度(px)— resize 模式下的拖拉下限。No resize 模式不使用(`width` 即下限)。\n * Default(resize mode 沒明示時)= 80px(`MIN_COLUMN_WIDTH`)。\n */\n minWidth?: number\n /**\n * Column 最大寬度(px)— resize 模式下的拖拉上限。預設無上限。\n */\n maxWidth?: number\n /**\n * Column 是否可被 user 拖拉 resize(opt-out per col)。Default `true`(when `enableColumnResize` 也 true)。\n *\n * Set `false` 表「此 col 寬度由內容決定不允許 resize」(2026-05-10 加,per user\n * 「操作列這種內容決定寬度的欄位應該不允許 resize」)。對齊 AG Grid `colDef.resizable` /\n * Material X-DataGrid 同 API。\n *\n * System cols(__select__ / __drag__ / __actions__)自動 false(內部 hard-code,\n * consumer 不需設)— 永遠固定寬。\n *\n * 典型 use case:custom 圖示欄、status indicator 欄、密度太緊不該 user 動的欄。\n */\n resizable?: boolean\n /**\n * Explicit opt-out from filter UI(預設 accessor column 有 type 即 filterable)。\n *\n * 用於:有 `type` 但不想在 filter UI 出現的 accessor column\n * (例如:internal sorting key、composite display 用的 hidden column)。\n */\n filterable?: boolean\n /** Number/currency formatting options */\n prefix?: string\n suffix?: string\n precision?: number\n locale?: string\n /** Select/multiSelect options — maps value to display label */\n options?: Array<{ value: string; label: string }>\n /** Date: Intl.DateTimeFormat options */\n formatOptions?: Intl.DateTimeFormatOptions\n /**\n * Date: 是否含時間部分(datetime mode)。對齊 Notion idiom — 不另設 datetime column type。\n *\n * - `false`(default):cell 顯示與 filter 比對僅日期(day-level 精度)\n * - `true`:cell 渲 date+time;filter 比對走 ms 精度(避開 Airtable 著名地雷)\n *\n * 在 advanced filter 中,`date` columnType 配 `includeTime=true` 時,\n * `date_*` ValueShape 自動 promote 到 `datetime_*`,渲 `<DatePicker showTime>` /\n * `<DatePickerRange showTime>`(詳 `filter-operators.ts` `getValueShape`)。\n */\n includeTime?: boolean\n /** Link: 自訂顯示文字(不設則自動從 URL 提取 hostname) */\n linkLabel?: string\n /**\n * Inline edit:column 是否可編輯。\n * - `true`:可編\n * - `false`(default):唯讀\n * - `(row) => boolean`:per-row 動態決定(e.g. row.status !== 'archived' 才能編)\n *\n * 互動 per type(對齊 Notion / Airtable):\n * - string / number / currency:click cell → inline `<Input>` autoFocus + selected\n * - date / select / multiSelect / person / multiPerson:click cell → 進 edit mode 的 Field control(用戶按 trigger 開 picker)\n * - boolean:不分 read/edit mode,直接 `<Checkbox>` 點即 toggle + commit\n * - url:read = 連結;**hover cell** 右側出 Pencil 按鈕,click 才進 `<Input>` edit mode(保留 click 連結語意)\n *\n * Esc cancel / blur or Enter commit。Commit 觸發 `onCellCommit`。\n */\n editable?: boolean | ((row: TData) => boolean)\n /**\n * Inline disabled state(2026-05-13 Stream C Cluster B Q3 ship,per codex Q3 verdict):\n * cell 不可操作 state(orthogonal to editable readonly)。\n * - `true`:cell 永遠 disabled\n * - `false` / undef(default):cell normal\n * - `(row) => boolean`:per-row 動態(e.g. row.archived 或 row.locked 才 disabled)\n *\n * 視覺(per `color.spec.md` `--bg-disabled` component-state token):\n * - TD 外殼:`bg-disabled` + `cursor-not-allowed` + 抑制 hover ring\n * - Cell content(Field family):`mode='disabled'` → 各 Field type 內部走具體 disabled token\n * - edit entry 抑制:disabled cell 點擊不進 edit(對齊 `editable && !disabled` invariant)\n *\n * 跟 `editable=false` 區分:editable=false = readonly(可看不可編);disabled = 不可操作 state。\n * 對齊 MUI X `isCellEditable` + cellClassName / AG Grid `cellClassRules` / Notion locked properties。\n */\n disabled?: boolean | ((row: TData) => boolean)\n /**\n * Locked column — column reorder 不可拖,Notion 「primary column」 pattern。\n * 對齊 SKU / ID 等不可移欄位。\n */\n locked?: boolean\n /** Person/multiPerson edit mode: people pool for picker(2026-05-05 v4 type-augmentation)。 */\n people?: Array<{ name: string; avatarUrl?: string; description?: string }>\n }\n}\n"],"names":[],"mappings":"AAoCO,MAAM,qBAA2D;AAAA,EACtE,QAAa,EAAE,OAAO,OAAA;AAAA,EACtB,QAAa,EAAE,OAAO,QAAA;AAAA,EACtB,UAAa,EAAE,OAAO,QAAA;AAAA,EACtB,MAAa,EAAE,OAAO,OAAA;AAAA,EACtB,MAAa,EAAE,OAAO,OAAA;AAAA,EACtB,QAAa,EAAE,OAAO,OAAA;AAAA,EACtB,aAAa,EAAE,OAAO,OAAA;AAAA,EACtB,QAAa,EAAE,OAAO,OAAA;AAAA,EACtB,aAAa,EAAE,OAAO,OAAA;AAAA,EACtB,SAAa,EAAE,OAAO,OAAA;AAAA,EACtB,KAAa,EAAE,OAAO,OAAA;AACxB;"}
|