@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,44 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
export interface FileItemProps extends Omit<React.HTMLAttributes<HTMLDivElement>, 'title'> {
|
|
3
|
+
name: string;
|
|
4
|
+
/**
|
|
5
|
+
* 兩種呈現 mode(精簡 vs 完整):
|
|
6
|
+
* - `compact`(預設):paperclip + filename 單行 inline
|
|
7
|
+
* - `rich`:縮圖 + 檔名 + size + status + progress 的完整 card 呈現
|
|
8
|
+
*/
|
|
9
|
+
mode?: 'compact' | 'rich';
|
|
10
|
+
status?: 'uploading' | 'completed' | 'error';
|
|
11
|
+
progress?: number;
|
|
12
|
+
/** rich mode: 檔案大小、狀態訊息。compact: 只有 error 才顯示。 ReactNode 支援 inline clickable link(如「View log」)。 */
|
|
13
|
+
description?: React.ReactNode;
|
|
14
|
+
thumbnailSrc?: string;
|
|
15
|
+
actions?: React.ReactNode;
|
|
16
|
+
onClick?: () => void;
|
|
17
|
+
/**
|
|
18
|
+
* Hover 動作(passive 狀態 icon 變互動 button 的 UX):
|
|
19
|
+
* - `onDownload`:有值時,`status="completed"` 的綠 ✓ icon 在 row hover 時
|
|
20
|
+
* 換成 Download ↓ button;click 觸發 onDownload。無值保持 passive 綠 ✓。
|
|
21
|
+
* - `onRetry`:有值時,`status="error"` 的紅 ✗ icon 在 row hover 時換成 RotateCw ⟲
|
|
22
|
+
* button;click 觸發 onRetry。無值保持 passive 紅 ✗。
|
|
23
|
+
*
|
|
24
|
+
* 世界級對照:Gmail / Slack / Dropbox 附件的 passive 狀態 → hover 變 action
|
|
25
|
+
* 的 UX,使用者知道檔案狀態且能立即行動。
|
|
26
|
+
*/
|
|
27
|
+
onDownload?: () => void;
|
|
28
|
+
onRetry?: () => void;
|
|
29
|
+
}
|
|
30
|
+
declare const FileItem: React.ForwardRefExoticComponent<FileItemProps & React.RefAttributes<HTMLDivElement>>;
|
|
31
|
+
export declare const fileItemMeta: {
|
|
32
|
+
readonly component: "FileItem";
|
|
33
|
+
readonly family: 2;
|
|
34
|
+
readonly variants: {};
|
|
35
|
+
readonly sizes: {};
|
|
36
|
+
readonly states: readonly ["default", "hover", "active", "focus-visible", "disabled"];
|
|
37
|
+
readonly tokens: {
|
|
38
|
+
readonly bg: readonly ["bg-neutral-hover", "bg-secondary", "bg-surface"];
|
|
39
|
+
readonly fg: readonly ["text-fg-muted", "text-fg-secondary"];
|
|
40
|
+
readonly ring: readonly [];
|
|
41
|
+
};
|
|
42
|
+
};
|
|
43
|
+
export { FileItem };
|
|
44
|
+
//# sourceMappingURL=file-item.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"file-item.d.ts","sourceRoot":"","sources":["../../../src/components/FileItem/file-item.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAA;AA6C9B,MAAM,WAAW,aAAc,SAAQ,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,EAAE,OAAO,CAAC;IACxF,IAAI,EAAE,MAAM,CAAA;IACZ;;;;OAIG;IACH,IAAI,CAAC,EAAE,SAAS,GAAG,MAAM,CAAA;IACzB,MAAM,CAAC,EAAE,WAAW,GAAG,WAAW,GAAG,OAAO,CAAA;IAC5C,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,mGAAmG;IACnG,WAAW,CAAC,EAAE,KAAK,CAAC,SAAS,CAAA;IAC7B,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,OAAO,CAAC,EAAE,KAAK,CAAC,SAAS,CAAA;IACzB,OAAO,CAAC,EAAE,MAAM,IAAI,CAAA;IACpB;;;;;;;;;OASG;IACH,UAAU,CAAC,EAAE,MAAM,IAAI,CAAA;IACvB,OAAO,CAAC,EAAE,MAAM,IAAI,CAAA;CACrB;AAGD,QAAA,MAAM,QAAQ,sFA+Nb,CAAA;AAKD,eAAO,MAAM,YAAY;;;;;;;;;;;CAef,CAAA;AAEV,OAAO,EAAE,QAAQ,EAAE,CAAA"}
|
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
import { jsxs, jsx } from "react/jsx-runtime";
|
|
2
|
+
import * as React from "react";
|
|
3
|
+
import { Download, RotateCw, XCircle, CircleCheck, Paperclip } from "lucide-react";
|
|
4
|
+
import { cn } from "../../lib/utils.js";
|
|
5
|
+
import { Avatar } from "../Avatar/avatar.js";
|
|
6
|
+
import { Button } from "../Button/button.js";
|
|
7
|
+
import { ProgressBar } from "../ProgressBar/progress-bar.js";
|
|
8
|
+
import { ItemPrefix, ItemContent } from "../../patterns/element-anatomy/item-anatomy.js";
|
|
9
|
+
const STATUS_ICON = {
|
|
10
|
+
completed: { icon: CircleCheck, color: "text-success" },
|
|
11
|
+
error: { icon: XCircle, color: "text-error" }
|
|
12
|
+
};
|
|
13
|
+
const PROGRESS_STATUS_MAP = {
|
|
14
|
+
uploading: "inProgress",
|
|
15
|
+
completed: "success",
|
|
16
|
+
error: "error"
|
|
17
|
+
};
|
|
18
|
+
const AVATAR_SIZE = 48;
|
|
19
|
+
const ICON_PX = 16;
|
|
20
|
+
const FileItem = React.forwardRef(
|
|
21
|
+
({
|
|
22
|
+
name,
|
|
23
|
+
mode = "compact",
|
|
24
|
+
status,
|
|
25
|
+
progress = 0,
|
|
26
|
+
description,
|
|
27
|
+
thumbnailSrc,
|
|
28
|
+
actions,
|
|
29
|
+
onClick,
|
|
30
|
+
onDownload,
|
|
31
|
+
onRetry,
|
|
32
|
+
className,
|
|
33
|
+
...props
|
|
34
|
+
}, ref) => {
|
|
35
|
+
const isRich = mode === "rich";
|
|
36
|
+
const hasStatus = !!status;
|
|
37
|
+
const statusConfig = status && status !== "uploading" ? STATUS_ICON[status] : null;
|
|
38
|
+
const progressWidth = status === "completed" ? 100 : progress;
|
|
39
|
+
const showDesc = isRich ? !!description : status === "error" && !!description;
|
|
40
|
+
const hoverClass = onClick ? "cursor-pointer" : "";
|
|
41
|
+
const progressBar = hasStatus ? /* @__PURE__ */ jsx(
|
|
42
|
+
ProgressBar,
|
|
43
|
+
{
|
|
44
|
+
value: progressWidth,
|
|
45
|
+
status: PROGRESS_STATUS_MAP[status],
|
|
46
|
+
height: isRich ? void 0 : 2,
|
|
47
|
+
"aria-label": `${name} 上傳進度`
|
|
48
|
+
}
|
|
49
|
+
) : null;
|
|
50
|
+
const suffixAlign = "h-[1lh]";
|
|
51
|
+
const slotHw = "var(--field-height-xs)";
|
|
52
|
+
const hoverAction = status === "completed" && onDownload ? { icon: Download, onClick: onDownload, label: "下載" } : status === "error" && onRetry ? { icon: RotateCw, onClick: onRetry, label: "重試" } : null;
|
|
53
|
+
const statusSlot = statusConfig ? /* @__PURE__ */ jsxs(
|
|
54
|
+
"span",
|
|
55
|
+
{
|
|
56
|
+
"data-unbounded": "true",
|
|
57
|
+
className: "relative inline-flex items-center justify-center shrink-0",
|
|
58
|
+
style: { width: slotHw, height: slotHw },
|
|
59
|
+
children: [
|
|
60
|
+
/* @__PURE__ */ jsx(
|
|
61
|
+
statusConfig.icon,
|
|
62
|
+
{
|
|
63
|
+
size: ICON_PX,
|
|
64
|
+
className: cn(
|
|
65
|
+
"shrink-0 transition-opacity",
|
|
66
|
+
statusConfig.color,
|
|
67
|
+
hoverAction && "group-hover/row:opacity-0"
|
|
68
|
+
),
|
|
69
|
+
"aria-hidden": true
|
|
70
|
+
}
|
|
71
|
+
),
|
|
72
|
+
hoverAction && /* @__PURE__ */ jsx(
|
|
73
|
+
Button,
|
|
74
|
+
{
|
|
75
|
+
variant: "text",
|
|
76
|
+
size: "xs",
|
|
77
|
+
iconOnly: true,
|
|
78
|
+
startIcon: hoverAction.icon,
|
|
79
|
+
"aria-label": hoverAction.label,
|
|
80
|
+
onClick: (e) => {
|
|
81
|
+
e.stopPropagation();
|
|
82
|
+
hoverAction.onClick();
|
|
83
|
+
},
|
|
84
|
+
className: "absolute inset-0 opacity-0 group-hover/row:opacity-100 transition-opacity"
|
|
85
|
+
}
|
|
86
|
+
)
|
|
87
|
+
]
|
|
88
|
+
}
|
|
89
|
+
) : null;
|
|
90
|
+
const suffix = /* @__PURE__ */ jsxs(
|
|
91
|
+
"div",
|
|
92
|
+
{
|
|
93
|
+
className: cn(
|
|
94
|
+
"flex items-center gap-2 shrink-0",
|
|
95
|
+
suffixAlign,
|
|
96
|
+
// data-unbounded chrome-canonical trick:let Button xs (24) live inside h-[1lh]
|
|
97
|
+
// wrapper(compact ~18.2 / rich ~18.2 scanning)without pushing row height。
|
|
98
|
+
// 視覺/hit area 仍 24,layout footprint 收斂到 1lh。同 overlay-surface
|
|
99
|
+
// 的 SurfaceHeader dismiss canonical(2026-04-22 v5)。
|
|
100
|
+
// **child selector `[&>[data-unbounded]]`(非 descendant)**:只針對 suffix
|
|
101
|
+
// wrapper **直接子元素**(statusSlot span、actions Button)套 margin,
|
|
102
|
+
// 避免 status slot 內部 hover-swap Button(nested)也套造成 layout 跳動。
|
|
103
|
+
"[&>[data-unbounded]]:my-[calc((1lh-var(--field-height-xs))/2)]"
|
|
104
|
+
),
|
|
105
|
+
children: [
|
|
106
|
+
status === "uploading" && isRich && /* @__PURE__ */ jsxs("span", { className: "text-fg-secondary tabular-nums", children: [
|
|
107
|
+
progress,
|
|
108
|
+
"%"
|
|
109
|
+
] }),
|
|
110
|
+
statusSlot,
|
|
111
|
+
actions
|
|
112
|
+
]
|
|
113
|
+
}
|
|
114
|
+
);
|
|
115
|
+
const contentRow = /* @__PURE__ */ jsxs("div", { className: "flex items-start gap-2", children: [
|
|
116
|
+
/* @__PURE__ */ jsx(
|
|
117
|
+
ItemContent,
|
|
118
|
+
{
|
|
119
|
+
label: name,
|
|
120
|
+
description: showDesc ? description : void 0,
|
|
121
|
+
mode: "scanning",
|
|
122
|
+
descriptionTone: status === "error" ? "error" : "secondary"
|
|
123
|
+
}
|
|
124
|
+
),
|
|
125
|
+
suffix
|
|
126
|
+
] });
|
|
127
|
+
const rowA11y = {};
|
|
128
|
+
const compactStaticBg = !progressBar ? "bg-secondary" : "";
|
|
129
|
+
if (isRich) {
|
|
130
|
+
return /* @__PURE__ */ jsxs(
|
|
131
|
+
"div",
|
|
132
|
+
{
|
|
133
|
+
ref,
|
|
134
|
+
className: cn(
|
|
135
|
+
"group/row flex items-start gap-2 px-3 py-3 w-full text-body leading-compact transition-colors",
|
|
136
|
+
"border border-divider rounded-md bg-surface",
|
|
137
|
+
hoverClass,
|
|
138
|
+
className
|
|
139
|
+
),
|
|
140
|
+
onClick,
|
|
141
|
+
...rowA11y,
|
|
142
|
+
...props,
|
|
143
|
+
children: [
|
|
144
|
+
/* @__PURE__ */ jsx(Avatar, { src: thumbnailSrc, alt: name, size: AVATAR_SIZE, shape: "square", className: "shrink-0" }),
|
|
145
|
+
/* @__PURE__ */ jsxs(
|
|
146
|
+
"div",
|
|
147
|
+
{
|
|
148
|
+
className: cn(
|
|
149
|
+
"flex flex-col flex-1 min-w-0 gap-2",
|
|
150
|
+
progressBar ? "justify-between" : "justify-center"
|
|
151
|
+
),
|
|
152
|
+
style: { minHeight: AVATAR_SIZE },
|
|
153
|
+
children: [
|
|
154
|
+
contentRow,
|
|
155
|
+
progressBar
|
|
156
|
+
]
|
|
157
|
+
}
|
|
158
|
+
)
|
|
159
|
+
]
|
|
160
|
+
}
|
|
161
|
+
);
|
|
162
|
+
}
|
|
163
|
+
return /* @__PURE__ */ jsxs(
|
|
164
|
+
"div",
|
|
165
|
+
{
|
|
166
|
+
ref,
|
|
167
|
+
className: cn(
|
|
168
|
+
"group/row relative flex items-start gap-2 px-3 py-2 w-full text-body leading-compact transition-colors rounded-md",
|
|
169
|
+
compactStaticBg,
|
|
170
|
+
hoverClass,
|
|
171
|
+
className
|
|
172
|
+
),
|
|
173
|
+
onClick,
|
|
174
|
+
...rowA11y,
|
|
175
|
+
...props,
|
|
176
|
+
children: [
|
|
177
|
+
/* @__PURE__ */ jsx(ItemPrefix, { children: /* @__PURE__ */ jsx(Paperclip, { size: ICON_PX, className: "shrink-0 text-fg-muted", "aria-hidden": true }) }),
|
|
178
|
+
/* @__PURE__ */ jsx("div", { className: "flex flex-col flex-1 min-w-0", children: contentRow }),
|
|
179
|
+
progressBar && /* @__PURE__ */ jsx("div", { className: "absolute bottom-0 right-3", style: { left: `calc(0.75rem + ${ICON_PX}px + 0.5rem)` }, children: progressBar })
|
|
180
|
+
]
|
|
181
|
+
}
|
|
182
|
+
);
|
|
183
|
+
}
|
|
184
|
+
);
|
|
185
|
+
FileItem.displayName = "FileItem";
|
|
186
|
+
const fileItemMeta = {
|
|
187
|
+
component: "FileItem",
|
|
188
|
+
family: 2,
|
|
189
|
+
variants: {},
|
|
190
|
+
sizes: {},
|
|
191
|
+
states: ["default", "hover", "active", "focus-visible", "disabled"],
|
|
192
|
+
tokens: {
|
|
193
|
+
bg: ["bg-neutral-hover", "bg-secondary", "bg-surface"],
|
|
194
|
+
fg: ["text-fg-muted", "text-fg-secondary"],
|
|
195
|
+
ring: []
|
|
196
|
+
}
|
|
197
|
+
};
|
|
198
|
+
export {
|
|
199
|
+
FileItem,
|
|
200
|
+
fileItemMeta
|
|
201
|
+
};
|
|
202
|
+
//# sourceMappingURL=file-item.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"file-item.js","sources":["../../../src/components/FileItem/file-item.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.\nimport * as React from 'react'\nimport { Paperclip, CircleCheck, XCircle, Download, RotateCw } from 'lucide-react'\nimport { cn } from '@/lib/utils'\nimport { Avatar } from '@/design-system/components/Avatar/avatar'\nimport { Button } from '@/design-system/components/Button/button'\nimport { ProgressBar } from '@/design-system/components/ProgressBar/progress-bar'\nimport { ItemContent, ItemPrefix } from '@/design-system/patterns/element-anatomy/item-anatomy'\n\n/**\n * FileItem — 檔案顯示 / 上傳進度\n *\n * Typography: 閱讀模式 — text-body (14px) 預設行高 (1.5)\n *\n * 兩種 mode(精簡 vs 完整內容呈現):\n *\n * compact(★ default): Paperclip 16px 在左。右側 content + bar。\n * py = gap = 4px (gap-1),對稱。\n * description 只有 error 才顯示。\n * bar 跟文字左邊對齊(在 icon 右邊的 column 內)。\n *\n * rich: Avatar 48px square 在左(顯示檔案內容縮圖)。右側 content + bar。\n * 多行 description(size / status message)。\n * 有 bar → justify-between(bar 底部對齊 avatar)\n * 無 bar → justify-center(文字垂直置中對齊 avatar)\n *\n * status 可選。不傳 = 已上傳檔案(無 bar,可點擊下載)。\n * onClick → hover:bg-neutral-hover + cursor-pointer。\n */\n\nconst STATUS_ICON = {\n completed: { icon: CircleCheck, color: 'text-success' },\n error: { icon: XCircle, color: 'text-error' },\n} as const\n\n// ProgressBar status 映射:uploading=inProgress(藍) / completed=success(綠) / error=error(紅)\n// 與 ProgressBar 元件的 status prop 對齊,不需再維護 PROGRESS_COLOR 本地 map。\nconst PROGRESS_STATUS_MAP = {\n uploading: 'inProgress',\n completed: 'success',\n error: 'error',\n} as const\n\nconst AVATAR_SIZE = 48\nconst ICON_PX = 16\n\nexport interface FileItemProps extends Omit<React.HTMLAttributes<HTMLDivElement>, 'title'> {\n name: string\n /**\n * 兩種呈現 mode(精簡 vs 完整):\n * - `compact`(預設):paperclip + filename 單行 inline\n * - `rich`:縮圖 + 檔名 + size + status + progress 的完整 card 呈現\n */\n mode?: 'compact' | 'rich'\n status?: 'uploading' | 'completed' | 'error'\n progress?: number\n /** rich mode: 檔案大小、狀態訊息。compact: 只有 error 才顯示。 ReactNode 支援 inline clickable link(如「View log」)。 */\n description?: React.ReactNode\n thumbnailSrc?: string\n actions?: React.ReactNode\n onClick?: () => void\n /**\n * Hover 動作(passive 狀態 icon 變互動 button 的 UX):\n * - `onDownload`:有值時,`status=\"completed\"` 的綠 ✓ icon 在 row hover 時\n * 換成 Download ↓ button;click 觸發 onDownload。無值保持 passive 綠 ✓。\n * - `onRetry`:有值時,`status=\"error\"` 的紅 ✗ icon 在 row hover 時換成 RotateCw ⟲\n * button;click 觸發 onRetry。無值保持 passive 紅 ✗。\n *\n * 世界級對照:Gmail / Slack / Dropbox 附件的 passive 狀態 → hover 變 action\n * 的 UX,使用者知道檔案狀態且能立即行動。\n */\n onDownload?: () => void\n onRetry?: () => void\n}\n\n// code-quality-allow: long-function — foundational composite main body — 拆 sub-fn 會複雜化 local state / ref / context binding\nconst FileItem = React.forwardRef<HTMLDivElement, FileItemProps>(\n (\n {\n name,\n mode = 'compact',\n status,\n progress = 0,\n description,\n thumbnailSrc,\n actions,\n onClick,\n onDownload,\n onRetry,\n className,\n ...props\n },\n ref,\n ) => {\n const isRich = mode === 'rich'\n const hasStatus = !!status\n const statusConfig = status && status !== 'uploading' ? STATUS_ICON[status] : null\n const progressWidth = status === 'completed' ? 100 : progress\n\n // compact 只有 error 才顯示 description\n const showDesc = isRich ? !!description : (status === 'error' && !!description)\n\n // Hover 行為 canonical(2026-04-23 user 校準):**FileItem 永不顯示 hover-bg**。\n // 三種型態都有永久 visual anchor:rich = border card / compact Type B = bg-secondary /\n // compact Type A = 底部 progress bar(分隔線型 affordance)——再加 hover-bg 是\n // double-emphasis,視覺雜。世界級共識(Slack / Notion / Figma / Gmail 皆無 hover-bg):\n // permanent-anchored 元件 hover 只靠 cursor + action icon fade / border highlight,\n // 不靠 row bg。onClick 存在時只給 `cursor-pointer`,affordance 靠 cursor + 點擊行為本身。\n const hoverClass = onClick ? 'cursor-pointer' : ''\n\n // 消費 ProgressBar 元件(SSOT);不再自 roll bar。\n // height override:compact mode 用 2px(極密集 row layout),rich mode 用預設 4px。\n // 這是 ProgressBar `height` prop 的唯一合法 consumer(見 progress-bar.tsx docblock)。\n // a11y(2026-04-25 axe aria-progressbar-name):aria-label 用 file name 作 context。\n const progressBar = hasStatus ? (\n <ProgressBar\n value={progressWidth}\n status={PROGRESS_STATUS_MAP[status!]}\n height={isRich ? undefined : 2}\n aria-label={`${name} 上傳進度`}\n />\n ) : null\n\n // suffix 對齊 label 第一行(item-anatomy「24px 閾值對齊規則」小 suffix canonical):\n // icons 16 ≤ 24 屬小 suffix,統一 `h-[1lh]` inline,不因 desc wrap 改公式。\n // 兩 mode 同公式,跟 item-anatomy 一致。\n const suffixAlign = 'h-[1lh]'\n\n // Status slot 幾何(2026-04-23 user 統一):rich + compact 都用 `var(--field-height-xs)`(24)\n // 容器,裡面 Button xs iconOnly variant=\"text\"(auto data-unbounded)。\n // Compact 不影響 row 高度 = suffix wrapper 的 data-unbounded CSS 讓 Button layout\n // 收斂到 1lh(同 compact row 內容高),視覺/touch target 仍 24。\n const slotHw = 'var(--field-height-xs)'\n\n const hoverAction =\n status === 'completed' && onDownload ? { icon: Download, onClick: onDownload, label: '下載' } :\n status === 'error' && onRetry ? { icon: RotateCw, onClick: onRetry, label: '重試' } :\n null\n\n const statusSlot = statusConfig ? (\n <span\n data-unbounded=\"true\"\n className=\"relative inline-flex items-center justify-center shrink-0\"\n style={{ width: slotHw, height: slotHw }}\n >\n {/* Passive 狀態 icon:預設可見;若有 hover-swap,row-hover 時淡出 */}\n <statusConfig.icon\n size={ICON_PX}\n className={cn(\n 'shrink-0 transition-opacity',\n statusConfig.color,\n hoverAction && 'group-hover/row:opacity-0',\n )}\n aria-hidden\n />\n {/* Active action:row-hover 時淡入(rich + compact 同 Button xs 統一) */}\n {hoverAction && (\n <Button\n variant=\"text\"\n size=\"xs\"\n iconOnly\n startIcon={hoverAction.icon}\n aria-label={hoverAction.label}\n onClick={(e) => { e.stopPropagation(); hoverAction.onClick() }}\n className=\"absolute inset-0 opacity-0 group-hover/row:opacity-100 transition-opacity\"\n />\n )}\n </span>\n ) : null\n\n const suffix = (\n <div\n className={cn(\n 'flex items-center gap-2 shrink-0',\n suffixAlign,\n // data-unbounded chrome-canonical trick:let Button xs (24) live inside h-[1lh]\n // wrapper(compact ~18.2 / rich ~18.2 scanning)without pushing row height。\n // 視覺/hit area 仍 24,layout footprint 收斂到 1lh。同 overlay-surface\n // 的 SurfaceHeader dismiss canonical(2026-04-22 v5)。\n // **child selector `[&>[data-unbounded]]`(非 descendant)**:只針對 suffix\n // wrapper **直接子元素**(statusSlot span、actions Button)套 margin,\n // 避免 status slot 內部 hover-swap Button(nested)也套造成 layout 跳動。\n '[&>[data-unbounded]]:my-[calc((1lh-var(--field-height-xs))/2)]',\n )}\n >\n {status === 'uploading' && isRich && (\n <span className=\"text-fg-secondary tabular-nums\">{progress}%</span>\n )}\n {statusSlot}\n {actions}\n </div>\n )\n\n // content row — 消費 ItemContent primitive(封裝 label + desc + mt-gap token SSOT)。\n // 兩 mode 共用:primitive 改 → 兩 mode 同步,不需 grep。\n // typography:scanning mode(2026-04-23 user 指示)—— label body(14/1.3) + desc caption(12/1.3);\n // row 本身加 `leading-compact` 配合 scanning idiom(同 MenuItem row)。\n const contentRow = (\n <div className=\"flex items-start gap-2\">\n <ItemContent\n label={name}\n description={showDesc ? description : undefined}\n mode=\"scanning\"\n descriptionTone={status === 'error' ? 'error' : 'secondary'}\n />\n {suffix}\n </div>\n )\n\n // a11y(2026-04-25 nested-interactive fix):FileItem row 含 inner interactive\n // (hover-swap action button / ProgressBar / Avatar hoverCard trigger)。原本\n // role='button' + tabIndex=0 整列可鍵盤點,與 inner buttons 構成 nested-interactive\n // (axe serious)。移除 row 層 button semantic → mouse 仍可點(onClick 保留),\n // 鍵盤 user 直接 tab 到 inner primary action。Trade-off:失去「整列 Enter 開啟」\n // 但滿足 WCAG;世界級對照:Slack message row / Notion page row 同模式 — row 只\n // mouse 點,inner 有 explicit 按鈕負責鍵盤。\n const rowA11y = {}\n\n // Compact 靜態背景(AR20):無進度條 → 顯示 `bg-secondary`(= neutral-3)作「檔案已上傳 /\n // 靜態列表」視覺區隔,跟「上傳中(有 progress bar)」對照。hover 不改 bg(見上方\n // hoverClass canonical:FileItem 永不顯示 hover-bg)。\n // **為什麼 bg-secondary 不 bg-neutral-3**:`bg-neutral-3` 不是合法 Tailwind utility\n // (primitive token `--color-neutral-3` 沒經 `@theme inline` 橋接);`bg-secondary`\n // 是 semantic token 橋接的 utility(見 `tokens/color/semantic.css`@theme inline),\n // 底色同樣指向 `--color-neutral-3`。對齊 Badge low / ProgressBar track SSOT。\n const compactStaticBg = !progressBar ? 'bg-secondary' : ''\n\n // ── rich(含縮圖完整呈現)——AR17 canonical:加邊框 + gap-2 ──\n // Rich mode 是「檔案 card」風格,外框讓每個 row 視覺上是獨立 card\n // (Slack / Notion / Linear attachment 慣例)\n if (isRich) {\n return (\n <div\n ref={ref}\n className={cn(\n 'group/row flex items-start gap-2 px-3 py-3 w-full text-body leading-compact transition-colors',\n 'border border-divider rounded-md bg-surface',\n hoverClass,\n className,\n )}\n onClick={onClick}\n {...rowA11y}\n {...props}\n >\n <Avatar src={thumbnailSrc} alt={name} size={AVATAR_SIZE} shape=\"square\" className=\"shrink-0\" />\n {/* Rich layout invariant(2026-04-23 user 校準):\n - content col minHeight = AVATAR_SIZE(48),確保 1-line desc 時內容 ≥ avatar 高\n - `justify-between`(有 bar)/`justify-center`(無 bar):\n * 1-line desc:label 頂 + progress bar 底 **自動對齊 avatar 頂/底**\n * 無 bar:content 垂直 center 對齊 avatar 中\n - `gap-2`:desc ↔ progress bar **至少 8px gap**(multi-line desc 時 bar 溢出仍保 8px)\n - row `items-start`:avatar top-align 作視覺引導(tight-stack box 內 item 邊界) */}\n <div\n className={cn(\n 'flex flex-col flex-1 min-w-0 gap-2',\n progressBar ? 'justify-between' : 'justify-center',\n )}\n style={{ minHeight: AVATAR_SIZE }}\n >\n {contentRow}\n {progressBar}\n </div>\n </div>\n )\n }\n\n // ── compact: py-2 對稱, bar absolute 底部 ──\n return (\n <div\n ref={ref}\n className={cn(\n 'group/row relative flex items-start gap-2 px-3 py-2 w-full text-body leading-compact transition-colors rounded-md',\n compactStaticBg,\n hoverClass,\n className,\n )}\n onClick={onClick}\n {...rowA11y}\n {...props}\n >\n <ItemPrefix>\n <Paperclip size={ICON_PX} className=\"shrink-0 text-fg-muted\" aria-hidden />\n </ItemPrefix>\n {/* Compact 共用 contentRow(via ItemContent primitive SSOT)—— 先前 inline\n hand-craft 導致 compact label↔desc gap 跟 rich 不同步。shared contentRow\n 保證兩 mode 修 primitive 一處全同步。 */}\n <div className=\"flex flex-col flex-1 min-w-0\">\n {contentRow}\n </div>\n\n {/* ProgressBar: absolute 底部, left 對齊 label(跳過 icon + gap) */}\n {progressBar && (\n <div className=\"absolute bottom-0 right-3\" style={{ left: `calc(0.75rem + ${ICON_PX}px + 0.5rem)` }}>\n {progressBar}\n </div>\n )}\n </div>\n )\n },\n)\nFileItem.displayName = 'FileItem'\n\n// Story auto-compile metadata — Phase 1 mechanical migration(2026-04-24)\n// Phase 2 fill needed: purpose descriptions + when rationale + world-class refs\nexport const fileItemMeta = {\n component: 'FileItem',\n family: 2,\n variants: {\n\n },\n sizes: {\n\n },\n states: ['default', 'hover', 'active', 'focus-visible', 'disabled'],\n tokens: {\n bg: ['bg-neutral-hover', 'bg-secondary', 'bg-surface'],\n fg: ['text-fg-muted', 'text-fg-secondary'],\n ring: [],\n },\n} as const\n\nexport { FileItem }\n"],"names":[],"mappings":";;;;;;;;AA8BA,MAAM,cAAc;AAAA,EAClB,WAAW,EAAE,MAAM,aAAa,OAAO,eAAA;AAAA,EACvC,OAAO,EAAE,MAAM,SAAS,OAAO,aAAA;AACjC;AAIA,MAAM,sBAAsB;AAAA,EAC1B,WAAW;AAAA,EACX,WAAW;AAAA,EACX,OAAO;AACT;AAEA,MAAM,cAAc;AACpB,MAAM,UAAU;AAgChB,MAAM,WAAW,MAAM;AAAA,EACrB,CACE;AAAA,IACE;AAAA,IACA,OAAO;AAAA,IACP;AAAA,IACA,WAAW;AAAA,IACX;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,GAAG;AAAA,EAAA,GAEL,QACG;AACH,UAAM,SAAS,SAAS;AACxB,UAAM,YAAY,CAAC,CAAC;AACpB,UAAM,eAAe,UAAU,WAAW,cAAc,YAAY,MAAM,IAAI;AAC9E,UAAM,gBAAgB,WAAW,cAAc,MAAM;AAGrD,UAAM,WAAW,SAAS,CAAC,CAAC,cAAe,WAAW,WAAW,CAAC,CAAC;AAQnE,UAAM,aAAa,UAAU,mBAAmB;AAMhD,UAAM,cAAc,YAClB;AAAA,MAAC;AAAA,MAAA;AAAA,QACC,OAAO;AAAA,QACP,QAAQ,oBAAoB,MAAO;AAAA,QACnC,QAAQ,SAAS,SAAY;AAAA,QAC7B,cAAY,GAAG,IAAI;AAAA,MAAA;AAAA,IAAA,IAEnB;AAKJ,UAAM,cAAc;AAMpB,UAAM,SAAS;AAEf,UAAM,cACJ,WAAW,eAAe,aAAa,EAAE,MAAM,UAAU,SAAS,YAAY,OAAO,SACrF,WAAW,WAAW,UAAiB,EAAE,MAAM,UAAU,SAAS,SAAY,OAAO,KAAA,IACrF;AAEF,UAAM,aAAa,eACjB;AAAA,MAAC;AAAA,MAAA;AAAA,QACC,kBAAe;AAAA,QACf,WAAU;AAAA,QACV,OAAO,EAAE,OAAO,QAAQ,QAAQ,OAAA;AAAA,QAGhC,UAAA;AAAA,UAAA;AAAA,YAAC,aAAa;AAAA,YAAb;AAAA,cACC,MAAM;AAAA,cACN,WAAW;AAAA,gBACT;AAAA,gBACA,aAAa;AAAA,gBACb,eAAe;AAAA,cAAA;AAAA,cAEjB,eAAW;AAAA,YAAA;AAAA,UAAA;AAAA,UAGZ,eACC;AAAA,YAAC;AAAA,YAAA;AAAA,cACC,SAAQ;AAAA,cACR,MAAK;AAAA,cACL,UAAQ;AAAA,cACR,WAAW,YAAY;AAAA,cACvB,cAAY,YAAY;AAAA,cACxB,SAAS,CAAC,MAAM;AAAE,kBAAE,gBAAA;AAAmB,4BAAY,QAAA;AAAA,cAAU;AAAA,cAC7D,WAAU;AAAA,YAAA;AAAA,UAAA;AAAA,QACZ;AAAA,MAAA;AAAA,IAAA,IAGF;AAEJ,UAAM,SACJ;AAAA,MAAC;AAAA,MAAA;AAAA,QACC,WAAW;AAAA,UACT;AAAA,UACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,UAQA;AAAA,QAAA;AAAA,QAGD,UAAA;AAAA,UAAA,WAAW,eAAe,UACzB,qBAAC,QAAA,EAAK,WAAU,kCAAkC,UAAA;AAAA,YAAA;AAAA,YAAS;AAAA,UAAA,GAAC;AAAA,UAE7D;AAAA,UACA;AAAA,QAAA;AAAA,MAAA;AAAA,IAAA;AAQL,UAAM,aACJ,qBAAC,OAAA,EAAI,WAAU,0BACb,UAAA;AAAA,MAAA;AAAA,QAAC;AAAA,QAAA;AAAA,UACC,OAAO;AAAA,UACP,aAAa,WAAW,cAAc;AAAA,UACtC,MAAK;AAAA,UACL,iBAAiB,WAAW,UAAU,UAAU;AAAA,QAAA;AAAA,MAAA;AAAA,MAEjD;AAAA,IAAA,GACH;AAUF,UAAM,UAAU,CAAA;AAShB,UAAM,kBAAkB,CAAC,cAAc,iBAAiB;AAKxD,QAAI,QAAQ;AACV,aACE;AAAA,QAAC;AAAA,QAAA;AAAA,UACC;AAAA,UACA,WAAW;AAAA,YACT;AAAA,YACA;AAAA,YACA;AAAA,YACA;AAAA,UAAA;AAAA,UAEF;AAAA,UACC,GAAG;AAAA,UACH,GAAG;AAAA,UAEJ,UAAA;AAAA,YAAA,oBAAC,QAAA,EAAO,KAAK,cAAc,KAAK,MAAM,MAAM,aAAa,OAAM,UAAS,WAAU,WAAA,CAAW;AAAA,YAQ7F;AAAA,cAAC;AAAA,cAAA;AAAA,gBACC,WAAW;AAAA,kBACT;AAAA,kBACA,cAAc,oBAAoB;AAAA,gBAAA;AAAA,gBAEpC,OAAO,EAAE,WAAW,YAAA;AAAA,gBAEnB,UAAA;AAAA,kBAAA;AAAA,kBACA;AAAA,gBAAA;AAAA,cAAA;AAAA,YAAA;AAAA,UACH;AAAA,QAAA;AAAA,MAAA;AAAA,IAGN;AAGA,WACE;AAAA,MAAC;AAAA,MAAA;AAAA,QACC;AAAA,QACA,WAAW;AAAA,UACT;AAAA,UACA;AAAA,UACA;AAAA,UACA;AAAA,QAAA;AAAA,QAEF;AAAA,QACC,GAAG;AAAA,QACH,GAAG;AAAA,QAEJ,UAAA;AAAA,UAAA,oBAAC,YAAA,EACC,8BAAC,WAAA,EAAU,MAAM,SAAS,WAAU,0BAAyB,eAAW,KAAA,CAAC,EAAA,CAC3E;AAAA,UAIA,oBAAC,OAAA,EAAI,WAAU,gCACZ,UAAA,YACH;AAAA,UAGC,eACC,oBAAC,OAAA,EAAI,WAAU,6BAA4B,OAAO,EAAE,MAAM,kBAAkB,OAAO,eAAA,GAChF,UAAA,YAAA,CACH;AAAA,QAAA;AAAA,MAAA;AAAA,IAAA;AAAA,EAIR;AACF;AACA,SAAS,cAAc;AAIhB,MAAM,eAAe;AAAA,EAC1B,WAAW;AAAA,EACX,QAAQ;AAAA,EACR,UAAU,CAAA;AAAA,EAGV,OAAO,CAAA;AAAA,EAGP,QAAQ,CAAC,WAAW,SAAS,UAAU,iBAAiB,UAAU;AAAA,EAClE,QAAQ;AAAA,IACN,IAAI,CAAC,oBAAoB,gBAAgB,YAAY;AAAA,IACrD,IAAI,CAAC,iBAAiB,mBAAmB;AAAA,IACzC,MAAM,CAAA;AAAA,EAAC;AAEX;"}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
/**
|
|
3
|
+
* FileUpload — 拖放 / 點擊上傳區塊
|
|
4
|
+
*
|
|
5
|
+
* 世界級對照:Ant Design `Upload.Dragger`、Polaris `DropZone`、Material community MUI-File-Input。
|
|
6
|
+
* 與本 DS 既有 FileItem(顯示已上傳檔案)配對 — 這裡 own「上傳觸發 + 拖放偵測」,
|
|
7
|
+
* 上傳後的檔案清單顯示交給 consumer 用 FileItem 渲染。
|
|
8
|
+
*
|
|
9
|
+
* ── 4 狀態 ──
|
|
10
|
+
* idle (default) — border-dashed border-divider bg-surface
|
|
11
|
+
* drag-over — border-dashed border-primary bg-primary-subtle
|
|
12
|
+
* loading — 上傳中(async)顯示 CircularProgress,阻擋新互動
|
|
13
|
+
* disabled — opacity-disabled pointer-events-none
|
|
14
|
+
*
|
|
15
|
+
* ── children 插槽 ──
|
|
16
|
+
* 預設渲染 `<Empty icon={Upload} title description />` — 重用 Empty 元件 own
|
|
17
|
+
* 的「icon + title + description 垂直居中」SSOT 避免視覺漂移。Empty 改字體 /
|
|
18
|
+
* gap / icon 尺寸時 FileUpload 自動跟進。若 consumer 傳 children 則整個覆寫。
|
|
19
|
+
*
|
|
20
|
+
* ── API ──
|
|
21
|
+
* onUpload: 使用者選取或拖放檔案時觸發,回傳 File[](至少 1 個)。
|
|
22
|
+
* multiple: 允許多檔。預設 false(單檔)。
|
|
23
|
+
* accept: MIME filter(例 "image/*,.pdf"),傳給 <input type=file>。
|
|
24
|
+
* maxSize: 單檔最大 bytes;超過靜默忽略(consumer 若需要錯誤提示,走 onReject)。
|
|
25
|
+
* onReject: 被 maxSize / accept 擋下的檔案(提供錯誤訊息顯示機會)。
|
|
26
|
+
*/
|
|
27
|
+
/**
|
|
28
|
+
* Uploaded / uploading file status item(for `files` prop)。
|
|
29
|
+
* Consumer 持 state(progress / status),FileUpload 只負責渲染。
|
|
30
|
+
*/
|
|
31
|
+
export interface FileUploadStatus {
|
|
32
|
+
id: string;
|
|
33
|
+
name: string;
|
|
34
|
+
/** bytes */
|
|
35
|
+
size?: number;
|
|
36
|
+
/** Upload 進度(0-100)— uploading 時顯示 progress bar */
|
|
37
|
+
progress?: number;
|
|
38
|
+
status?: 'uploading' | 'completed' | 'error';
|
|
39
|
+
/** Error 訊息 / size 等 description */
|
|
40
|
+
description?: React.ReactNode;
|
|
41
|
+
/** Thumbnail URL for rich mode */
|
|
42
|
+
thumbnailSrc?: string;
|
|
43
|
+
}
|
|
44
|
+
export interface FileUploadProps extends Omit<React.HTMLAttributes<HTMLDivElement>, 'onDrop'> {
|
|
45
|
+
onUpload?: (files: File[]) => void;
|
|
46
|
+
onReject?: (files: File[], reason: 'size' | 'type') => void;
|
|
47
|
+
multiple?: boolean;
|
|
48
|
+
accept?: string;
|
|
49
|
+
maxSize?: number;
|
|
50
|
+
disabled?: boolean;
|
|
51
|
+
/**
|
|
52
|
+
* Loading 狀態(async 上傳 / 伺服器處理中)。
|
|
53
|
+
* - 顯示 CircularProgress 取代預設 Empty 內容
|
|
54
|
+
* - 阻擋新點擊 / drag 事件(避免 double-submit)
|
|
55
|
+
* - 宣告 `aria-busy="true"` 讓 screen reader 感知處理中
|
|
56
|
+
* - Consumer 負責在上傳完成後自己切回 `loading={false}`
|
|
57
|
+
*/
|
|
58
|
+
loading?: boolean;
|
|
59
|
+
/** Loading 狀態的文字標題(預設「上傳中…」) */
|
|
60
|
+
loadingTitle?: string;
|
|
61
|
+
/** 標題文字(預設「Click or drag file here to upload」) */
|
|
62
|
+
title?: string;
|
|
63
|
+
/** 說明文字(預設「Support for a single or bulk upload」) */
|
|
64
|
+
description?: string;
|
|
65
|
+
/** 若傳入 children,覆寫預設 Empty 結構 */
|
|
66
|
+
children?: React.ReactNode;
|
|
67
|
+
/**
|
|
68
|
+
* Uploaded / uploading 檔案清單。傳入 → FileUpload 在 drop zone 下方渲染列表。
|
|
69
|
+
* 不傳 → 不顯示(consumer 可自行用 FileItem 組成,pre-2026-04-24 行為)。
|
|
70
|
+
* 每項有 status 狀態會對應 icon:
|
|
71
|
+
* - `uploading`:CircularProgress(+ progress bar 若 progress 給)
|
|
72
|
+
* - `completed`:綠色 ✓(success 狀態視覺確認)
|
|
73
|
+
* - `error`:紅色 ✗(+ description 顯示錯誤訊息)
|
|
74
|
+
*/
|
|
75
|
+
files?: FileUploadStatus[];
|
|
76
|
+
/** File list 每項顯示模式。Default: 'compact'(單行);'rich' = 含 thumbnail / size / progress bar */
|
|
77
|
+
fileListMode?: 'compact' | 'rich';
|
|
78
|
+
/** File list 移除 callback。有值 → 每項右側顯示 X dismiss button;無 → 不可移除(view-only) */
|
|
79
|
+
onRemove?: (id: string) => void;
|
|
80
|
+
/** File list dismiss button ARIA label template。預設 `移除 {name}`。For i18n. */
|
|
81
|
+
removeAriaLabel?: (name: string) => string;
|
|
82
|
+
}
|
|
83
|
+
declare const FileUpload: React.ForwardRefExoticComponent<FileUploadProps & React.RefAttributes<HTMLDivElement>>;
|
|
84
|
+
export declare const fileUploadMeta: {
|
|
85
|
+
readonly component: "FileUpload";
|
|
86
|
+
readonly family: null;
|
|
87
|
+
readonly variants: {};
|
|
88
|
+
readonly sizes: {};
|
|
89
|
+
readonly states: readonly ["default", "hover", "active", "focus-visible", "disabled"];
|
|
90
|
+
readonly tokens: {
|
|
91
|
+
readonly bg: readonly ["bg-neutral-hover", "bg-primary-subtle", "bg-surface"];
|
|
92
|
+
readonly fg: readonly [];
|
|
93
|
+
readonly ring: readonly ["ring-ring"];
|
|
94
|
+
};
|
|
95
|
+
};
|
|
96
|
+
export { FileUpload };
|
|
97
|
+
//# sourceMappingURL=file-upload.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"file-upload.d.ts","sourceRoot":"","sources":["../../../src/components/FileUpload/file-upload.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAA;AAQ9B;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AAEH;;;GAGG;AACH,MAAM,WAAW,gBAAgB;IAC/B,EAAE,EAAE,MAAM,CAAA;IACV,IAAI,EAAE,MAAM,CAAA;IACZ,YAAY;IACZ,IAAI,CAAC,EAAE,MAAM,CAAA;IACb,mDAAmD;IACnD,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,MAAM,CAAC,EAAE,WAAW,GAAG,WAAW,GAAG,OAAO,CAAA;IAC5C,oCAAoC;IACpC,WAAW,CAAC,EAAE,KAAK,CAAC,SAAS,CAAA;IAC7B,kCAAkC;IAClC,YAAY,CAAC,EAAE,MAAM,CAAA;CACtB;AAED,MAAM,WAAW,eAAgB,SAAQ,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,EAAE,QAAQ,CAAC;IAC3F,QAAQ,CAAC,EAAE,CAAC,KAAK,EAAE,IAAI,EAAE,KAAK,IAAI,CAAA;IAClC,QAAQ,CAAC,EAAE,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,KAAK,IAAI,CAAA;IAC3D,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB;;;;;;OAMG;IACH,OAAO,CAAC,EAAE,OAAO,CAAA;IACjB,gCAAgC;IAChC,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,kDAAkD;IAClD,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,oDAAoD;IACpD,WAAW,CAAC,EAAE,MAAM,CAAA;IACpB,iCAAiC;IACjC,QAAQ,CAAC,EAAE,KAAK,CAAC,SAAS,CAAA;IAC1B;;;;;;;OAOG;IACH,KAAK,CAAC,EAAE,gBAAgB,EAAE,CAAA;IAC1B,yFAAyF;IACzF,YAAY,CAAC,EAAE,SAAS,GAAG,MAAM,CAAA;IACjC,6EAA6E;IAC7E,QAAQ,CAAC,EAAE,CAAC,EAAE,EAAE,MAAM,KAAK,IAAI,CAAA;IAC/B,4EAA4E;IAC5E,eAAe,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,MAAM,CAAA;CAC3C;AAGD,QAAA,MAAM,UAAU,wFA4Lf,CAAA;AA0BD,eAAO,MAAM,cAAc;;;;;;;;;;;CAejB,CAAA;AAEV,OAAO,EAAE,UAAU,EAAE,CAAA"}
|
|
@@ -0,0 +1,231 @@
|
|
|
1
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
2
|
+
import * as React from "react";
|
|
3
|
+
import { X, Upload } from "lucide-react";
|
|
4
|
+
import { cn } from "../../lib/utils.js";
|
|
5
|
+
import { Empty } from "../Empty/empty.js";
|
|
6
|
+
import { CircularProgress } from "../CircularProgress/circular-progress.js";
|
|
7
|
+
import { FileItem } from "../FileItem/file-item.js";
|
|
8
|
+
import { Button } from "../Button/button.js";
|
|
9
|
+
const FileUpload = React.forwardRef(
|
|
10
|
+
({
|
|
11
|
+
onUpload,
|
|
12
|
+
onReject,
|
|
13
|
+
multiple = false,
|
|
14
|
+
accept,
|
|
15
|
+
maxSize,
|
|
16
|
+
disabled = false,
|
|
17
|
+
loading = false,
|
|
18
|
+
loadingTitle = "上傳中…",
|
|
19
|
+
// i18n-allow: DS default; consumer override via loadingTitle prop
|
|
20
|
+
title = "Click or drag file here to upload",
|
|
21
|
+
// i18n-allow: DS default; consumer override via title prop
|
|
22
|
+
description = multiple ? "Support for a single or bulk upload" : "Support for a single file upload",
|
|
23
|
+
// i18n-allow: DS default; consumer override via description prop
|
|
24
|
+
children,
|
|
25
|
+
files,
|
|
26
|
+
fileListMode = "compact",
|
|
27
|
+
onRemove,
|
|
28
|
+
removeAriaLabel = (name) => `移除 ${name}`,
|
|
29
|
+
// i18n-allow: DS default; consumer override via removeAriaLabel prop
|
|
30
|
+
className,
|
|
31
|
+
onClick,
|
|
32
|
+
...props
|
|
33
|
+
}, ref) => {
|
|
34
|
+
const inputRef = React.useRef(null);
|
|
35
|
+
const [isDragOver, setDragOver] = React.useState(false);
|
|
36
|
+
const filterAndDispatch = (files2) => {
|
|
37
|
+
if (!files2 || files2.length === 0) return;
|
|
38
|
+
const accepted = [];
|
|
39
|
+
const rejectedBySize = [];
|
|
40
|
+
const rejectedByType = [];
|
|
41
|
+
Array.from(files2).forEach((f) => {
|
|
42
|
+
if (maxSize != null && f.size > maxSize) {
|
|
43
|
+
rejectedBySize.push(f);
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
if (accept && !matchAccept(f, accept)) {
|
|
47
|
+
rejectedByType.push(f);
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
accepted.push(f);
|
|
51
|
+
});
|
|
52
|
+
if (rejectedBySize.length) onReject == null ? void 0 : onReject(rejectedBySize, "size");
|
|
53
|
+
if (rejectedByType.length) onReject == null ? void 0 : onReject(rejectedByType, "type");
|
|
54
|
+
if (accepted.length) onUpload == null ? void 0 : onUpload(multiple ? accepted : accepted.slice(0, 1));
|
|
55
|
+
};
|
|
56
|
+
const handleClick = (e) => {
|
|
57
|
+
var _a;
|
|
58
|
+
if (!disabled) (_a = inputRef.current) == null ? void 0 : _a.click();
|
|
59
|
+
onClick == null ? void 0 : onClick(e);
|
|
60
|
+
};
|
|
61
|
+
const state = disabled ? "disabled" : loading ? "loading" : isDragOver ? "drag-over" : "idle";
|
|
62
|
+
const isBlocked = disabled || loading;
|
|
63
|
+
const hasFiles = Array.isArray(files) && files.length > 0;
|
|
64
|
+
const fileListNode = hasFiles ? /* @__PURE__ */ jsx(
|
|
65
|
+
"ul",
|
|
66
|
+
{
|
|
67
|
+
className: cn(
|
|
68
|
+
// stack vertically with consistent gap;bg-surface 在容器外的 chrome / consumer bg 之上
|
|
69
|
+
"flex flex-col gap-2 w-full",
|
|
70
|
+
"mt-3"
|
|
71
|
+
),
|
|
72
|
+
"aria-label": "Uploaded files",
|
|
73
|
+
children: files.map((f) => /* @__PURE__ */ jsx("li", { className: "list-none", children: /* @__PURE__ */ jsx(
|
|
74
|
+
FileItem,
|
|
75
|
+
{
|
|
76
|
+
mode: fileListMode,
|
|
77
|
+
name: f.name,
|
|
78
|
+
status: f.status,
|
|
79
|
+
progress: f.progress,
|
|
80
|
+
description: f.description ?? (f.size != null ? formatBytes(f.size) : void 0),
|
|
81
|
+
thumbnailSrc: f.thumbnailSrc,
|
|
82
|
+
actions: onRemove ? (
|
|
83
|
+
// Collection remove(per-file)— 不是 dismiss surface,故不套 `dismiss` prop。
|
|
84
|
+
// 視覺與 dismiss 一致(text variant + fg-muted dim)— 對齊 inline-action.spec.md
|
|
85
|
+
// 「Dismiss canonical — X close only」L249-251:onRemove callback 不觸發 dismiss prop。
|
|
86
|
+
/* @__PURE__ */ jsx(
|
|
87
|
+
Button,
|
|
88
|
+
{
|
|
89
|
+
iconOnly: true,
|
|
90
|
+
variant: "text",
|
|
91
|
+
size: "xs",
|
|
92
|
+
startIcon: X,
|
|
93
|
+
"aria-label": removeAriaLabel(f.name),
|
|
94
|
+
onClick: (e) => {
|
|
95
|
+
e.stopPropagation();
|
|
96
|
+
onRemove(f.id);
|
|
97
|
+
},
|
|
98
|
+
className: "text-fg-muted hover:text-foreground"
|
|
99
|
+
}
|
|
100
|
+
)
|
|
101
|
+
) : void 0
|
|
102
|
+
}
|
|
103
|
+
) }, f.id))
|
|
104
|
+
}
|
|
105
|
+
) : null;
|
|
106
|
+
return /* @__PURE__ */ jsxs("div", { ref, className: cn("w-full", hasFiles && "flex flex-col"), children: [
|
|
107
|
+
/* @__PURE__ */ jsxs(
|
|
108
|
+
"div",
|
|
109
|
+
{
|
|
110
|
+
role: "button",
|
|
111
|
+
tabIndex: isBlocked ? -1 : 0,
|
|
112
|
+
"aria-disabled": disabled || void 0,
|
|
113
|
+
"aria-busy": loading || void 0,
|
|
114
|
+
"data-state": state,
|
|
115
|
+
onClick: handleClick,
|
|
116
|
+
onKeyDown: (e) => {
|
|
117
|
+
var _a;
|
|
118
|
+
if (isBlocked) return;
|
|
119
|
+
if (e.key === "Enter" || e.key === " ") {
|
|
120
|
+
e.preventDefault();
|
|
121
|
+
(_a = inputRef.current) == null ? void 0 : _a.click();
|
|
122
|
+
}
|
|
123
|
+
},
|
|
124
|
+
onDragEnter: (e) => {
|
|
125
|
+
if (isBlocked) return;
|
|
126
|
+
e.preventDefault();
|
|
127
|
+
setDragOver(true);
|
|
128
|
+
},
|
|
129
|
+
onDragLeave: (e) => {
|
|
130
|
+
if (isBlocked) return;
|
|
131
|
+
e.preventDefault();
|
|
132
|
+
setDragOver(false);
|
|
133
|
+
},
|
|
134
|
+
onDragOver: (e) => {
|
|
135
|
+
if (isBlocked) return;
|
|
136
|
+
e.preventDefault();
|
|
137
|
+
},
|
|
138
|
+
onDrop: (e) => {
|
|
139
|
+
if (isBlocked) return;
|
|
140
|
+
e.preventDefault();
|
|
141
|
+
setDragOver(false);
|
|
142
|
+
filterAndDispatch(e.dataTransfer.files);
|
|
143
|
+
},
|
|
144
|
+
className: cn(
|
|
145
|
+
// 寬度 w-full 填滿 consumer 容器(user 明示「寬度填滿」);高度由 padding + 內容物決定(不固定 h)
|
|
146
|
+
"flex flex-col items-center justify-center gap-2 text-center w-full",
|
|
147
|
+
// 對稱 padding p-[var(--layout-space-loose)]:四邊等距,density-aware(md=24px / lg=32px),對齊 DS chrome padding canonical。
|
|
148
|
+
// 不再硬寫 px-6 py-10(不對稱+非 token)。內容物(icon + title + description)垂直堆疊由 gap-2 控制
|
|
149
|
+
"rounded-md border-2 border-dashed p-[var(--layout-space-loose)]",
|
|
150
|
+
"cursor-pointer transition-colors",
|
|
151
|
+
"focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2",
|
|
152
|
+
// idle
|
|
153
|
+
"border-divider bg-surface hover:bg-neutral-hover",
|
|
154
|
+
// drag-over
|
|
155
|
+
"data-[state=drag-over]:border-primary data-[state=drag-over]:bg-primary-subtle data-[state=drag-over]:hover:bg-primary-subtle",
|
|
156
|
+
// loading(阻擋新互動但不 opacity-disabled — 避免跟 disabled 視覺撞,保持「處理中」語意)
|
|
157
|
+
"data-[state=loading]:cursor-progress data-[state=loading]:pointer-events-none",
|
|
158
|
+
// disabled
|
|
159
|
+
"data-[state=disabled]:opacity-disabled data-[state=disabled]:pointer-events-none data-[state=disabled]:cursor-not-allowed",
|
|
160
|
+
className
|
|
161
|
+
),
|
|
162
|
+
...props,
|
|
163
|
+
children: [
|
|
164
|
+
/* @__PURE__ */ jsx(
|
|
165
|
+
"input",
|
|
166
|
+
{
|
|
167
|
+
ref: inputRef,
|
|
168
|
+
type: "file",
|
|
169
|
+
className: "hidden",
|
|
170
|
+
multiple,
|
|
171
|
+
accept,
|
|
172
|
+
disabled,
|
|
173
|
+
onChange: (e) => filterAndDispatch(e.target.files)
|
|
174
|
+
}
|
|
175
|
+
),
|
|
176
|
+
loading ? /* @__PURE__ */ jsx(
|
|
177
|
+
Empty,
|
|
178
|
+
{
|
|
179
|
+
icon: /* @__PURE__ */ jsx(CircularProgress, { size: 48 }),
|
|
180
|
+
title: loadingTitle
|
|
181
|
+
}
|
|
182
|
+
) : children ?? /* @__PURE__ */ jsx(
|
|
183
|
+
Empty,
|
|
184
|
+
{
|
|
185
|
+
icon: Upload,
|
|
186
|
+
title,
|
|
187
|
+
description
|
|
188
|
+
}
|
|
189
|
+
)
|
|
190
|
+
]
|
|
191
|
+
}
|
|
192
|
+
),
|
|
193
|
+
fileListNode
|
|
194
|
+
] });
|
|
195
|
+
}
|
|
196
|
+
);
|
|
197
|
+
FileUpload.displayName = "FileUpload";
|
|
198
|
+
function formatBytes(n) {
|
|
199
|
+
if (n < 1024) return `${n} B`;
|
|
200
|
+
if (n < 1024 * 1024) return `${(n / 1024).toFixed(1)} KB`;
|
|
201
|
+
if (n < 1024 * 1024 * 1024) return `${(n / (1024 * 1024)).toFixed(1)} MB`;
|
|
202
|
+
return `${(n / (1024 * 1024 * 1024)).toFixed(1)} GB`;
|
|
203
|
+
}
|
|
204
|
+
function matchAccept(file, accept) {
|
|
205
|
+
const patterns = accept.split(",").map((s) => s.trim().toLowerCase());
|
|
206
|
+
const fileName = file.name.toLowerCase();
|
|
207
|
+
const fileType = file.type.toLowerCase();
|
|
208
|
+
return patterns.some((p) => {
|
|
209
|
+
if (p.startsWith(".")) return fileName.endsWith(p);
|
|
210
|
+
if (p.endsWith("/*")) return fileType.startsWith(p.slice(0, -1));
|
|
211
|
+
return fileType === p;
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
const fileUploadMeta = {
|
|
215
|
+
component: "FileUpload",
|
|
216
|
+
family: null,
|
|
217
|
+
// non-family composite / overlay / layout
|
|
218
|
+
variants: {},
|
|
219
|
+
sizes: {},
|
|
220
|
+
states: ["default", "hover", "active", "focus-visible", "disabled"],
|
|
221
|
+
tokens: {
|
|
222
|
+
bg: ["bg-neutral-hover", "bg-primary-subtle", "bg-surface"],
|
|
223
|
+
fg: [],
|
|
224
|
+
ring: ["ring-ring"]
|
|
225
|
+
}
|
|
226
|
+
};
|
|
227
|
+
export {
|
|
228
|
+
FileUpload,
|
|
229
|
+
fileUploadMeta
|
|
230
|
+
};
|
|
231
|
+
//# sourceMappingURL=file-upload.js.map
|