@qijenchen/design-system 0.1.0-beta.3

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.
Files changed (119) hide show
  1. package/package.json +93 -0
  2. package/src/README.md +32 -0
  3. package/src/components/Accordion/accordion.tsx +104 -0
  4. package/src/components/Alert/alert.tsx +188 -0
  5. package/src/components/AppShell/_demo-helpers.tsx +198 -0
  6. package/src/components/AppShell/app-shell.tsx +364 -0
  7. package/src/components/AspectRatio/aspect-ratio.tsx +58 -0
  8. package/src/components/Avatar/avatar.tsx +368 -0
  9. package/src/components/Badge/badge.tsx +104 -0
  10. package/src/components/Breadcrumb/breadcrumb.tsx +609 -0
  11. package/src/components/BulkActionBar/bulk-action-bar.tsx +156 -0
  12. package/src/components/Button/button-group.tsx +96 -0
  13. package/src/components/Button/button.tsx +539 -0
  14. package/src/components/Calendar/calendar.tsx +411 -0
  15. package/src/components/Carousel/carousel.tsx +371 -0
  16. package/src/components/Chart/chart.tsx +376 -0
  17. package/src/components/Checkbox/checkbox-group.tsx +94 -0
  18. package/src/components/Checkbox/checkbox.tsx +237 -0
  19. package/src/components/Chip/chip.tsx +359 -0
  20. package/src/components/CircularProgress/circular-progress.tsx +204 -0
  21. package/src/components/Coachmark/coachmark.tsx +255 -0
  22. package/src/components/Combobox/combobox.tsx +826 -0
  23. package/src/components/Command/command.tsx +187 -0
  24. package/src/components/DataTable/active-editor-controller.ts +72 -0
  25. package/src/components/DataTable/cell-registry.tsx +520 -0
  26. package/src/components/DataTable/column-types.ts +180 -0
  27. package/src/components/DataTable/data-table-column-visibility-panel.tsx +261 -0
  28. package/src/components/DataTable/data-table-filter-panel.tsx +813 -0
  29. package/src/components/DataTable/data-table-interaction-layer.tsx +483 -0
  30. package/src/components/DataTable/data-table-sort-manager.tsx +210 -0
  31. package/src/components/DataTable/data-table.css +165 -0
  32. package/src/components/DataTable/data-table.tsx +2924 -0
  33. package/src/components/DataTable/filter-operators.ts +225 -0
  34. package/src/components/DataTable/filter-tree.ts +313 -0
  35. package/src/components/DataTable/lib/column-meta.ts +79 -0
  36. package/src/components/DateGrid/date-grid.tsx +209 -0
  37. package/src/components/DatePicker/date-picker.tsx +1114 -0
  38. package/src/components/DescriptionList/description-list.tsx +141 -0
  39. package/src/components/Dialog/dialog.tsx +267 -0
  40. package/src/components/DropdownMenu/dropdown-menu.tsx +475 -0
  41. package/src/components/Empty/empty.tsx +108 -0
  42. package/src/components/Field/field-context.ts +136 -0
  43. package/src/components/Field/field-types.ts +52 -0
  44. package/src/components/Field/field-wrapper.tsx +348 -0
  45. package/src/components/Field/field.tsx +535 -0
  46. package/src/components/FieldControlGroup/field-control-group.tsx +136 -0
  47. package/src/components/FileItem/file-item.tsx +322 -0
  48. package/src/components/FileUpload/file-upload.tsx +326 -0
  49. package/src/components/FileViewer/file-viewer-types.ts +76 -0
  50. package/src/components/FileViewer/file-viewer.tsx +1065 -0
  51. package/src/components/FileViewer/image-renderer.tsx +256 -0
  52. package/src/components/HoverCard/hover-card.tsx +79 -0
  53. package/src/components/Input/input.tsx +233 -0
  54. package/src/components/LinkInput/link-input.tsx +304 -0
  55. package/src/components/Menu/menu-item.tsx +334 -0
  56. package/src/components/NameCard/name-card.tsx +319 -0
  57. package/src/components/Notice/notice.tsx +196 -0
  58. package/src/components/NumberInput/number-input.tsx +203 -0
  59. package/src/components/OverflowIndicator/overflow-indicator.tsx +156 -0
  60. package/src/components/PeoplePicker/avatar-stack-overflow.ts +100 -0
  61. package/src/components/PeoplePicker/people-picker-helpers.ts +76 -0
  62. package/src/components/PeoplePicker/people-picker.tsx +455 -0
  63. package/src/components/PeoplePicker/person-display.tsx +358 -0
  64. package/src/components/Popover/popover.tsx +183 -0
  65. package/src/components/ProgressBar/progress-bar.tsx +157 -0
  66. package/src/components/README.md +58 -0
  67. package/src/components/RadioGroup/radio-group.tsx +261 -0
  68. package/src/components/Rating/rating.tsx +295 -0
  69. package/src/components/ScrollArea/scroll-area.tsx +110 -0
  70. package/src/components/SegmentedControl/segmented-control.tsx +304 -0
  71. package/src/components/Select/select.tsx +658 -0
  72. package/src/components/SelectMenu/select-menu.tsx +430 -0
  73. package/src/components/SelectionControl/selection-item.tsx +261 -0
  74. package/src/components/Separator/separator.tsx +48 -0
  75. package/src/components/Sheet/sheet.tsx +240 -0
  76. package/src/components/Sidebar/sidebar.tsx +1280 -0
  77. package/src/components/Skeleton/skeleton.tsx +35 -0
  78. package/src/components/Slider/slider.tsx +158 -0
  79. package/src/components/Steps/steps.tsx +850 -0
  80. package/src/components/Switch/switch.tsx +285 -0
  81. package/src/components/Tabs/tabs.tsx +515 -0
  82. package/src/components/Tag/tag.tsx +246 -0
  83. package/src/components/Textarea/textarea.tsx +280 -0
  84. package/src/components/TimePicker/time-columns.tsx +260 -0
  85. package/src/components/TimePicker/time-picker.tsx +419 -0
  86. package/src/components/Toast/toast.tsx +129 -0
  87. package/src/components/Tooltip/tooltip.tsx +68 -0
  88. package/src/components/TreeView/tree-view.tsx +1031 -0
  89. package/src/hooks/use-controllable.ts +40 -0
  90. package/src/hooks/use-is-narrow-viewport.ts +19 -0
  91. package/src/hooks/use-is-touch-device.ts +21 -0
  92. package/src/hooks/use-overflow-items.ts +256 -0
  93. package/src/index.ts +85 -0
  94. package/src/lib/README.md +82 -0
  95. package/src/lib/drag-visual.ts +272 -0
  96. package/src/lib/i18n/README.md +60 -0
  97. package/src/lib/i18n/i18n-context.tsx +129 -0
  98. package/src/lib/multi-select-ordering.ts +61 -0
  99. package/src/lib/utils.ts +93 -0
  100. package/src/patterns/README.md +67 -0
  101. package/src/patterns/element-anatomy/item-anatomy.tsx +744 -0
  102. package/src/patterns/header-canonical/chrome-header.tsx +175 -0
  103. package/src/patterns/header-canonical/header-canonical.css +27 -0
  104. package/src/patterns/horizontal-overflow/horizontal-overflow.tsx +217 -0
  105. package/src/patterns/overlay-surface/overlay-surface.tsx +191 -0
  106. package/src/patterns/resize-handle/resize-handle.tsx +188 -0
  107. package/src/stories-helpers/anatomy/anatomy-utils.tsx +64 -0
  108. package/src/tokens/README.md +53 -0
  109. package/src/tokens/color/primitives.css +429 -0
  110. package/src/tokens/color/semantic.css +539 -0
  111. package/src/tokens/elevation/overlay-geometry.ts +13 -0
  112. package/src/tokens/layoutSpace/layoutSpace.css +36 -0
  113. package/src/tokens/motion/motion.css +30 -0
  114. package/src/tokens/motion/motion.ts +17 -0
  115. package/src/tokens/opacity/opacity.css +23 -0
  116. package/src/tokens/radius/radius.css +19 -0
  117. package/src/tokens/typography/typography.css +118 -0
  118. package/src/tokens/uiSize/icon-size.ts +52 -0
  119. package/src/tokens/uiSize/uiSize.css +125 -0
@@ -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 }