@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,198 @@
|
|
|
1
|
+
// AppShell stories 共用 helper(對齊 sidebar.stories.tsx IconCollapse baseline)
|
|
2
|
+
// @story-baseline: packages/design-system/src/components/Sidebar/sidebar.stories.tsx#IconCollapse
|
|
3
|
+
//
|
|
4
|
+
// **嚴格**對齊既有 production-grade Sidebar story baseline,避免 AppShell stories 跟
|
|
5
|
+
// Sidebar 既有範例視覺偏移(2026-05-20 user 抓 anti-drift)。
|
|
6
|
+
// Showcase + Anatomy stories 全部 consume 此 file 不重發明 simplified mock。
|
|
7
|
+
|
|
8
|
+
import {
|
|
9
|
+
Inbox,
|
|
10
|
+
Calendar,
|
|
11
|
+
Settings,
|
|
12
|
+
Users,
|
|
13
|
+
BarChart3,
|
|
14
|
+
LayoutDashboard,
|
|
15
|
+
} from 'lucide-react'
|
|
16
|
+
import {
|
|
17
|
+
Sidebar,
|
|
18
|
+
SidebarHeader,
|
|
19
|
+
SidebarContent,
|
|
20
|
+
SidebarGroup,
|
|
21
|
+
SidebarGroupContent,
|
|
22
|
+
SidebarMenu,
|
|
23
|
+
SidebarMenuItem,
|
|
24
|
+
SidebarMenuButton,
|
|
25
|
+
SidebarFooter,
|
|
26
|
+
SidebarTrigger,
|
|
27
|
+
} from '@/design-system/components/Sidebar/sidebar'
|
|
28
|
+
import { ChromeHeader } from '@/design-system/patterns/header-canonical/chrome-header'
|
|
29
|
+
import {
|
|
30
|
+
ItemAvatar,
|
|
31
|
+
} from '@/design-system/patterns/element-anatomy/item-anatomy'
|
|
32
|
+
import { Avatar } from '@/design-system/components/Avatar/avatar'
|
|
33
|
+
import {
|
|
34
|
+
NameCard,
|
|
35
|
+
NameCardDefaultActions,
|
|
36
|
+
} from '@/design-system/components/NameCard/name-card'
|
|
37
|
+
|
|
38
|
+
// ── MAIN_NAV(對齊 sidebar.stories.tsx baseline)────────────────────────
|
|
39
|
+
|
|
40
|
+
export const MAIN_NAV = [
|
|
41
|
+
{ id: 'dashboard', label: 'Dashboard', icon: LayoutDashboard },
|
|
42
|
+
{ id: 'inbox', label: 'Inbox', icon: Inbox },
|
|
43
|
+
{ id: 'team', label: 'Team', icon: Users },
|
|
44
|
+
{ id: 'insights', label: 'Insights', icon: BarChart3 },
|
|
45
|
+
{ id: 'calendar', label: 'Calendar', icon: Calendar },
|
|
46
|
+
{ id: 'settings', label: 'Settings', icon: Settings },
|
|
47
|
+
] as const
|
|
48
|
+
|
|
49
|
+
// ── WorkspaceBrand(對齊 sidebar.stories.tsx)────────────────────────────
|
|
50
|
+
|
|
51
|
+
// 2026-05-21 v15 — Chrome header avatar SSOT, semantic-correct revise(per user 抓 v14
|
|
52
|
+
// RowSizeProvider hijack 是 semantic 漂移 + 「想辦法在語言正確下修到正確」directive):
|
|
53
|
+
//
|
|
54
|
+
// Chrome header **不是 row context**(per `item-anatomy.spec.md:550` 規則 scope 是 row primitive
|
|
55
|
+
// consumer + `header-canonical.spec.md` 4.5 chrome header avatar SSOT)。Chrome header avatar
|
|
56
|
+
// 是 spec-level canonical 24px、density-fixed、row-size-fixed,跟 row-anatomy 的 sm/md/lg
|
|
57
|
+
// lookup 邏輯無關。
|
|
58
|
+
//
|
|
59
|
+
// 因此 chrome header 內 avatar **用 raw `<Avatar size={24}>`,不用 `<ItemAvatar>`**:
|
|
60
|
+
// - ItemAvatar 透過 RowSizeContext lookup 是 row primitive(Sidebar / SelectMenu / TreeView 等)
|
|
61
|
+
// 的 anatomy helper,目的避免 asChild consumer 寫死跨 row size 漂移
|
|
62
|
+
// - Chrome header 沒有 sm/md/lg row size 概念,無 lookup 需求,用 raw Avatar 才語義正確
|
|
63
|
+
// - 父 `<div flex items-center>` 已 provide 縱向對齊,ItemPrefix wrapper(`h-[1lh]`)冗餘
|
|
64
|
+
//
|
|
65
|
+
// SSOT 鎖定:`<Avatar size={24}>` 對應 `--chrome-header-avatar-size: 1.5rem` token
|
|
66
|
+
// (`header-canonical.css`)+ `header-canonical.spec.md` 4.5 canonical「24px raw Avatar」。
|
|
67
|
+
// 公式端透過 var() 連動;JS 端透過 spec authority + comment cite 連動。改 24 → 同步 token + 此 hardcode。
|
|
68
|
+
export const WorkspaceBrand = () => (
|
|
69
|
+
<div className="flex items-center gap-2 min-w-0">
|
|
70
|
+
{/* 24 per header-canonical.spec.md 4.5 chrome header avatar canonical; sync with `--chrome-header-avatar-size` */}
|
|
71
|
+
<Avatar size={24} shape="square" color="blue" solid alt="Acme Inc" />
|
|
72
|
+
<span className="text-body-lg font-medium truncate group-data-[collapsible=icon]:hidden">Acme Inc</span>
|
|
73
|
+
</div>
|
|
74
|
+
)
|
|
75
|
+
|
|
76
|
+
// ── UserFooter(對齊 sidebar.stories.tsx)────────────────────────────────
|
|
77
|
+
|
|
78
|
+
export const UserFooter = () => (
|
|
79
|
+
<SidebarMenu>
|
|
80
|
+
<SidebarMenuItem>
|
|
81
|
+
<SidebarMenuButton asChild>
|
|
82
|
+
<div role="group" aria-label="當前使用者">
|
|
83
|
+
<ItemAvatar
|
|
84
|
+
alt="Alan Chen"
|
|
85
|
+
color="blue"
|
|
86
|
+
hoverCard={
|
|
87
|
+
<NameCard
|
|
88
|
+
name="Alan Chen"
|
|
89
|
+
subtitle="Design|D-0042"
|
|
90
|
+
avatar={{ alt: 'Alan Chen', color: 'blue' }}
|
|
91
|
+
status="online"
|
|
92
|
+
actions={<NameCardDefaultActions />}
|
|
93
|
+
/>
|
|
94
|
+
}
|
|
95
|
+
/>
|
|
96
|
+
<span data-sidebar="menu-label" className="min-w-0 flex-1 truncate">Alan Chen</span>
|
|
97
|
+
</div>
|
|
98
|
+
</SidebarMenuButton>
|
|
99
|
+
</SidebarMenuItem>
|
|
100
|
+
</SidebarMenu>
|
|
101
|
+
)
|
|
102
|
+
|
|
103
|
+
// ── AcmeSidebar(完整 production-grade,對齊 sidebar IconCollapse story)──
|
|
104
|
+
// `includeWorkspaceBrand` default true(primary-sidebar 派 Linear/Notion 慣例:workspace brand 在 sidebar 頂)。
|
|
105
|
+
// `false` 用於 primary-header mode:workspace brand 移到 globalHeader 左側(GitHub logo / Slack workspace bar 慣例)。
|
|
106
|
+
|
|
107
|
+
export function AcmeSidebar({
|
|
108
|
+
viewportInsetTop,
|
|
109
|
+
includeWorkspaceBrand = true,
|
|
110
|
+
}: {
|
|
111
|
+
viewportInsetTop?: string
|
|
112
|
+
includeWorkspaceBrand?: boolean
|
|
113
|
+
} = {}) {
|
|
114
|
+
return (
|
|
115
|
+
<Sidebar collapsible="icon" viewportInsetTop={viewportInsetTop}>
|
|
116
|
+
{includeWorkspaceBrand && (
|
|
117
|
+
<SidebarHeader>
|
|
118
|
+
<WorkspaceBrand />
|
|
119
|
+
</SidebarHeader>
|
|
120
|
+
)}
|
|
121
|
+
<SidebarContent>
|
|
122
|
+
<SidebarGroup>
|
|
123
|
+
<SidebarGroupContent>
|
|
124
|
+
<SidebarMenu>
|
|
125
|
+
{MAIN_NAV.map((item) => (
|
|
126
|
+
<SidebarMenuItem key={item.id}>
|
|
127
|
+
<SidebarMenuButton
|
|
128
|
+
id={item.id}
|
|
129
|
+
startIcon={item.icon}
|
|
130
|
+
tooltip={item.label}
|
|
131
|
+
>
|
|
132
|
+
{item.label}
|
|
133
|
+
</SidebarMenuButton>
|
|
134
|
+
</SidebarMenuItem>
|
|
135
|
+
))}
|
|
136
|
+
</SidebarMenu>
|
|
137
|
+
</SidebarGroupContent>
|
|
138
|
+
</SidebarGroup>
|
|
139
|
+
</SidebarContent>
|
|
140
|
+
<SidebarFooter>
|
|
141
|
+
<UserFooter />
|
|
142
|
+
</SidebarFooter>
|
|
143
|
+
</Sidebar>
|
|
144
|
+
)
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// ── GlobalHeader(primary-header mode 用,跨頁 chrome:WorkspaceBrand 左 + 跨頁 actions 右)──
|
|
148
|
+
// 2026-05-21 加 per user clarification「primary-header = primary-sidebar + 一條 global header」。
|
|
149
|
+
// 對齊 GitHub top nav(logo 左 / search 中 / account 右)+ Slack workspace bar 慣例。
|
|
150
|
+
// 消費 ChromeHeader(per `header-canonical.spec.md` Element + Background ownership 段:
|
|
151
|
+
// top-level chrome → 自畫 bg-surface)。
|
|
152
|
+
|
|
153
|
+
export function GlobalHeader({ rightSlot }: { rightSlot?: React.ReactNode } = {}) {
|
|
154
|
+
// 2026-05-21 v2 ship Option B(per user「primary header toggle 為了與 sidebar menu item
|
|
155
|
+
// startIcon 水平對齊...container 寬度 = sidebar-width-icon」+ Issue 2 geometry formula 落地):
|
|
156
|
+
// 用 ChromeHeader `leadingRail` slot(width = `--sidebar-width-icon` = 2*loose + icon-size)。
|
|
157
|
+
// Toggle center x = rail 寬度中點 = sidebar collapsed icon center x = 完美 vertical 對齊。
|
|
158
|
+
return (
|
|
159
|
+
<ChromeHeader className="bg-surface" leadingRail={<SidebarTrigger />}>
|
|
160
|
+
<WorkspaceBrand />
|
|
161
|
+
<div className="flex-1" />
|
|
162
|
+
{rightSlot}
|
|
163
|
+
</ChromeHeader>
|
|
164
|
+
)
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
// ── PageHeader(top-level chrome header,消費 ChromeHeader primitive)──
|
|
168
|
+
// 2026-05-20 migrate 消費 ChromeHeader,撤回自刻 `<header>` + 重複 className
|
|
169
|
+
// (per header-canonical.spec.md「6. Background ownership」段「Top-level chrome
|
|
170
|
+
// header 自畫 bg-surface」+「Element canonical」段「ChromeHeader 內部用 `<header>`」)。
|
|
171
|
+
// global header aside toggle 已撤回(2026-05-20 user「圖二 global header 不該有」)— 由
|
|
172
|
+
// DataTable rowActions Info icon 主入口(row-driven)取代,page header 純 title。
|
|
173
|
+
|
|
174
|
+
export function PageHeader({
|
|
175
|
+
title,
|
|
176
|
+
tabsSlot,
|
|
177
|
+
includeSidebarTrigger = true,
|
|
178
|
+
}: {
|
|
179
|
+
title: string
|
|
180
|
+
/**
|
|
181
|
+
* Optional tabs row(per header-canonical.spec.md W1-W6 + Background ownership 段)。
|
|
182
|
+
* 提供時 ChromeHeader 自動 column mode + suppress border + delegate paint 給 TabsList。
|
|
183
|
+
*/
|
|
184
|
+
tabsSlot?: React.ReactNode
|
|
185
|
+
/**
|
|
186
|
+
* 是否含 SidebarTrigger(2026-05-21 加 per user「primary-header mode 的 sidebar toggle 應該只放在 primary header 才對」)。
|
|
187
|
+
* - `primary-sidebar` mode = true(預設):PageHeader 是 chrome 第一層,trigger 自然在這
|
|
188
|
+
* - `primary-header` mode = false:SidebarTrigger 已在 GlobalHeader,PageHeader 不該重複
|
|
189
|
+
*/
|
|
190
|
+
includeSidebarTrigger?: boolean
|
|
191
|
+
}) {
|
|
192
|
+
return (
|
|
193
|
+
<ChromeHeader className="bg-surface" tabsSlot={tabsSlot}>
|
|
194
|
+
{includeSidebarTrigger && <SidebarTrigger />}
|
|
195
|
+
<h1 className="text-body-lg font-medium flex-1 truncate">{title}</h1>
|
|
196
|
+
</ChromeHeader>
|
|
197
|
+
)
|
|
198
|
+
}
|
|
@@ -0,0 +1,364 @@
|
|
|
1
|
+
// @benchmark-cited: 2026-05-19 — Mantine AppShell / Ant Layout / Material 3 Drawer / Atlassian Navigation System cite in app-shell.spec.md frontmatter.
|
|
2
|
+
/**
|
|
3
|
+
* AppShell — web service page-level layout primitive。
|
|
4
|
+
*
|
|
5
|
+
* 組合 Sidebar + ChromeHeader + Aside + main 成完整 page shell。SSOT 邊界:本 pattern only
|
|
6
|
+
* own slot composition + layout mode + Aside responsive mode;不 own sidebar / header /
|
|
7
|
+
* sheet 視覺(各自 spec own)。
|
|
8
|
+
*
|
|
9
|
+
* 對齊 Mantine AppShell compound API + Ant Layout slot 模式 + Material 3 standard/modal
|
|
10
|
+
* drawer canonical(per spec.md frontmatter cite)。
|
|
11
|
+
*
|
|
12
|
+
* Spec SSOT:`patterns/app-shell/app-shell.spec.md`
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
import * as React from 'react'
|
|
16
|
+
import { X as XIcon } from 'lucide-react'
|
|
17
|
+
import {
|
|
18
|
+
Sheet,
|
|
19
|
+
SheetContent,
|
|
20
|
+
} from '@/design-system/components/Sheet/sheet'
|
|
21
|
+
import { Button } from '@/design-system/components/Button/button'
|
|
22
|
+
import { ScrollArea } from '@/design-system/components/ScrollArea/scroll-area'
|
|
23
|
+
import { ChromeHeader } from '@/design-system/patterns/header-canonical/chrome-header'
|
|
24
|
+
import { useIsNarrowViewport } from '@/design-system/hooks/use-is-narrow-viewport'
|
|
25
|
+
import { cn } from '@/lib/utils'
|
|
26
|
+
|
|
27
|
+
// ── Types ────────────────────────────────────────────────────────────────────
|
|
28
|
+
|
|
29
|
+
type AppShellLayout = 'primary-sidebar' | 'primary-header'
|
|
30
|
+
|
|
31
|
+
export interface AppShellProps extends React.HTMLAttributes<HTMLDivElement> {
|
|
32
|
+
/** primary-sidebar (Linear/Notion 派) | primary-header (GitHub/Slack 派);預設 primary-sidebar */
|
|
33
|
+
layout?: AppShellLayout
|
|
34
|
+
/** Sidebar 元素(必傳 Sidebar primitive,per Consumer 紀律)*/
|
|
35
|
+
sidebar?: React.ReactNode
|
|
36
|
+
/**
|
|
37
|
+
* Local page header(永遠 render 在 main column 頂部,當前頁 actions / breadcrumb / page-level filter)。
|
|
38
|
+
* 兩 layout mode 都會 render 此 slot — `primary-header` mode 多了 globalHeader 在上方,
|
|
39
|
+
* **不取代** local header(per 2026-05-20 user clarification「primary-header = primary-sidebar + 一條 global header」)。
|
|
40
|
+
*/
|
|
41
|
+
header?: React.ReactNode
|
|
42
|
+
/**
|
|
43
|
+
* Global header(僅 `primary-header` mode render)。橫跨整 viewport 的頂部 bar:
|
|
44
|
+
* account avatar / workspace switcher / global search / notifications。
|
|
45
|
+
* 對齊 GitHub top nav / Slack workspace bar / Gmail logo bar 慣例。
|
|
46
|
+
* `primary-sidebar` mode 傳此 prop 會被忽略。
|
|
47
|
+
*/
|
|
48
|
+
globalHeader?: React.ReactNode
|
|
49
|
+
/** Aside 元素(`<AppShellAside>` sub-component);可選 */
|
|
50
|
+
aside?: React.ReactNode
|
|
51
|
+
/** Aside open state(modal mode 必須)*/
|
|
52
|
+
asideOpen?: boolean
|
|
53
|
+
onAsideOpenChange?: (open: boolean) => void
|
|
54
|
+
/** Main content;`<main>` landmark + padding=0 */
|
|
55
|
+
children: React.ReactNode
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
export interface AppShellAsideProps {
|
|
59
|
+
/** Required:modal mode 走 Sheet → aria-labelledby 強制,per sheet.spec.md:98 */
|
|
60
|
+
title: string
|
|
61
|
+
/** Width(number 或 breakpoint-keyed object);clamp min:240 max:640 */
|
|
62
|
+
width?: number | { md?: number; xl?: number }
|
|
63
|
+
/** Children content */
|
|
64
|
+
children: React.ReactNode
|
|
65
|
+
className?: string
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// ── Context ──────────────────────────────────────────────────────────────────
|
|
69
|
+
|
|
70
|
+
interface AppShellContextValue {
|
|
71
|
+
layout: AppShellLayout
|
|
72
|
+
asideOpen: boolean
|
|
73
|
+
setAsideOpen: (open: boolean) => void
|
|
74
|
+
isMobile: boolean
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const AppShellContext = React.createContext<AppShellContextValue | null>(null)
|
|
78
|
+
|
|
79
|
+
function useAppShell(): AppShellContextValue {
|
|
80
|
+
const ctx = React.useContext(AppShellContext)
|
|
81
|
+
if (!ctx) throw new Error('AppShellAside must be used within <AppShell>')
|
|
82
|
+
return ctx
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Mobile breakpoint:**消費既有 `useIsNarrowViewport`**(`hooks/use-is-narrow-viewport.ts` SSOT,
|
|
86
|
+
// 768px,跟 Sidebar SSOT 同源)— 不發明 local hook,per codex Layer B D2/D4 verdict 避 drift。
|
|
87
|
+
|
|
88
|
+
// xl breakpoint(對齊 Tailwind v4 xl = 1280px,DS-wide consensus)
|
|
89
|
+
const XL_BREAKPOINT_PX = 1280
|
|
90
|
+
|
|
91
|
+
function useIsXl(): boolean {
|
|
92
|
+
const [isXl, setIsXl] = React.useState<boolean>(() => {
|
|
93
|
+
if (typeof window === 'undefined') return false
|
|
94
|
+
return window.matchMedia(`(min-width: ${XL_BREAKPOINT_PX}px)`).matches
|
|
95
|
+
})
|
|
96
|
+
|
|
97
|
+
React.useEffect(() => {
|
|
98
|
+
const mq = window.matchMedia(`(min-width: ${XL_BREAKPOINT_PX}px)`)
|
|
99
|
+
const handler = (e: MediaQueryListEvent) => setIsXl(e.matches)
|
|
100
|
+
mq.addEventListener('change', handler)
|
|
101
|
+
return () => mq.removeEventListener('change', handler)
|
|
102
|
+
}, [])
|
|
103
|
+
|
|
104
|
+
return isXl
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// ── Width resolve(consumer 自傳 + clamp 240-640)──────────────────────────────
|
|
108
|
+
|
|
109
|
+
const ASIDE_WIDTH_MIN = 240
|
|
110
|
+
const ASIDE_WIDTH_MAX = 640
|
|
111
|
+
const ASIDE_WIDTH_DEFAULT = 320
|
|
112
|
+
|
|
113
|
+
function resolveAsideWidth(width: AppShellAsideProps['width'], isXl: boolean): number {
|
|
114
|
+
if (typeof width === 'number') {
|
|
115
|
+
return Math.max(ASIDE_WIDTH_MIN, Math.min(ASIDE_WIDTH_MAX, width))
|
|
116
|
+
}
|
|
117
|
+
if (width && typeof width === 'object') {
|
|
118
|
+
// breakpoint-keyed:xl viewport(≥1280px)用 xl,否則 md
|
|
119
|
+
const v = (isXl ? width.xl ?? width.md : width.md) ?? ASIDE_WIDTH_DEFAULT
|
|
120
|
+
return Math.max(ASIDE_WIDTH_MIN, Math.min(ASIDE_WIDTH_MAX, v))
|
|
121
|
+
}
|
|
122
|
+
return ASIDE_WIDTH_DEFAULT
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
// ── Skip-to-main link(a11y WCAG 2.4.1)───────────────────────────────────────
|
|
126
|
+
|
|
127
|
+
function SkipToMain() {
|
|
128
|
+
return (
|
|
129
|
+
<a
|
|
130
|
+
href="#app-shell-main"
|
|
131
|
+
className={cn(
|
|
132
|
+
'sr-only focus:not-sr-only',
|
|
133
|
+
'focus:fixed focus:top-2 focus:left-2 focus:z-50',
|
|
134
|
+
'focus:px-3 focus:py-2 focus:rounded-md',
|
|
135
|
+
'focus:bg-surface focus:text-foreground focus:shadow-[var(--elevation-200)]',
|
|
136
|
+
'focus:outline-none focus:ring-2 focus:ring-primary'
|
|
137
|
+
)}
|
|
138
|
+
>
|
|
139
|
+
Skip to main content
|
|
140
|
+
</a>
|
|
141
|
+
)
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// ── AppShell root ────────────────────────────────────────────────────────────
|
|
145
|
+
|
|
146
|
+
const AppShell = React.forwardRef<HTMLDivElement, AppShellProps>(
|
|
147
|
+
(
|
|
148
|
+
{
|
|
149
|
+
layout = 'primary-sidebar',
|
|
150
|
+
sidebar,
|
|
151
|
+
header,
|
|
152
|
+
globalHeader,
|
|
153
|
+
aside,
|
|
154
|
+
asideOpen: asideOpenProp,
|
|
155
|
+
onAsideOpenChange,
|
|
156
|
+
children,
|
|
157
|
+
className,
|
|
158
|
+
...props
|
|
159
|
+
},
|
|
160
|
+
ref
|
|
161
|
+
) => {
|
|
162
|
+
const [asideOpenInternal, setAsideOpenInternal] = React.useState(false)
|
|
163
|
+
const isControlled = asideOpenProp !== undefined
|
|
164
|
+
const asideOpen = isControlled ? asideOpenProp : asideOpenInternal
|
|
165
|
+
|
|
166
|
+
const setAsideOpen = React.useCallback(
|
|
167
|
+
(open: boolean) => {
|
|
168
|
+
if (!isControlled) setAsideOpenInternal(open)
|
|
169
|
+
onAsideOpenChange?.(open)
|
|
170
|
+
},
|
|
171
|
+
[isControlled, onAsideOpenChange]
|
|
172
|
+
)
|
|
173
|
+
|
|
174
|
+
const isMobile = useIsNarrowViewport()
|
|
175
|
+
|
|
176
|
+
// ── Keyboard: cmd+. toggle aside ──
|
|
177
|
+
// ⌘B sidebar toggle by Sidebar SSOT(本 component 不重覆 register)
|
|
178
|
+
React.useEffect(() => {
|
|
179
|
+
const onKey = (e: KeyboardEvent) => {
|
|
180
|
+
if (e.key === '.' && (e.metaKey || e.ctrlKey)) {
|
|
181
|
+
e.preventDefault()
|
|
182
|
+
setAsideOpen(!asideOpen)
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
window.addEventListener('keydown', onKey)
|
|
186
|
+
return () => window.removeEventListener('keydown', onKey)
|
|
187
|
+
}, [asideOpen, setAsideOpen])
|
|
188
|
+
|
|
189
|
+
const ctxValue = React.useMemo<AppShellContextValue>(
|
|
190
|
+
() => ({ layout, asideOpen, setAsideOpen, isMobile }),
|
|
191
|
+
[layout, asideOpen, setAsideOpen, isMobile]
|
|
192
|
+
)
|
|
193
|
+
|
|
194
|
+
// ── Layout grid(2 mode)──
|
|
195
|
+
// primary-sidebar:
|
|
196
|
+
// row1: [sidebar (頂天)][main col (header + main)][aside (頂天)]
|
|
197
|
+
// primary-header:
|
|
198
|
+
// row1: [header (橫跨整 viewport, banner role)]
|
|
199
|
+
// row2: [sidebar][main][aside]
|
|
200
|
+
|
|
201
|
+
// AppShellAside 自決 inline vs modal mode(via AppShellContext.isMobile)。
|
|
202
|
+
// AppShell 一律只 render `{aside}` 一次,AppShellAside 內部根據 isMobile 決定 render 形式。
|
|
203
|
+
|
|
204
|
+
if (layout === 'primary-header') {
|
|
205
|
+
// primary-header layout(2026-05-21 v2 — user clarification「primary-header = primary-sidebar + 一條 global header」):
|
|
206
|
+
// 結構:row1 globalHeader(全寬 global bar)/ row2 [sidebar][main col: localHeader + main][aside]
|
|
207
|
+
// - globalHeader = 跨頁 account / workspace switcher / notifications(對齊 GitHub top nav / Slack workspace bar)
|
|
208
|
+
// - header = local page header,**仍存在**(per page actions / breadcrumb,對齊 GitHub repo header / Slack channel header / Gmail email-list toolbar 2-layer 慣例)
|
|
209
|
+
// Consumer 必傳 `<Sidebar viewportInsetTop="var(--chrome-header-height)">` 讓 sidebar 從 globalHeader 下方起算。
|
|
210
|
+
return (
|
|
211
|
+
<AppShellContext.Provider value={ctxValue}>
|
|
212
|
+
<div
|
|
213
|
+
ref={ref}
|
|
214
|
+
className={cn('flex h-svh w-full flex-col overflow-hidden bg-canvas', className)}
|
|
215
|
+
{...props}
|
|
216
|
+
>
|
|
217
|
+
<SkipToMain />
|
|
218
|
+
{/* Row 1:Global header(account / workspace switcher / notifications,橫跨整 viewport)*/}
|
|
219
|
+
{globalHeader && <div className="flex-shrink-0">{globalHeader}</div>}
|
|
220
|
+
{/* Row 2:[sidebar][main col][aside]horizontal row */}
|
|
221
|
+
<div className="flex flex-1 min-h-0 w-full">
|
|
222
|
+
{sidebar}
|
|
223
|
+
{/* Main column:local header + main content(per user model「primary-header = primary-sidebar + global header」)*/}
|
|
224
|
+
<div className="flex flex-col flex-1 min-w-0 min-h-0 overflow-hidden">
|
|
225
|
+
{header && <div className="flex-shrink-0">{header}</div>}
|
|
226
|
+
<main
|
|
227
|
+
id="app-shell-main"
|
|
228
|
+
tabIndex={-1}
|
|
229
|
+
className="flex-1 min-w-0 min-h-0 overflow-y-auto focus:outline-none"
|
|
230
|
+
>
|
|
231
|
+
{children}
|
|
232
|
+
</main>
|
|
233
|
+
</div>
|
|
234
|
+
{aside}
|
|
235
|
+
</div>
|
|
236
|
+
</div>
|
|
237
|
+
</AppShellContext.Provider>
|
|
238
|
+
)
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
// primary-sidebar layout
|
|
242
|
+
return (
|
|
243
|
+
<AppShellContext.Provider value={ctxValue}>
|
|
244
|
+
<div
|
|
245
|
+
ref={ref}
|
|
246
|
+
className={cn('flex h-svh w-full overflow-hidden bg-canvas', className)}
|
|
247
|
+
{...props}
|
|
248
|
+
>
|
|
249
|
+
<SkipToMain />
|
|
250
|
+
{/* Sidebar — 頂天 */}
|
|
251
|
+
{sidebar}
|
|
252
|
+
{/* Main column(header + main 垂直堆)*/}
|
|
253
|
+
<div className="flex flex-col flex-1 min-w-0 min-h-0 overflow-hidden">
|
|
254
|
+
{header && (
|
|
255
|
+
// Header 在 main column 內(main col sibling,非 main descendant)→ 跟 W3C ARIA in HTML
|
|
256
|
+
// banner rule 對照:`<header>` 在 main descendant 才不是 banner,本 ChromeHeader 是 <div>
|
|
257
|
+
// 所以本來就不會被 banner role 計算。仍包 wrap div not <header> 確保不無意觸發 banner。
|
|
258
|
+
<div className="flex-shrink-0">{header}</div>
|
|
259
|
+
)}
|
|
260
|
+
<main
|
|
261
|
+
id="app-shell-main"
|
|
262
|
+
tabIndex={-1}
|
|
263
|
+
className="flex-1 min-h-0 overflow-y-auto focus:outline-none"
|
|
264
|
+
>
|
|
265
|
+
{children}
|
|
266
|
+
</main>
|
|
267
|
+
</div>
|
|
268
|
+
{/* Aside slot — desktop inline OR mobile Sheet,內部自決 */}
|
|
269
|
+
{aside}
|
|
270
|
+
</div>
|
|
271
|
+
</AppShellContext.Provider>
|
|
272
|
+
)
|
|
273
|
+
}
|
|
274
|
+
)
|
|
275
|
+
AppShell.displayName = 'AppShell'
|
|
276
|
+
|
|
277
|
+
// ── AppShellAside sub-component ──────────────────────────────────────────────
|
|
278
|
+
|
|
279
|
+
/**
|
|
280
|
+
* AppShellAside — right panel:standard inline(desktop) vs modal overlay(mobile)。
|
|
281
|
+
*
|
|
282
|
+
* Desktop(viewport ≥ 768px):
|
|
283
|
+
* - Render 直接放 layout grid 右側(asideOpen=true 才 mount,close hide via parent)
|
|
284
|
+
* - 不蓋 mask / background 可操作 / 佔 layout 寬
|
|
285
|
+
* - Vertical extent:primary-sidebar → 頂天立地 / primary-header → header 下方
|
|
286
|
+
*
|
|
287
|
+
* Mobile(viewport < 768px):
|
|
288
|
+
* - Render 走 Sheet primitive(side="right",per sheet.spec.md)
|
|
289
|
+
* - Mask 蓋 / background 不可操作 / 不佔 layout 寬
|
|
290
|
+
* - title 強制(aria-labelledby per sheet.spec.md:98)
|
|
291
|
+
*/
|
|
292
|
+
const AppShellAside = React.forwardRef<HTMLElement, AppShellAsideProps>(
|
|
293
|
+
({ title, width, children, className }, ref) => {
|
|
294
|
+
const { asideOpen, setAsideOpen, isMobile } = useAppShell()
|
|
295
|
+
const isXl = useIsXl()
|
|
296
|
+
const resolvedWidth = resolveAsideWidth(width, isXl)
|
|
297
|
+
|
|
298
|
+
// Shared frame:always-on header(title + close X)+ body(ScrollArea + layoutSpace 規則 1B 父層 padding)
|
|
299
|
+
// 對齊 codex Layer B 2026-05-20「container mode 可變,panel role/content 不該變」+ Notion/Figma right
|
|
300
|
+
// panel 共識(modal vs inline 結構相同,host wrapper 不同)。
|
|
301
|
+
// 2026-05-20 migrate 消費 ChromeHeader primitive(撤回自刻 + 撤回 bg-surface 疊加 — 對齊
|
|
302
|
+
// `header-canonical.spec.md`「6. Background ownership」段「Nested chrome header 透明繼承
|
|
303
|
+
// parent」:aside container 自身已 bg-surface,內 header 不該再畫 bg 避免疊加 drift)。
|
|
304
|
+
const frame = (
|
|
305
|
+
<>
|
|
306
|
+
<ChromeHeader>
|
|
307
|
+
<h2 className="text-body-lg font-medium flex-1 truncate">{title}</h2>
|
|
308
|
+
<Button
|
|
309
|
+
iconOnly
|
|
310
|
+
dismiss
|
|
311
|
+
size="sm"
|
|
312
|
+
startIcon={XIcon}
|
|
313
|
+
aria-label="關閉"
|
|
314
|
+
onClick={() => setAsideOpen(false)}
|
|
315
|
+
/>
|
|
316
|
+
</ChromeHeader>
|
|
317
|
+
<ScrollArea className="flex-1 min-h-0">
|
|
318
|
+
{children}
|
|
319
|
+
</ScrollArea>
|
|
320
|
+
</>
|
|
321
|
+
)
|
|
322
|
+
|
|
323
|
+
// Modal mode(mobile)— Sheet from right
|
|
324
|
+
if (isMobile) {
|
|
325
|
+
return (
|
|
326
|
+
<Sheet open={asideOpen} onOpenChange={setAsideOpen}>
|
|
327
|
+
<SheetContent
|
|
328
|
+
side="right"
|
|
329
|
+
className="w-[min(90vw,var(--app-shell-aside-modal-width))] flex flex-col p-0 [&>button]:hidden"
|
|
330
|
+
style={{ ['--app-shell-aside-modal-width' as string]: `${resolvedWidth}px` }}
|
|
331
|
+
>
|
|
332
|
+
{frame}
|
|
333
|
+
</SheetContent>
|
|
334
|
+
</Sheet>
|
|
335
|
+
)
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
// Standard inline mode(desktop)
|
|
339
|
+
if (!asideOpen) return null
|
|
340
|
+
|
|
341
|
+
return (
|
|
342
|
+
<aside
|
|
343
|
+
ref={ref}
|
|
344
|
+
aria-label={title}
|
|
345
|
+
className={cn(
|
|
346
|
+
'flex flex-col h-full min-h-0 overflow-hidden',
|
|
347
|
+
'bg-surface border-l border-divider',
|
|
348
|
+
className
|
|
349
|
+
)}
|
|
350
|
+
style={{ width: resolvedWidth }}
|
|
351
|
+
>
|
|
352
|
+
{frame}
|
|
353
|
+
</aside>
|
|
354
|
+
)
|
|
355
|
+
}
|
|
356
|
+
)
|
|
357
|
+
AppShellAside.displayName = 'AppShellAside'
|
|
358
|
+
|
|
359
|
+
// ── Exports ──────────────────────────────────────────────────────────────────
|
|
360
|
+
|
|
361
|
+
// code-quality-allow: dead-export useAppShell — public compound API hook(consumer 可自拼 custom aside layout,
|
|
362
|
+
// 對齊 Radix `useDialogContext` / MUI `useFormControl` 慣例)。內部 AppShellAside 已消費(L294),
|
|
363
|
+
// audit script 抓「無 cross-file import」是 false positive(2026-05-21 D2 codify)。
|
|
364
|
+
export { AppShell, AppShellAside, useAppShell }
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
// @benchmark-unverified-blanket: file-level retraction per M22 (d) — claims herein not individually URL-cited; treat as unverified visual/usage rumor unless retrofit per-claim. Hook escape preserved.
|
|
2
|
+
import * as React from 'react'
|
|
3
|
+
import * as AspectRatioPrimitive from '@radix-ui/react-aspect-ratio'
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* AspectRatio — 固定長寬比容器(Radix AspectRatio primitive 薄包裝)
|
|
7
|
+
*
|
|
8
|
+
* 世界級對照:shadcn `AspectRatio` / Ant 無獨立元件(CSS 方案)/ Material 無
|
|
9
|
+
*
|
|
10
|
+
* ── 為什麼需要 ──
|
|
11
|
+
* CSS `aspect-ratio` 屬性雖然現代瀏覽器都支援,但 Radix primitive 提供 SSR-safe
|
|
12
|
+
* padding-bottom 方案 + consistent API,避免邊緣 bug(image 未載入時容器坍塌 /
|
|
13
|
+
* content-fit 差異)。保 safe + 一致視覺。
|
|
14
|
+
*
|
|
15
|
+
* ── 標準 ratio(DS 慣例) ──
|
|
16
|
+
* 16/9 — 寬螢幕影片、onboarding feature tour 截圖(Coachmark media 預設)
|
|
17
|
+
* 4/3 — 老電視 / 照片基本 ratio、產品 thumbnail
|
|
18
|
+
* 1/1 — Avatar / 方形貼文預覽 / icon preview
|
|
19
|
+
* 3/4 — 直式照片(人物 portrait)
|
|
20
|
+
* 21/9 — Ultra-wide banner(hero section)
|
|
21
|
+
*
|
|
22
|
+
* Consumer 傳 `ratio={n/m}` 數字計算(如 16/9 = 1.7777)。
|
|
23
|
+
*
|
|
24
|
+
* ── 常見消費者 ──
|
|
25
|
+
* Coachmark media / Carousel item image / Card thumbnail(未來)/ Chart preview
|
|
26
|
+
*/
|
|
27
|
+
|
|
28
|
+
export type AspectRatioProps = React.ComponentPropsWithoutRef<typeof AspectRatioPrimitive.Root>
|
|
29
|
+
|
|
30
|
+
// shadcn canonical:顯式 forwardRef + displayName(雖 Radix primitive 已 forwardRef,
|
|
31
|
+
// 此 wrapper 確保本 DS 每個 named export 在 Inspector 顯示正確 displayName,
|
|
32
|
+
// 且 props passthrough + ref 行為在 code 層面明確可見)
|
|
33
|
+
const AspectRatio = React.forwardRef<
|
|
34
|
+
React.ElementRef<typeof AspectRatioPrimitive.Root>,
|
|
35
|
+
AspectRatioProps
|
|
36
|
+
>((props, ref) => <AspectRatioPrimitive.Root ref={ref} {...props} />)
|
|
37
|
+
AspectRatio.displayName = 'AspectRatio'
|
|
38
|
+
|
|
39
|
+
// Story auto-compile metadata — Phase 1 mechanical migration(2026-04-24)
|
|
40
|
+
// Phase 2 fill needed: purpose descriptions + when rationale + world-class refs
|
|
41
|
+
export const aspectRatioMeta = {
|
|
42
|
+
component: 'AspectRatio',
|
|
43
|
+
family: null, // non-family composite / overlay / layout
|
|
44
|
+
variants: {
|
|
45
|
+
|
|
46
|
+
},
|
|
47
|
+
sizes: {
|
|
48
|
+
|
|
49
|
+
},
|
|
50
|
+
states: ['default', 'hover', 'active', 'focus-visible', 'disabled'],
|
|
51
|
+
tokens: {
|
|
52
|
+
bg: [],
|
|
53
|
+
fg: [],
|
|
54
|
+
ring: [],
|
|
55
|
+
},
|
|
56
|
+
} as const
|
|
57
|
+
|
|
58
|
+
export { AspectRatio }
|