@zendir/ui 0.2.20 → 0.3.0
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/CHANGELOG.md +192 -1
- package/README.md +70 -28
- package/dist/index.d.ts +1 -1
- package/dist/index.js +57 -41
- package/dist/index.js.map +1 -1
- package/dist/react/3d/CesiumCaptureSource.d.ts +119 -0
- package/dist/react/3d/CesiumCaptureSource.js +307 -0
- package/dist/react/3d/CesiumCaptureSource.js.map +1 -0
- package/dist/react/3d/ZenSpace3D.js +1253 -0
- package/dist/react/3d/ZenSpace3D.js.map +1 -0
- package/dist/react/3d/ZenSpace3DCesium.js +579 -0
- package/dist/react/3d/ZenSpace3DCesium.js.map +1 -0
- package/dist/react/3d/ZenSpace3DTypes.d.ts +28 -1
- package/dist/react/3d/ZenSpace3DUtils.d.ts +17 -173
- package/dist/react/3d/ZenSpace3DUtils.js +28 -0
- package/dist/react/3d/ZenSpace3DUtils.js.map +1 -0
- package/dist/react/3d/capturePngAnalysis.d.ts +16 -0
- package/dist/react/3d/index.d.ts +10 -12
- package/dist/react/3d/threeLoader.js +18 -0
- package/dist/react/3d/threeLoader.js.map +1 -0
- package/dist/react/astro/MonitoringIcon.js +1 -1
- package/dist/react/astro/MonitoringIcon.js.map +1 -1
- package/dist/react/astro/SimulationControls.js +2 -2
- package/dist/react/astro/SimulationControls.js.map +1 -1
- package/dist/react/astro/UnifiedTimeline.js +4 -4
- package/dist/react/astro/UnifiedTimeline.js.map +1 -1
- package/dist/react/charts/GroundTrackMap.d.ts +2 -15
- package/dist/react/charts/GroundTrackMap.js +1 -1
- package/dist/react/charts/GroundTrackMap.js.map +1 -1
- package/dist/react/charts/unified/AstroChart.js +34 -13
- package/dist/react/charts/unified/AstroChart.js.map +1 -1
- package/dist/react/chatgpt/AppCard.d.ts +0 -4
- package/dist/react/chatgpt/index.d.ts +0 -19
- package/dist/react/context/SpatialSelectionContext.d.ts +40 -0
- package/dist/react/context/SpatialSelectionContext.js +10 -0
- package/dist/react/context/SpatialSelectionContext.js.map +1 -0
- package/dist/react/context/index.d.ts +2 -0
- package/dist/react/core/{DataTable.d.ts → data/DataTable.d.ts} +1 -1
- package/dist/react/core/{DataTable.js → data/DataTable.js} +4 -4
- package/dist/react/core/data/DataTable.js.map +1 -0
- package/dist/react/core/{DataValue.d.ts → data/DataValue.d.ts} +2 -2
- package/dist/react/core/{DataValue.js → data/DataValue.js} +2 -2
- package/dist/react/core/data/DataValue.js.map +1 -0
- package/dist/react/core/{propertyConfig.d.ts → data/propertyConfig.d.ts} +2 -2
- package/dist/react/core/data/propertyConfig.js.map +1 -0
- package/dist/react/core/{AstroIcon.js → display/AstroIcon.js} +1 -1
- package/dist/react/core/display/AstroIcon.js.map +1 -0
- package/dist/react/core/{Badge.d.ts → display/Badge.d.ts} +1 -1
- package/dist/react/core/{Badge.js → display/Badge.js} +2 -2
- package/dist/react/core/display/Badge.js.map +1 -0
- package/dist/react/core/{CardHeader.d.ts → display/CardHeader.d.ts} +1 -1
- package/dist/react/core/{CardHeader.js → display/CardHeader.js} +2 -2
- package/dist/react/core/display/CardHeader.js.map +1 -0
- package/dist/react/core/{Container.d.ts → display/Container.d.ts} +1 -1
- package/dist/react/core/{Container.js → display/Container.js} +3 -3
- package/dist/react/core/display/Container.js.map +1 -0
- package/dist/react/core/{CopyButton.js → display/CopyButton.js} +1 -1
- package/dist/react/core/display/CopyButton.js.map +1 -0
- package/dist/react/core/{GlassCard.d.ts → display/GlassCard.d.ts} +1 -1
- package/dist/react/core/{GlassCard.js → display/GlassCard.js} +2 -2
- package/dist/react/core/display/GlassCard.js.map +1 -0
- package/dist/react/core/{HeaderIconWithStatus.d.ts → display/HeaderIconWithStatus.d.ts} +1 -1
- package/dist/react/core/{HeaderIconWithStatus.js → display/HeaderIconWithStatus.js} +1 -1
- package/dist/react/core/display/HeaderIconWithStatus.js.map +1 -0
- package/dist/react/core/{Icon.d.ts → display/Icon.d.ts} +1 -1
- package/dist/react/core/{Icon.js → display/Icon.js} +1 -1
- package/dist/react/core/display/Icon.js.map +1 -0
- package/dist/react/core/{Typography.d.ts → display/Typography.d.ts} +13 -4
- package/dist/react/core/{Typography.js → display/Typography.js} +1 -1
- package/dist/react/core/display/Typography.js.map +1 -0
- package/dist/react/core/{ConfirmDialog.js → feedback/ConfirmDialog.js} +1 -1
- package/dist/react/core/feedback/ConfirmDialog.js.map +1 -0
- package/dist/react/core/{Dialog.js → feedback/Dialog.js} +2 -2
- package/dist/react/core/feedback/Dialog.js.map +1 -0
- package/dist/react/core/{Toast.js → feedback/Toast.js} +3 -3
- package/dist/react/core/feedback/Toast.js.map +1 -0
- package/dist/react/core/index.d.ts +85 -83
- package/dist/react/core/{Button.js → inputs/Button.js} +2 -2
- package/dist/react/core/inputs/Button.js.map +1 -0
- package/dist/react/core/{Checkbox.js → inputs/Checkbox.js} +2 -2
- package/dist/react/core/inputs/Checkbox.js.map +1 -0
- package/dist/react/core/{Input.d.ts → inputs/Input.d.ts} +1 -1
- package/dist/react/core/{Input.js → inputs/Input.js} +3 -3
- package/dist/react/core/inputs/Input.js.map +1 -0
- package/dist/react/core/{LimitsBar.js → inputs/LimitsBar.js} +1 -1
- package/dist/react/core/inputs/LimitsBar.js.map +1 -0
- package/dist/react/core/{NumberInput.d.ts → inputs/NumberInput.d.ts} +2 -2
- package/dist/react/core/{NumberInput.js → inputs/NumberInput.js} +3 -3
- package/dist/react/core/inputs/NumberInput.js.map +1 -0
- package/dist/react/core/{PinInput.js → inputs/PinInput.js} +2 -2
- package/dist/react/core/inputs/PinInput.js.map +1 -0
- package/dist/react/core/{Select.js → inputs/Select.js} +3 -3
- package/dist/react/core/inputs/Select.js.map +1 -0
- package/dist/react/core/{Toggle.js → inputs/Toggle.js} +2 -2
- package/dist/react/core/inputs/Toggle.js.map +1 -0
- package/dist/react/core/{AppBar.d.ts → navigation/AppBar.d.ts} +1 -1
- package/dist/react/core/{AppBar.js → navigation/AppBar.js} +7 -7
- package/dist/react/core/navigation/AppBar.js.map +1 -0
- package/dist/react/core/{Pagination.js → navigation/Pagination.js} +2 -2
- package/dist/react/core/navigation/Pagination.js.map +1 -0
- package/dist/react/core/{SideNav.d.ts → navigation/SideNav.d.ts} +1 -1
- package/dist/react/core/{SideNav.js → navigation/SideNav.js} +8 -9
- package/dist/react/core/navigation/SideNav.js.map +1 -0
- package/dist/react/core/{Tabs.js → navigation/Tabs.js} +2 -2
- package/dist/react/core/navigation/Tabs.js.map +1 -0
- package/dist/react/core/{Popover.js → overlays/Popover.js} +1 -1
- package/dist/react/core/overlays/Popover.js.map +1 -0
- package/dist/react/core/{SidePanel.js → overlays/SidePanel.js} +7 -7
- package/dist/react/core/overlays/SidePanel.js.map +1 -0
- package/dist/react/core/{Tooltip.js → overlays/Tooltip.js} +2 -2
- package/dist/react/core/overlays/Tooltip.js.map +1 -0
- package/dist/react/core/{ActivityPlanner.js → widgets/ActivityPlanner.js} +1 -1
- package/dist/react/core/widgets/ActivityPlanner.js.map +1 -0
- package/dist/react/core/widgets/Capture.d.ts +140 -0
- package/dist/react/core/widgets/Capture.js +804 -0
- package/dist/react/core/widgets/Capture.js.map +1 -0
- package/dist/react/core/{ChatPanel.d.ts → widgets/ChatPanel.d.ts} +1 -1
- package/dist/react/core/{ChatPanel.js → widgets/ChatPanel.js} +5 -4
- package/dist/react/core/widgets/ChatPanel.js.map +1 -0
- package/dist/react/core/{ColorPickerPanel.d.ts → widgets/ColorPickerPanel.d.ts} +1 -1
- package/dist/react/core/{ColorPickerPanel.js → widgets/ColorPickerPanel.js} +3 -3
- package/dist/react/core/widgets/ColorPickerPanel.js.map +1 -0
- package/dist/react/core/{CommandBuilder.js → widgets/CommandBuilder.js} +1 -1
- package/dist/react/core/widgets/CommandBuilder.js.map +1 -0
- package/dist/react/core/{ConnectionForm.d.ts → widgets/ConnectionForm.d.ts} +1 -1
- package/dist/react/core/{ConnectionForm.js → widgets/ConnectionForm.js} +2 -2
- package/dist/react/core/widgets/ConnectionForm.js.map +1 -0
- package/dist/react/core/{FileExplorer.js → widgets/FileExplorer.js} +2 -2
- package/dist/react/core/widgets/FileExplorer.js.map +1 -0
- package/dist/react/core/{HexViewer.js → widgets/HexViewer.js} +1 -1
- package/dist/react/core/widgets/HexViewer.js.map +1 -0
- package/dist/react/core/{ImageGallery.d.ts → widgets/ImageGallery.d.ts} +1 -1
- package/dist/react/core/{ImageGallery.js → widgets/ImageGallery.js} +3 -3
- package/dist/react/core/widgets/ImageGallery.js.map +1 -0
- package/dist/react/core/{LogViewer.d.ts → widgets/LogViewer.d.ts} +13 -3
- package/dist/react/core/{LogViewer.js → widgets/LogViewer.js} +28 -8
- package/dist/react/core/widgets/LogViewer.js.map +1 -0
- package/dist/react/core/{MessageStream.d.ts → widgets/MessageStream.d.ts} +2 -2
- package/dist/react/core/{MessageStream.js → widgets/MessageStream.js} +4 -4
- package/dist/react/core/widgets/MessageStream.js.map +1 -0
- package/dist/react/core/{MissionCalendar.js → widgets/MissionCalendar.js} +2 -2
- package/dist/react/core/widgets/MissionCalendar.js.map +1 -0
- package/dist/react/core/{PacketViewer.js → widgets/PacketViewer.js} +1 -1
- package/dist/react/core/widgets/PacketViewer.js.map +1 -0
- package/dist/react/core/widgets/capture-placeholder.png.js +5 -0
- package/dist/react/core/widgets/capture-placeholder.png.js.map +1 -0
- package/dist/react/hooks/index.d.ts +9 -11
- package/dist/react/hooks/useAccessWindows.d.ts +15 -19
- package/dist/react/hooks/useGroundTrackHistory.d.ts +34 -0
- package/dist/react/hooks/useSimulationScene.d.ts +141 -0
- package/dist/react/hooks/useSimulationScene.js +401 -0
- package/dist/react/hooks/useSimulationScene.js.map +1 -0
- package/dist/react/hooks/useZendirSession.d.ts +44 -69
- package/dist/react/index.d.ts +10 -5
- package/dist/react/panels/LayerControlPanel.d.ts +54 -0
- package/dist/react/panels/LayerControlPanel.js +184 -0
- package/dist/react/panels/LayerControlPanel.js.map +1 -0
- package/dist/react/panels/ObjectInventoryPanel.d.ts +57 -0
- package/dist/react/panels/ObjectInventoryPanel.js +261 -0
- package/dist/react/panels/ObjectInventoryPanel.js.map +1 -0
- package/dist/react/panels/index.d.ts +15 -0
- package/dist/react/theme/ThemeProvider.d.ts +2 -0
- package/dist/react/theme/ThemeProvider.js +50 -72
- package/dist/react/theme/ThemeProvider.js.map +1 -1
- package/dist/react/types.d.ts +32 -3
- package/dist/react/types.js.map +1 -1
- package/dist/react.js +57 -41
- package/dist/react.js.map +1 -1
- package/dist/shaders/atmosphere.frag.js +5 -0
- package/dist/shaders/atmosphere.frag.js.map +1 -0
- package/dist/shaders/atmosphere.vert.js +5 -0
- package/dist/shaders/atmosphere.vert.js.map +1 -0
- package/dist/shaders/stars.frag.js +5 -0
- package/dist/shaders/stars.frag.js.map +1 -0
- package/dist/shaders/stars.vert.js +5 -0
- package/dist/shaders/stars.vert.js.map +1 -0
- package/dist/style.css +6 -4
- package/dist/tokens/css-vars.d.ts +91 -0
- package/dist/tokens/css-vars.js +228 -0
- package/dist/tokens/css-vars.js.map +1 -0
- package/dist/tokens/index.d.ts +71 -18
- package/dist/tokens/index.js +206 -97
- package/dist/tokens/index.js.map +1 -1
- package/dist/tokens/tokens.css +50 -50
- package/package.json +27 -22
- package/sdk-stub.js +10 -5
- package/dist/react/3d/EarthViewer.d.ts +0 -46
- package/dist/react/3d/SolarSystemViewer.d.ts +0 -43
- package/dist/react/chatgpt/ChatGPTCard.d.ts +0 -6
- package/dist/react/core/ActivityPlanner.js.map +0 -1
- package/dist/react/core/AppBar.js.map +0 -1
- package/dist/react/core/AstroIcon.js.map +0 -1
- package/dist/react/core/Badge.js.map +0 -1
- package/dist/react/core/Button.js.map +0 -1
- package/dist/react/core/CardHeader.js.map +0 -1
- package/dist/react/core/ChatPanel.js.map +0 -1
- package/dist/react/core/Checkbox.js.map +0 -1
- package/dist/react/core/ColorPickerPanel.js.map +0 -1
- package/dist/react/core/CommandBuilder.js.map +0 -1
- package/dist/react/core/ConfirmDialog.js.map +0 -1
- package/dist/react/core/ConnectionForm.js.map +0 -1
- package/dist/react/core/Container.js.map +0 -1
- package/dist/react/core/CopyButton.js.map +0 -1
- package/dist/react/core/DataTable.js.map +0 -1
- package/dist/react/core/DataValue.js.map +0 -1
- package/dist/react/core/Dialog.js.map +0 -1
- package/dist/react/core/FileExplorer.js.map +0 -1
- package/dist/react/core/GlassCard.js.map +0 -1
- package/dist/react/core/HeaderIconWithStatus.js.map +0 -1
- package/dist/react/core/HexViewer.js.map +0 -1
- package/dist/react/core/Icon.js.map +0 -1
- package/dist/react/core/ImageGallery.js.map +0 -1
- package/dist/react/core/Input.js.map +0 -1
- package/dist/react/core/LimitsBar.js.map +0 -1
- package/dist/react/core/LogViewer.js.map +0 -1
- package/dist/react/core/MessageStream.js.map +0 -1
- package/dist/react/core/MissionCalendar.js.map +0 -1
- package/dist/react/core/NumberInput.js.map +0 -1
- package/dist/react/core/PacketViewer.js.map +0 -1
- package/dist/react/core/Pagination.js.map +0 -1
- package/dist/react/core/PinInput.js.map +0 -1
- package/dist/react/core/Popover.js.map +0 -1
- package/dist/react/core/Select.js.map +0 -1
- package/dist/react/core/SideNav.js.map +0 -1
- package/dist/react/core/SidePanel.js.map +0 -1
- package/dist/react/core/Tabs.js.map +0 -1
- package/dist/react/core/Toast.js.map +0 -1
- package/dist/react/core/Toggle.js.map +0 -1
- package/dist/react/core/Tooltip.js.map +0 -1
- package/dist/react/core/Typography.js.map +0 -1
- package/dist/react/core/propertyConfig.js.map +0 -1
- package/dist/react/hooks/useSimulationTime.d.ts +0 -61
- package/dist/react/hooks/useSpacecraftPosition.d.ts +0 -50
- package/dist/react/hooks/useTelemetry.d.ts +0 -55
- package/dist/types.d.ts +0 -1
- package/dist/types.js +0 -2
- package/dist/types.js.map +0 -1
- /package/dist/react/core/{propertyConfig.js → data/propertyConfig.js} +0 -0
- /package/dist/react/core/{AstroIcon.d.ts → display/AstroIcon.d.ts} +0 -0
- /package/dist/react/core/{CopyButton.d.ts → display/CopyButton.d.ts} +0 -0
- /package/dist/react/core/{ConfirmDialog.d.ts → feedback/ConfirmDialog.d.ts} +0 -0
- /package/dist/react/core/{Dialog.d.ts → feedback/Dialog.d.ts} +0 -0
- /package/dist/react/core/{Toast.d.ts → feedback/Toast.d.ts} +0 -0
- /package/dist/react/core/{Button.d.ts → inputs/Button.d.ts} +0 -0
- /package/dist/react/core/{Checkbox.d.ts → inputs/Checkbox.d.ts} +0 -0
- /package/dist/react/core/{LimitsBar.d.ts → inputs/LimitsBar.d.ts} +0 -0
- /package/dist/react/core/{PinInput.d.ts → inputs/PinInput.d.ts} +0 -0
- /package/dist/react/core/{Select.d.ts → inputs/Select.d.ts} +0 -0
- /package/dist/react/core/{Toggle.d.ts → inputs/Toggle.d.ts} +0 -0
- /package/dist/react/core/{Pagination.d.ts → navigation/Pagination.d.ts} +0 -0
- /package/dist/react/core/{Tabs.d.ts → navigation/Tabs.d.ts} +0 -0
- /package/dist/react/core/{Popover.d.ts → overlays/Popover.d.ts} +0 -0
- /package/dist/react/core/{SidePanel.d.ts → overlays/SidePanel.d.ts} +0 -0
- /package/dist/react/core/{Tooltip.d.ts → overlays/Tooltip.d.ts} +0 -0
- /package/dist/react/core/{ActivityPlanner.d.ts → widgets/ActivityPlanner.d.ts} +0 -0
- /package/dist/react/core/{CommandBuilder.d.ts → widgets/CommandBuilder.d.ts} +0 -0
- /package/dist/react/core/{FileExplorer.d.ts → widgets/FileExplorer.d.ts} +0 -0
- /package/dist/react/core/{HexViewer.d.ts → widgets/HexViewer.d.ts} +0 -0
- /package/dist/react/core/{MissionCalendar.d.ts → widgets/MissionCalendar.d.ts} +0 -0
- /package/dist/react/core/{PacketViewer.d.ts → widgets/PacketViewer.d.ts} +0 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ThemeProvider.js","sources":["../../../src/react/theme/ThemeProvider.tsx"],"sourcesContent":["/**\n * @zendir/ui - Theme Provider\n * \n * Enterprise-grade theming system supporting Astro UX Design System and Zendir themes.\n * Full Tier 3 compliance with official Astro tokens (dark + light themes).\n * \n * Features:\n * - Multiple theme variants (Zen Hybrid default, astro, purple-hue, transparent)\n * - Light and dark modes\n * - Animation tokens with reduced-motion support\n * - Focus ring styles for accessibility\n * - CSS variable injection\n * - Local storage persistence\n */\n\nimport React, { createContext, useContext, useState, useEffect, useMemo, ReactNode } from 'react';\n\nfunction adjustHexBrightness(hex: string, factor: number): string {\n const normalized = hex.replace('#', '').trim();\n const expanded = normalized.length === 3\n ? normalized.split('').map(ch => ch + ch).join('')\n : normalized;\n\n if (!/^[0-9a-fA-F]{6}$/.test(expanded)) {\n return hex;\n }\n\n const clamp = (value: number) => Math.max(0, Math.min(255, Math.round(value)));\n const r = clamp(parseInt(expanded.slice(0, 2), 16) * factor);\n const g = clamp(parseInt(expanded.slice(2, 4), 16) * factor);\n const b = clamp(parseInt(expanded.slice(4, 6), 16) * factor);\n\n const toHex = (value: number) => value.toString(16).padStart(2, '0');\n return `#${toHex(r)}${toHex(g)}${toHex(b)}`;\n}\n\nexport type ThemeVariant = 'astro' | 'purple-hue' | 'hybrid' | 'transparent' | 'transparent-bold' | 'transparent-minimal';\nexport type ThemeMode = 'light' | 'dark';\n\nexport interface ThemeColors {\n background: {\n base: string;\n surface: string;\n elevated: string;\n overlay: string;\n };\n border: {\n default: string;\n muted: string;\n focus: string;\n /** Transparent theme: thin faded top-to-bottom border (box-shadow, use with border: 'none') */\n fadedBoxShadow?: string;\n /** Card border style (computed from theme – solid or transparent faded). Use ...tokens.colors.border.cardStyle */\n cardStyle?: React.CSSProperties;\n /** Card border for dashed/empty state. Use ...tokens.colors.border.cardStyleDashed */\n cardStyleDashed?: React.CSSProperties;\n /** Card border using default color (e.g. NavBall, EclipseTimer). Use ...tokens.colors.border.cardStyleDefault */\n cardStyleDefault?: React.CSSProperties;\n };\n text: {\n primary: string;\n secondary: string;\n tertiary: string;\n /** Muted/placeholder text (alias for tertiary when not set) */\n muted?: string;\n inverse: string;\n };\n status: {\n normal: string;\n standby: string;\n caution: string;\n serious: string;\n critical: string;\n off: string;\n };\n semantic: {\n success: string;\n warning: string;\n error: string;\n info: string;\n };\n accent: {\n primary: string;\n secondary: string;\n tertiary: string;\n };\n interactive: {\n default: string;\n hover: string;\n active: string;\n disabled: string;\n /** Transparent theme: button/control bg (low opacity purple) when not hovered */\n transparentDefault?: string;\n /** Transparent theme: button/control bg on hover (higher opacity purple) */\n transparentHover?: string;\n /** Transparent theme: input field bg (semi-transparent purple tint) */\n transparentInputBg?: string;\n };\n}\n\nexport interface ThemeAnimation {\n /** Fast transitions (hover states) - 150ms */\n fast: string;\n /** Normal transitions (most interactions) - 250ms */\n normal: string;\n /** Slow transitions (page/mode changes) - 400ms */\n slow: string;\n /** Spring animation for playful interactions */\n spring: string;\n /** Easing function */\n easing: {\n default: string;\n in: string;\n out: string;\n inOut: string;\n };\n /** Duration values in milliseconds */\n duration: {\n instant: number;\n fast: number;\n normal: number;\n slow: number;\n };\n}\n\nexport interface ThemeFocus {\n /** Focus ring color */\n color: string;\n /** Focus ring width */\n width: string;\n /** Focus ring offset */\n offset: string;\n /** Full focus ring CSS */\n ring: string;\n /** Focus ring style object */\n style: React.CSSProperties;\n}\n\n/**\n * Astro UX Typography Style Token\n * Based on https://www.astrouxds.com/foundations/typography/\n */\nexport interface TypographyStyle {\n fontSize: string;\n fontWeight: number;\n letterSpacing: string;\n lineHeight: string;\n}\n\nexport interface ThemeTokens {\n colors: ThemeColors;\n spacing: {\n /** 2px — micro adjustments, decorative offsets */\n xxs: string;\n /** 4px — tight gaps, icon padding */\n xs: string;\n /** 8px — compact spacing, small gaps */\n sm: string;\n /** 12px — form fields, compact cards (between sm and md) */\n smd: string;\n /** 16px — default card padding, section gaps */\n md: string;\n /** 20px — slightly generous spacing */\n mdl: string;\n /** 24px — generous spacing, content area padding */\n lg: string;\n /** 32px — large section gaps */\n xl: string;\n /** 48px — extra large spacing */\n xxl: string;\n };\n borderRadius: {\n none: string;\n /** 1px — very subtle rounding */\n xs: string;\n /** 2px — small elements, badges, tooltips */\n sm: string;\n /** 4px — inputs, selects, standard controls */\n md: string;\n /** 8px — cards, containers */\n lg: string;\n /** 12px — prominent surfaces, modals */\n xl: string;\n /** 9999px — pills, fully rounded */\n full: string;\n };\n /** Standardised element heights for interactive controls */\n elementSize: {\n /** 28px — compact controls */\n sm: string;\n /** 36px — default controls */\n md: string;\n /** 44px — prominent controls, touch targets */\n lg: string;\n };\n typography: {\n fontFamily: {\n primary: string;\n mono: string;\n };\n // Astro UX Typography Tokens (rem-based)\n // https://www.astrouxds.com/foundations/typography/\n display: {\n 1: TypographyStyle;\n 2: TypographyStyle;\n };\n heading: {\n 1: TypographyStyle;\n '1Bold': TypographyStyle;\n 2: TypographyStyle;\n 3: TypographyStyle;\n 4: TypographyStyle;\n 5: TypographyStyle;\n 6: TypographyStyle;\n };\n body: {\n 1: TypographyStyle;\n '1Bold': TypographyStyle;\n 2: TypographyStyle;\n '2Bold': TypographyStyle;\n 3: TypographyStyle;\n '3Bold': TypographyStyle;\n };\n control: {\n 1: TypographyStyle;\n '1Bold': TypographyStyle;\n };\n // Legacy/convenience tokens (keeping for backward compatibility)\n fontSize: {\n micro: string; // 9px - very tight spaces\n xxs: string; // 10px - compact labels\n xs: string;\n sm: string;\n base: string;\n md: string;\n lg: string;\n xl: string;\n xxl: string;\n xxxl: string;\n };\n fontWeight: {\n light: number;\n normal: number;\n medium: number;\n semibold: number;\n bold: number;\n };\n lineHeight: {\n tight: string;\n normal: string;\n relaxed: string;\n };\n letterSpacing: {\n tight: string;\n normal: string;\n wide: string;\n };\n };\n shadows: {\n none: string;\n sm: string;\n md: string;\n lg: string;\n xl: string;\n glow: (color: string) => string;\n };\n animation: ThemeAnimation;\n focus: ThemeFocus;\n /** Semantic layout tokens for consistent spacing across cards, forms, and sections */\n layout: LayoutTokens;\n /** Semantic border tokens for consistent element styling (derived from theme colors) */\n borders: BorderTokens;\n}\n\n// ============================================================================\n// Layout Tokens Interface\n// Semantic spacing for cards, sections, forms, and nested surfaces.\n// These decouple layout intent from raw pixel values, enabling consistent\n// spacing across all consuming applications.\n// ============================================================================\n\nexport interface LayoutTokens {\n /** Card / panel container spacing */\n card: {\n /** Internal padding of card containers */\n padding: string;\n /** Gap between cards in grid layouts (desktop) */\n gap: string;\n /** Gap between cards in grid layouts (mobile / compact) */\n gapCompact: string;\n /** Card heading: icon size, title typography, gap — single system for all cards/charts */\n heading: {\n iconSize: number;\n iconSizeCompact: number;\n titleFontSize: string;\n /** Smaller title for chart headers (e.g. Power Trends) so they match card visual weight */\n titleFontSizeChart: string;\n titleFontWeight: number;\n gap: number;\n };\n };\n /** Section heading spacing (heading + divider + content) */\n section: {\n /** Padding below heading text, above the divider border */\n headerPaddingBottom: string;\n /** Margin below the divider border, before content */\n headerMarginBottom: string;\n /** Padding above the first content element after heading */\n contentPaddingTop: string;\n };\n /** Form and input field spacing */\n form: {\n /** Vertical gap between stacked form fields */\n fieldGap: string;\n /** Horizontal gap between side-by-side inline fields */\n inlineGap: string;\n /** Padding inside grouped / nested field containers */\n groupPadding: string;\n /** Gap between items within a field group */\n groupGap: string;\n /** Gap between label text and helper icons (e.g. tooltip) */\n labelGap: string;\n };\n /** Nested surface / panel spacing (e.g. grouped fields within a card) */\n surface: {\n /** Internal padding of nested surfaces */\n padding: string;\n /** Border radius of nested surfaces */\n borderRadius: string;\n /** Gap between items in compact nested surfaces */\n gap: string;\n };\n}\n\n// ============================================================================\n// Border Tokens Interface\n// Standardised border widths, element borders, focus rings, and dividers.\n// These are DERIVED from theme colors at runtime via computeBorderTokens().\n// ============================================================================\n\nexport interface BorderTokens {\n /** Border widths — standardised across all interactive elements */\n width: {\n /** Thin (1px) — inputs, cards, dividers, dropdowns */\n thin: string;\n /** Medium (1.5px) — secondary buttons, emphasised inputs */\n medium: string;\n /** Thick (2px) — checkboxes, accent borders, active tabs */\n thick: string;\n };\n /** Pre-computed input / control borders (width + style + color) */\n input: {\n /** Default resting state */\n default: string;\n /** Hover state (accent at 50% opacity) */\n hover: string;\n /** Focused state (full accent) */\n focus: string;\n /** Error / critical state */\n error: string;\n };\n /** Focus ring box-shadow values for interactive elements */\n focusRing: {\n /** Standard ring for inputs and selects */\n default: string;\n /** Ring for buttons (double-ring pattern) */\n button: string;\n /** Subtle ring for checkboxes, tabs, toggles */\n subtle: string;\n };\n /** Section divider border (headings, dialog headers/footers, tab bars) */\n divider: string;\n /** Dropdown / popover container border */\n dropdown: string;\n /** Separator line inside elements (unit separators, table rows) */\n separator: string;\n}\n\n/**\n * Derives border tokens from a theme's resolved colors.\n * Called once per theme change inside the ThemeProvider useMemo.\n */\nfunction computeBorderTokens(colors: ThemeColors): BorderTokens {\n const thin = '1px';\n const medium = '1.5px';\n const thick = '2px';\n const accent = colors.accent.primary;\n const muted = colors.border.muted;\n const critical = colors.status.critical;\n const bgBase = colors.background.base;\n\n return {\n width: { thin, medium, thick },\n input: {\n default: `${thin} solid ${muted}`,\n hover: `${thin} solid ${accent}80`,\n focus: `${thin} solid ${accent}`,\n error: `${thin} solid ${critical}`,\n },\n focusRing: {\n // WCAG 2.4.13: focus indicators ≥ 3:1 contrast — raised from 20%/30% to 50%/60%\n default: `0 0 0 3px ${accent}50, 0 0 20px ${accent}15`,\n button: `0 0 0 2px ${bgBase}, 0 0 0 4px ${accent}`,\n subtle: `0 0 0 2px ${accent}60`,\n },\n divider: `${thin} solid ${muted}`,\n dropdown: `${thin} solid ${accent}30`,\n separator: `${thin} solid ${muted}`,\n };\n}\n\n/**\n * Base theme type used for static theme definitions.\n * `borders` is omitted because it is derived from colors at runtime\n * inside the ThemeProvider useMemo (see computeBorderTokens).\n */\ntype ThemeTokensBase = Omit<ThemeTokens, 'borders'>;\n\n// ============================================================================\n// Animation Tokens (Shared)\n// ============================================================================\n\nconst animationTokens: ThemeAnimation = {\n fast: 'all 150ms cubic-bezier(0.4, 0, 0.2, 1)',\n normal: 'all 250ms cubic-bezier(0.4, 0, 0.2, 1)',\n slow: 'all 400ms cubic-bezier(0.4, 0, 0.2, 1)',\n spring: 'all 300ms cubic-bezier(0.34, 1.56, 0.64, 1)',\n easing: {\n default: 'cubic-bezier(0.4, 0, 0.2, 1)',\n in: 'ease-in',\n out: 'ease-out',\n inOut: 'ease-in-out',\n },\n duration: {\n instant: 0,\n fast: 150,\n normal: 250,\n slow: 400,\n },\n};\n\n// ============================================================================\n// Focus Tokens (Dark)\n// ============================================================================\n\nconst focusTokensDark: ThemeFocus = {\n color: '#4dacff',\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #4dacff',\n style: {\n outline: '2px solid #4dacff',\n outlineOffset: '2px',\n },\n};\n\nconst focusTokensLight: ThemeFocus = {\n color: '#0066cc',\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #0066cc',\n style: {\n outline: '2px solid #0066cc',\n outlineOffset: '2px',\n },\n};\n\n// ============================================================================\n// Shadow Tokens\n// ============================================================================\n\nconst shadowsDark = {\n none: 'none',\n sm: '0 1px 2px rgba(0, 0, 0, 0.3)',\n md: '0 4px 8px rgba(0, 0, 0, 0.4)',\n lg: '0 8px 16px rgba(0, 0, 0, 0.5)',\n xl: '0 16px 32px rgba(0, 0, 0, 0.6)',\n glow: (color: string) => `0 0 20px ${color}40, 0 0 40px ${color}20`,\n};\n\nconst shadowsLight = {\n none: 'none',\n sm: '0 1px 2px rgba(0, 0, 0, 0.1)',\n md: '0 4px 8px rgba(0, 0, 0, 0.15)',\n lg: '0 8px 16px rgba(0, 0, 0, 0.2)',\n xl: '0 16px 32px rgba(0, 0, 0, 0.25)',\n glow: (color: string) => `0 0 20px ${color}30, 0 0 40px ${color}15`,\n};\n\n// ============================================================================\n// Layout Tokens (Shared)\n// Semantic layout spacing based on 4px grid. Shared across all themes.\n// ============================================================================\n\nconst layoutTokens: LayoutTokens = {\n card: {\n padding: '16px', // 4 × 4px\n gap: '24px', // 6 × 4px — desktop grid gap (matches outer padding)\n gapCompact: '16px', // 4 × 4px — mobile grid gap\n heading: {\n iconSize: 18, // default card header icon (px) — matches ISS/Telemetry card header size\n iconSizeCompact: 16, // compact (e.g. chart inline header)\n titleFontSize: '1rem', // card titles (ISS, Telemetry, Mission) — reduced from 1.125rem\n titleFontSizeChart: '0.875rem', // chart titles (Power Trends) — slightly smaller\n titleFontWeight: 500,\n gap: 8, // gap between icon and title (px)\n },\n },\n section: {\n headerPaddingBottom: '8px', // 2 × 4px\n headerMarginBottom: '0px',\n contentPaddingTop: '8px', // 2 × 4px\n },\n form: {\n fieldGap: '12px', // 3 × 4px — between stacked fields\n inlineGap: '12px', // 3 × 4px — between side-by-side fields\n groupPadding: '12px', // 3 × 4px — inside grouped containers\n groupGap: '12px', // 3 × 4px — within group items\n labelGap: '3px', // label text ↔ tooltip icon\n },\n surface: {\n padding: '12px', // 3 × 4px\n borderRadius: '12px', // 3 × 4px\n gap: '8px', // 2 × 4px\n },\n};\n\n// ============================================================================\n// Astro UX Dark Theme\n// ============================================================================\n\nconst astroThemeDark: ThemeTokensBase = {\n colors: {\n background: {\n base: '#101923',\n surface: '#1b2d3e',\n elevated: '#243b53',\n overlay: 'rgba(16, 25, 35, 0.95)',\n },\n border: {\n default: '#2b659b',\n muted: '#172635',\n focus: '#4dacff',\n },\n text: {\n primary: '#ffffff',\n secondary: '#b7c5d3',\n tertiary: '#8fa4b7',\n muted: '#8fa4b7',\n inverse: '#1b2d3e',\n },\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#4dacff',\n },\n accent: {\n primary: '#4dacff',\n secondary: '#56f000',\n tertiary: '#a371f7',\n },\n interactive: {\n default: '#4dacff',\n hover: '#92cbff',\n active: '#0066cc',\n disabled: '#6b7280',\n },\n },\n spacing: {\n xxs: '2px',\n xs: '4px',\n sm: '8px',\n smd: '12px',\n md: '16px',\n mdl: '20px',\n lg: '24px',\n xl: '32px',\n xxl: '48px',\n },\n borderRadius: {\n none: '0',\n xs: '1px',\n sm: '2px',\n md: '4px',\n lg: '8px',\n xl: '12px',\n full: '9999px',\n },\n elementSize: {\n sm: '28px',\n md: '36px',\n lg: '44px',\n },\n typography: {\n /**\n * Font families per AstroUXDS specification\n * @see https://www.astrouxds.com/foundations/typography/\n */\n fontFamily: {\n /** Primary font: Roboto with comprehensive system fallbacks */\n primary: '\"Roboto\", -apple-system, BlinkMacSystemFont, \"Segoe UI\", \"Helvetica Neue\", Arial, sans-serif',\n /** Monospace font: Roboto Mono for data/code with system fallbacks */\n mono: '\"Roboto Mono\", \"SF Mono\", \"Consolas\", \"Liberation Mono\", monospace',\n },\n // Astro UX Display Styles\n // https://www.astrouxds.com/foundations/typography/\n display: {\n 1: { fontSize: '3.75rem', fontWeight: 300, letterSpacing: '-0.5px', lineHeight: '4.375rem' },\n 2: { fontSize: '3rem', fontWeight: 400, letterSpacing: '0', lineHeight: '3.5rem' },\n },\n // Astro UX Heading Styles\n heading: {\n 1: { fontSize: '2.125rem', fontWeight: 400, letterSpacing: '0.25px', lineHeight: '2.5rem' },\n '1Bold': { fontSize: '2.125rem', fontWeight: 700, letterSpacing: '0.25px', lineHeight: '2rem' },\n 2: { fontSize: '1.5rem', fontWeight: 400, letterSpacing: '0', lineHeight: '1.75rem' },\n 3: { fontSize: '1.25rem', fontWeight: 500, letterSpacing: '0.15px', lineHeight: '1.5rem' },\n 4: { fontSize: '1.25rem', fontWeight: 300, letterSpacing: '0.15px', lineHeight: '1.5rem' },\n 5: { fontSize: '1.125rem', fontWeight: 400, letterSpacing: '0', lineHeight: '1.5rem' },\n 6: { fontSize: '1.125rem', fontWeight: 300, letterSpacing: '0', lineHeight: '1.5rem' },\n },\n // Astro UX Body Styles\n body: {\n 1: { fontSize: '1rem', fontWeight: 400, letterSpacing: '0.5px', lineHeight: '1.5rem' },\n '1Bold': { fontSize: '1rem', fontWeight: 700, letterSpacing: '0.5px', lineHeight: '1.5rem' },\n 2: { fontSize: '0.875rem', fontWeight: 400, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n '2Bold': { fontSize: '0.875rem', fontWeight: 700, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n 3: { fontSize: '0.75rem', fontWeight: 400, letterSpacing: '0', lineHeight: '1rem' },\n '3Bold': { fontSize: '0.75rem', fontWeight: 700, letterSpacing: '0', lineHeight: '1rem' },\n },\n // Astro UX Control Body Styles\n control: {\n 1: { fontSize: '1rem', fontWeight: 400, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n '1Bold': { fontSize: '1rem', fontWeight: 700, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n },\n // Legacy convenience tokens (rem-based for Astro compliance)\n // Note: Use Typography component for new code\n fontSize: {\n micro: '0.5625rem', // 9px - very tight spaces\n xxs: '0.625rem', // 10px - compact labels (xs in old system)\n xs: '0.625rem', // 10px - compact (kept for backward compatibility)\n sm: '0.75rem', // 12px (Body 3)\n base: '0.875rem', // 14px (Body 2)\n md: '0.875rem', // 14px (Body 2)\n lg: '1rem', // 16px (Body 1)\n xl: '1.25rem', // 20px (Heading 3/4)\n xxl: '1.5rem', // 24px (Heading 2)\n xxxl: '2.125rem', // 34px (Heading 1)\n },\n /**\n * Font weights per AstroUXDS specification\n * Only 300, 400, 500, 700 are officially supported\n * @see https://www.astrouxds.com/foundations/typography/\n */\n fontWeight: {\n light: 300,\n normal: 400,\n medium: 500,\n /** @deprecated Use medium (500) instead - Astro spec only supports 300, 400, 500, 700 */\n semibold: 500, // Changed from 600 to 500 for Astro compliance\n bold: 700,\n },\n lineHeight: {\n tight: '1.25rem',\n normal: '1.5rem',\n relaxed: '1.75rem',\n },\n letterSpacing: {\n tight: '-0.5px',\n normal: '0',\n wide: '0.5px',\n },\n },\n shadows: shadowsDark,\n animation: animationTokens,\n focus: focusTokensDark,\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Astro UX Light Theme\n// ============================================================================\n\nconst astroThemeLight: ThemeTokensBase = {\n colors: {\n background: {\n base: '#eaeef4',\n surface: '#f5f8fc',\n elevated: '#ffffff',\n overlay: 'rgba(234, 238, 244, 0.95)',\n },\n border: {\n default: '#2b659b',\n muted: '#d1d5db',\n focus: '#0066cc',\n },\n text: {\n primary: '#1b2d3e',\n secondary: '#51667c',\n tertiary: '#556a7c', // WCAG AA: 4.81:1 on #eaeef4, 5.24:1 on #f5f8fc, 5.61:1 on #fff (was #8fa4b7 → 2.25:1 FAIL)\n muted: '#556a7c', // WCAG AA compliant (was #8fa4b7)\n inverse: '#ffffff',\n },\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#4dacff',\n },\n accent: {\n primary: '#0066cc',\n secondary: '#56f000',\n tertiary: '#7c3aed',\n },\n interactive: {\n default: '#0066cc',\n hover: '#0052a3',\n active: '#003d7a',\n disabled: '#a4abb6',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: astroThemeDark.borderRadius,\n elementSize: astroThemeDark.elementSize,\n typography: astroThemeDark.typography,\n shadows: shadowsLight,\n animation: animationTokens,\n focus: focusTokensLight,\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Hybrid Theme (Zendir Background + Astro Status + Purple Accents)\n// Modern, award-winning design with Zendir purple personality\n// ============================================================================\n\nconst focusTokensPurpleHue: ThemeFocus = {\n color: '#8b5cf6',\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #8b5cf6',\n style: {\n outline: '2px solid #8b5cf6',\n outlineOffset: '2px',\n },\n};\n\nconst hybridTheme: ThemeTokensBase = {\n colors: {\n background: {\n base: '#09090b',\n surface: 'rgba(24, 24, 27, 0.98)',\n elevated: 'rgba(39, 39, 42, 0.98)',\n overlay: 'rgba(0, 0, 0, 0.85)',\n },\n border: {\n default: 'rgba(139, 92, 246, 0.3)', // Purple tint\n muted: 'rgba(63, 63, 70, 0.5)',\n focus: '#8b5cf6', // Zendir purple\n },\n text: {\n primary: '#ffffff',\n secondary: '#a1a1aa',\n tertiary: '#8e8e96', // WCAG AA: 6.12:1 on #09090b, 5.45:1 on #18181b, 4.58:1 on #27272a (was #71717a → 4.07:1 FAIL)\n muted: '#8e8e96', // WCAG AA compliant (was #71717a)\n inverse: '#09090b',\n },\n // Astro UXD Status Colors (unchanged for compliance)\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#8b5cf6', // Purple for info\n },\n // Zendir Purple Accent Palette\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6',\n hover: '#a78bfa',\n active: '#7c3aed',\n disabled: '#52525b',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '6px',\n lg: '10px',\n xl: '14px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n glow: (color: string) => `0 0 20px ${color}50, 0 0 40px ${color}25, 0 4px 12px ${color}20`,\n },\n animation: animationTokens,\n focus: focusTokensPurpleHue,\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Purple-Hue Theme - Full Purple Personality\n// Award-winning dark theme with purple accents\n// ============================================================================\n\nconst purpleHueTheme: ThemeTokensBase = {\n colors: {\n background: {\n base: '#0a0a0f',\n surface: 'rgba(18, 18, 28, 0.98)',\n elevated: 'rgba(28, 28, 42, 0.98)',\n overlay: 'rgba(0, 0, 0, 0.9)',\n },\n border: {\n default: 'rgba(139, 92, 246, 0.25)',\n muted: 'rgba(55, 48, 82, 0.5)',\n focus: '#8b5cf6',\n },\n text: {\n primary: '#fafafa',\n secondary: '#a1a1aa',\n tertiary: '#8e8e96', // WCAG AA: 6.08:1 on #0a0a0f, 5.02:1 on elevated (was #71717a → 4.05:1 FAIL)\n muted: '#8e8e96', // WCAG AA compliant (was #71717a)\n inverse: '#0a0a0f',\n },\n // Astro UXD Status Colors (for compliance)\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#a78bfa',\n },\n // Zendir Purple Accent Palette\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6',\n hover: '#a78bfa',\n active: '#7c3aed',\n disabled: '#52525b',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '6px',\n lg: '10px',\n xl: '14px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n // Same typography as hybrid (Astro UX font stack with full fallbacks) for consistent card font across themes\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n glow: (color: string) => `0 0 25px ${color}60, 0 0 50px ${color}30, 0 4px 16px ${color}25`,\n },\n animation: animationTokens,\n focus: {\n ...focusTokensPurpleHue,\n },\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Transparent Purple-Hue Theme - Glassmorphic Panels\n// Optimized for glass card overlays with transparent backgrounds\n// ============================================================================\n\nconst transparentTheme: ThemeTokensBase = {\n colors: {\n background: {\n // Highly transparent backgrounds for glass effect\n base: 'rgba(8, 12, 20, 0.95)',\n surface: 'rgba(15, 20, 35, 0.85)',\n elevated: 'rgba(25, 35, 55, 0.80)',\n overlay: 'rgba(0, 0, 0, 0.75)',\n },\n border: {\n // Purple accent borders (Zendir brand)\n default: 'rgba(139, 92, 246, 0.25)',\n muted: 'rgba(139, 92, 246, 0.15)',\n focus: '#8b5cf6',\n // Thin faded card border with purple hint\n fadedBoxShadow: 'inset 0 1px 0 rgba(139, 92, 246, 0.12), inset 0 -1px 0 rgba(0, 0, 0, 0.05), inset 1px 0 0 rgba(139, 92, 246, 0.06), inset -1px 0 0 rgba(139, 92, 246, 0.06)',\n },\n text: {\n primary: '#f0f4f8',\n secondary: '#94a3b8',\n tertiary: '#7b8da1', // WCAG AA: 5.71:1 on base, 5.34:1 on elevated (was #64748b → 3.66:1 FAIL)\n muted: '#7b8da1', // WCAG AA compliant (was #64748b)\n inverse: '#0f172a',\n },\n // Astro UXD Status Colors\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#8b5cf6',\n },\n // Purple accent (matching hybrid theme for consistency)\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6', // Purple for interactive elements\n hover: '#a78bfa', // Light purple on hover\n active: '#7c3aed', // Deep purple on active\n disabled: '#475569',\n // Purple-hue transparent backgrounds for buttons/inputs (transparent + transparent-bold)\n transparentDefault: 'rgba(139, 92, 246, 0.12)',\n transparentHover: 'rgba(139, 92, 246, 0.32)',\n transparentInputBg: 'rgba(139, 92, 246, 0.08)',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '8px',\n lg: '12px',\n xl: '16px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n // Same typography as hybrid for consistent card font across themes\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n // Enhanced glow shadows for glass effect\n glow: (color: string) => `0 0 30px ${color}50, 0 0 60px ${color}25, 0 4px 20px rgba(0,0,0,0.4)`,\n },\n animation: animationTokens,\n focus: {\n color: '#8b5cf6', // Purple focus ring\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #8b5cf6',\n style: {\n outline: '2px solid #8b5cf6',\n outlineOffset: '2px',\n },\n },\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Transparent Minimal Theme - Ultra-clean glassmorphic panels\n// Extremely subtle borders, minimal visual noise, clean dark surfaces\n// ============================================================================\n\nconst transparentMinimalTheme: ThemeTokensBase = {\n colors: {\n background: {\n // Same as transparent theme\n base: 'rgba(8, 12, 20, 0.95)',\n surface: 'rgba(15, 20, 35, 0.85)',\n elevated: 'rgba(25, 35, 55, 0.80)',\n overlay: 'rgba(0, 0, 0, 0.75)',\n },\n border: {\n // Accent-colored borders for minimal theme (Zendir purple)\n default: 'rgba(139, 92, 246, 0.25)',\n muted: 'rgba(139, 92, 246, 0.15)',\n focus: '#8b5cf6',\n // Subtle card border using accent color\n fadedBoxShadow: 'inset 0 1px 0 rgba(139, 92, 246, 0.1), inset 0 -1px 0 rgba(0, 0, 0, 0.1), inset 1px 0 0 rgba(139, 92, 246, 0.05), inset -1px 0 0 rgba(139, 92, 246, 0.05)',\n },\n text: {\n primary: '#f0f4f8',\n secondary: '#94a3b8',\n tertiary: '#7b8da1', // WCAG AA: 5.71:1 on base, 5.34:1 on elevated (was #64748b → 3.66:1 FAIL)\n muted: '#7b8da1', // WCAG AA compliant (was #64748b)\n inverse: '#0f172a',\n },\n // Astro UXD Status Colors\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#8b5cf6',\n },\n // Purple accent (matching transparent theme for consistency)\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6', // Purple for interactive elements\n hover: '#a78bfa', // Light purple on hover\n active: '#7c3aed', // Deep purple on active\n disabled: '#475569',\n // Purple-hue transparent backgrounds for buttons/inputs (same as transparent theme)\n transparentDefault: 'rgba(139, 92, 246, 0.12)',\n transparentHover: 'rgba(139, 92, 246, 0.32)',\n transparentInputBg: 'rgba(139, 92, 246, 0.08)',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '8px',\n lg: '12px',\n xl: '16px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n // Same typography as hybrid for consistent card font across themes\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n // Same glow as transparent theme\n glow: (color: string) => `0 0 30px ${color}50, 0 0 60px ${color}25, 0 4px 20px rgba(0,0,0,0.4)`,\n },\n animation: animationTokens,\n focus: {\n color: '#8b5cf6', // Purple focus ring\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #8b5cf6',\n style: {\n outline: '2px solid #8b5cf6',\n outlineOffset: '2px',\n },\n },\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Theme Context\n// ============================================================================\n\nexport interface ThemeContextValue {\n theme: ThemeVariant;\n mode: ThemeMode;\n tokens: ThemeTokens;\n prefersReducedMotion: boolean;\n /** Custom accent color override (null = use theme default) */\n accentColor: string | null;\n setTheme: (theme: ThemeVariant) => void;\n setMode: (mode: ThemeMode) => void;\n toggleMode: () => void;\n /** Override the accent color for the current theme */\n setAccentColor: (color: string | null) => void;\n}\n\nconst ThemeContext = createContext<ThemeContextValue | undefined>(undefined);\n\nexport interface ThemeProviderProps {\n children: ReactNode;\n /** Default theme variant */\n theme?: ThemeVariant;\n /** Alias for `theme` */\n defaultVariant?: ThemeVariant;\n /** Default color mode */\n mode?: ThemeMode;\n /** Alias for `mode` */\n defaultMode?: ThemeMode;\n /** Persist preferences to localStorage */\n persist?: boolean;\n}\n\n// ============================================================================\n// Scrollbar Styling System\n// ============================================================================\n// Generates global CSS for thin, elegant, theme-aware scrollbars.\n//\n// Features:\n// - Ultra-thin (6px) scrollbar tracks with soft border radius\n// - Accent-colored thumb with hover/active brightness shifts\n// - Semi-transparent track that blends into the surface\n// - Auto-hide behaviour on modern browsers (overlay mode)\n// - Firefox support via scrollbar-width / scrollbar-color\n// - WCAG 2.1 AA compliant: sufficient contrast, respects reduced-motion\n// - Smooth transitions (disabled when prefers-reduced-motion)\n// - Scoped .zendir-scroll utility class for individual containers\n// ============================================================================\n\n/**\n * Converts a hex color to an rgba string helper.\n * Accepts 3/6/8-char hex codes (#rgb, #rrggbb, #rrggbbaa).\n */\nfunction hexToRgba(hex: string, alpha: number): string {\n const clean = hex.replace('#', '');\n let r: number, g: number, b: number;\n if (clean.length === 3) {\n r = parseInt(clean[0] + clean[0], 16);\n g = parseInt(clean[1] + clean[1], 16);\n b = parseInt(clean[2] + clean[2], 16);\n } else {\n r = parseInt(clean.slice(0, 2), 16);\n g = parseInt(clean.slice(2, 4), 16);\n b = parseInt(clean.slice(4, 6), 16);\n }\n return `rgba(${r}, ${g}, ${b}, ${alpha})`;\n}\n\nfunction buildScrollbarCSS(tokens: ThemeTokens, mode: ThemeMode, reducedMotion: boolean): string {\n const accent = tokens.colors.accent.primary;\n const _surface = tokens.colors.background.surface; // eslint-disable-line @typescript-eslint/no-unused-vars -- reserved for future scrollbar track customisation\n const _base = tokens.colors.background.base; // eslint-disable-line @typescript-eslint/no-unused-vars -- reserved for future scrollbar corner styling\n const isLight = mode === 'light';\n\n // Thumb colors: accent-tinted, low opacity at rest, stronger on hover/active\n const thumbRest = isLight ? hexToRgba(accent, 0.28) : hexToRgba(accent, 0.35);\n const thumbHover = isLight ? hexToRgba(accent, 0.48) : hexToRgba(accent, 0.55);\n const thumbActive = isLight ? hexToRgba(accent, 0.64) : hexToRgba(accent, 0.72);\n\n // Track: nearly invisible surface blend\n const trackColor = isLight ? hexToRgba(accent, 0.04) : hexToRgba(accent, 0.06);\n const trackHover = isLight ? hexToRgba(accent, 0.08) : hexToRgba(accent, 0.10);\n\n const transition = reducedMotion ? 'none' : 'background-color 200ms ease, width 200ms ease, opacity 200ms ease';\n const thumbRadius = '9999px'; // pill shape\n\n // Firefox colors (two-value syntax: thumb track)\n const ffThumb = thumbRest;\n const ffTrack = trackColor;\n\n return `\n/* ═══════════════════════════════════════════════════════════════════════════\n Zendir Scrollbar System — Thin · Elegant · Theme-Connected\n ═══════════════════════════════════════════════════════════════════════════ */\n\n/* ── Firefox ────────────────────────────────────────────────────────────── */\n*,\n.zendir-scroll {\n scrollbar-width: thin;\n scrollbar-color: ${ffThumb} ${ffTrack};\n}\n\n/* ── Webkit / Blink (Chrome, Edge, Safari, Opera) ───────────────────────── */\n\n/* Global thin scrollbar */\n::-webkit-scrollbar {\n width: 6px;\n height: 6px;\n background: transparent;\n}\n\n/* Track — nearly invisible, subtle on hover */\n::-webkit-scrollbar-track {\n background: ${trackColor};\n border-radius: ${thumbRadius};\n margin: 2px;\n}\n::-webkit-scrollbar-track:hover {\n background: ${trackHover};\n}\n\n/* Thumb — accent-tinted pill */\n::-webkit-scrollbar-thumb {\n background: ${thumbRest};\n border-radius: ${thumbRadius};\n border: 1px solid transparent;\n background-clip: padding-box;\n transition: ${transition};\n min-height: 32px;\n}\n::-webkit-scrollbar-thumb:hover {\n background: ${thumbHover};\n border-color: transparent;\n background-clip: padding-box;\n}\n::-webkit-scrollbar-thumb:active {\n background: ${thumbActive};\n border-color: transparent;\n background-clip: padding-box;\n}\n\n/* Corner — where horizontal & vertical scrollbars meet */\n::-webkit-scrollbar-corner {\n background: transparent;\n}\n\n/* ── Utility class: zendir-scroll ────────────────────────────────────────\n Apply to individual containers for scoped thin scrollbar.\n Also enables overlay behaviour where supported.\n e.g. <div className=\"zendir-scroll\" style={{overflow:'auto'}}>…</div>\n ─────────────────────────────────────────────────────────────────────── */\n.zendir-scroll {\n overflow: auto;\n overflow: overlay; /* Chrome / Edge: auto-hide scrollbar */\n}\n.zendir-scroll::-webkit-scrollbar {\n width: 5px;\n height: 5px;\n}\n.zendir-scroll::-webkit-scrollbar-track {\n background: transparent;\n border-radius: ${thumbRadius};\n}\n.zendir-scroll::-webkit-scrollbar-thumb {\n background: ${thumbRest};\n border-radius: ${thumbRadius};\n transition: ${transition};\n}\n.zendir-scroll::-webkit-scrollbar-thumb:hover {\n background: ${thumbHover};\n}\n.zendir-scroll::-webkit-scrollbar-thumb:active {\n background: ${thumbActive};\n}\n\n/* ── Utility: zendir-scroll--hidden ─────────────────────────────────────\n Hides scrollbar visually but keeps scroll behaviour (accessible).\n Content remains scrollable via mouse wheel / touch / keyboard.\n ─────────────────────────────────────────────────────────────────────── */\n.zendir-scroll--hidden {\n overflow: auto;\n -ms-overflow-style: none;\n scrollbar-width: none;\n}\n.zendir-scroll--hidden::-webkit-scrollbar {\n display: none;\n}\n\n/* ── Focus-visible: keyboard users can still scroll with focus ─────────── */\n.zendir-scroll:focus-visible {\n outline: 2px solid ${accent};\n outline-offset: -2px;\n border-radius: ${tokens.borderRadius.md};\n}\n\n/* ── Reduced Motion: disable scrollbar transitions ─────────────────────── */\n@media (prefers-reduced-motion: reduce) {\n ::-webkit-scrollbar-thumb,\n .zendir-scroll::-webkit-scrollbar-thumb {\n transition: none !important;\n }\n}\n\n/* ── High Contrast Mode: ensure scrollbar is visible ───────────────────── */\n@media (forced-colors: active) {\n ::-webkit-scrollbar-thumb {\n background: ButtonText !important;\n }\n ::-webkit-scrollbar-track {\n background: Canvas !important;\n }\n}\n`.trim();\n}\n\n/**\n * useScrollbarStyles — returns inline React.CSSProperties for a\n * scrollable container that matches the current Zendir theme.\n *\n * Use when you need programmatic control rather than the\n * `zendir-scroll` CSS class.\n *\n * @example\n * ```tsx\n * const scrollStyle = useScrollbarStyles();\n * <div style={{ maxHeight: 400, ...scrollStyle }}>…</div>\n * ```\n */\nexport function useScrollbarStyles(): React.CSSProperties {\n const { tokens, mode } = useTheme();\n return useMemo((): React.CSSProperties => {\n const accent = tokens.colors.accent.primary;\n const isLight = mode === 'light';\n const thumbColor = isLight ? hexToRgba(accent, 0.28) : hexToRgba(accent, 0.35);\n const trackColor = isLight ? hexToRgba(accent, 0.04) : hexToRgba(accent, 0.06);\n return {\n // Firefox\n scrollbarWidth: 'thin' as const,\n scrollbarColor: `${thumbColor} ${trackColor}`,\n overflow: 'auto',\n };\n }, [tokens, mode]);\n}\n\n/**\n * ThemeProvider - Enterprise theme context provider\n * \n * Provides theme tokens, mode switching, and accessibility features\n * to all child components.\n * \n * @example\n * ```tsx\n * <ThemeProvider theme=\"hybrid\" mode=\"dark\"> // Zen (Hybrid) is the default\n * <App />\n * </ThemeProvider>\n * ```\n */\nexport function ThemeProvider({\n children,\n theme: themeProp,\n defaultVariant,\n mode: modeProp,\n defaultMode: defaultModeProp,\n persist = true,\n}: ThemeProviderProps): React.ReactElement {\n const defaultTheme = themeProp ?? defaultVariant ?? 'hybrid';\n const defaultMode = modeProp ?? defaultModeProp ?? 'dark';\n // Initialize theme from localStorage or defaults\n const [theme, setThemeState] = useState<ThemeVariant>(() => {\n if (persist && typeof window !== 'undefined') {\n const saved = localStorage.getItem('zendir-ui-theme');\n if (saved === 'astro' || saved === 'zendir' || saved === 'hybrid' || saved === 'transparent' || saved === 'transparent-bold' || saved === 'transparent-minimal') {\n return saved as ThemeVariant;\n }\n }\n return defaultTheme;\n });\n\n const [mode, setModeState] = useState<ThemeMode>(() => {\n if (persist && typeof window !== 'undefined') {\n const saved = localStorage.getItem('zendir-ui-mode');\n if (saved === 'light' || saved === 'dark') {\n return saved;\n }\n }\n return defaultMode;\n });\n\n // Detect reduced motion preference\n const [prefersReducedMotion, setPrefersReducedMotion] = useState(false);\n\n useEffect(() => {\n if (typeof window === 'undefined') return;\n\n const mediaQuery = window.matchMedia('(prefers-reduced-motion: reduce)');\n setPrefersReducedMotion(mediaQuery.matches);\n\n const handler = (e: MediaQueryListEvent) => setPrefersReducedMotion(e.matches);\n mediaQuery.addEventListener('change', handler);\n return () => mediaQuery.removeEventListener('change', handler);\n }, []);\n\n // Theme setters\n const setTheme = (newTheme: ThemeVariant) => {\n setThemeState(newTheme);\n if (persist && typeof window !== 'undefined') {\n localStorage.setItem('zendir-ui-theme', newTheme);\n }\n };\n\n const setMode = (newMode: ThemeMode) => {\n setModeState(newMode);\n if (persist && typeof window !== 'undefined') {\n localStorage.setItem('zendir-ui-mode', newMode);\n }\n };\n\n const toggleMode = () => {\n setMode(mode === 'dark' ? 'light' : 'dark');\n };\n\n // Accent color override\n const [accentColor, setAccentColorState] = useState<string | null>(() => {\n if (persist && typeof window !== 'undefined') {\n return localStorage.getItem('zendir-ui-accent') || null;\n }\n return null;\n });\n\n const setAccentColor = (color: string | null) => {\n setAccentColorState(color);\n if (persist && typeof window !== 'undefined') {\n if (color) {\n localStorage.setItem('zendir-ui-accent', color);\n } else {\n localStorage.removeItem('zendir-ui-accent');\n }\n }\n };\n\n // Get theme tokens (card border styles computed so all cards use theme-driven borders)\n const tokens = useMemo((): ThemeTokens => {\n // If light mode is selected, always use Astro Light theme (unified light mode)\n // regardless of the selected theme variant\n const base =\n mode === 'light'\n ? astroThemeLight\n : theme === 'astro'\n ? astroThemeDark\n : theme === 'hybrid'\n ? hybridTheme\n : theme === 'transparent' || theme === 'transparent-bold'\n ? transparentTheme\n : theme === 'transparent-minimal'\n ? transparentMinimalTheme\n : purpleHueTheme;\n const isTransparent =\n theme === 'transparent' || theme === 'transparent-bold' || theme === 'transparent-minimal';\n const isMinimal = theme === 'transparent-minimal';\n \n // For transparent-minimal: use visible accent-colored borders.\n // For other transparent themes: use subtle fadedBoxShadow.\n // Accent overrides are intentionally scoped to transparent variants only,\n // so switching to Astro/Hybrid/Purple-Hue restores each theme's native accent.\n const resolvedAccent = (isTransparent && accentColor)\n ? {\n primary: accentColor,\n secondary: adjustHexBrightness(accentColor, 1.2),\n tertiary: adjustHexBrightness(accentColor, 0.85),\n }\n : base.colors.accent;\n\n const accentPrimary = resolvedAccent.primary;\n const accentBorderMuted = `${accentPrimary}40`; // 25% opacity\n const accentBorderDefault = `${accentPrimary}66`; // 40% opacity\n const accentFadedBoxShadow = `inset 0 1px 0 ${hexToRgba(accentPrimary, 0.12)}, inset 0 -1px 0 rgba(0, 0, 0, 0.05), inset 1px 0 0 ${hexToRgba(accentPrimary, 0.06)}, inset -1px 0 0 ${hexToRgba(accentPrimary, 0.06)}`;\n \n const cardStyle = isMinimal\n ? { border: `1px solid ${accentBorderMuted}` }\n : isTransparent\n ? { border: 'none' as const, boxShadow: accentFadedBoxShadow }\n : { border: `1px solid ${base.colors.border.muted}` };\n const cardStyleDashed = isMinimal\n ? { border: `1px dashed ${accentBorderMuted}` }\n : isTransparent\n ? { border: 'none' as const, boxShadow: accentFadedBoxShadow }\n : { border: `1px dashed ${base.colors.border.muted}` };\n const cardStyleDefault = isMinimal\n ? { border: `1px solid ${accentBorderDefault}` }\n : isTransparent\n ? { border: 'none' as const, boxShadow: accentFadedBoxShadow }\n : { border: `1px solid ${base.colors.border.default}` };\n // Compute border tokens from the resolved theme colors\n const resolvedColors: ThemeColors = {\n ...base.colors,\n accent: resolvedAccent,\n border: {\n ...base.colors.border,\n cardStyle,\n cardStyleDashed,\n cardStyleDefault,\n },\n };\n\n return {\n ...base,\n colors: resolvedColors,\n borders: computeBorderTokens(resolvedColors),\n } as ThemeTokens;\n }, [theme, mode, accentColor]);\n\n // Apply CSS variables to document\n useEffect(() => {\n if (typeof document === 'undefined') return;\n\n const root = document.documentElement;\n const { colors, typography, spacing, borderRadius, animation, focus, shadows } = tokens;\n\n // Theme mode attribute\n root.setAttribute('data-theme', mode);\n root.setAttribute('data-variant', theme);\n\n // Background colors\n root.style.setProperty('--color-background-base', colors.background.base);\n root.style.setProperty('--color-background-surface', colors.background.surface);\n root.style.setProperty('--color-background-elevated', colors.background.elevated);\n root.style.setProperty('--color-background-overlay', colors.background.overlay);\n\n // Border colors\n root.style.setProperty('--color-border', colors.border.default);\n root.style.setProperty('--color-border-muted', colors.border.muted);\n root.style.setProperty('--color-border-focus', colors.border.focus);\n\n // Text colors\n root.style.setProperty('--color-text-primary', colors.text.primary);\n root.style.setProperty('--color-text-secondary', colors.text.secondary);\n root.style.setProperty('--color-text-tertiary', colors.text.tertiary);\n root.style.setProperty('--color-text-inverse', colors.text.inverse);\n\n // Status colors\n root.style.setProperty('--color-status-normal', colors.status.normal);\n root.style.setProperty('--color-status-standby', colors.status.standby);\n root.style.setProperty('--color-status-caution', colors.status.caution);\n root.style.setProperty('--color-status-serious', colors.status.serious);\n root.style.setProperty('--color-status-critical', colors.status.critical);\n root.style.setProperty('--color-status-off', colors.status.off);\n\n // Interactive colors\n root.style.setProperty('--color-interactive-default', colors.interactive.default);\n root.style.setProperty('--color-interactive-hover', colors.interactive.hover);\n root.style.setProperty('--color-interactive-active', colors.interactive.active);\n root.style.setProperty('--color-interactive-disabled', colors.interactive.disabled);\n\n // Accent colors\n root.style.setProperty('--color-accent-primary', colors.accent.primary);\n root.style.setProperty('--color-accent-secondary', colors.accent.secondary);\n root.style.setProperty('--color-accent-tertiary', colors.accent.tertiary);\n\n // Typography\n // @ts-expect-error - heading may not be strictly typed on older builds\n root.style.setProperty('--font-family-heading', typography.fontFamily.heading || typography.fontFamily.primary);\n root.style.setProperty('--font-family-primary', typography.fontFamily.primary);\n root.style.setProperty('--font-family-mono', typography.fontFamily.mono);\n\n // Spacing (full scale so CSS and layout can pull from SDK)\n root.style.setProperty('--spacing-xxs', spacing.xxs);\n root.style.setProperty('--spacing-xs', spacing.xs);\n root.style.setProperty('--spacing-sm', spacing.sm);\n root.style.setProperty('--spacing-smd', spacing.smd);\n root.style.setProperty('--spacing-md', spacing.md);\n root.style.setProperty('--spacing-mdl', spacing.mdl);\n root.style.setProperty('--spacing-lg', spacing.lg);\n root.style.setProperty('--spacing-xl', spacing.xl);\n root.style.setProperty('--spacing-xxl', spacing.xxl);\n\n // Border radius\n root.style.setProperty('--radius-sm', borderRadius.sm);\n root.style.setProperty('--radius-md', borderRadius.md);\n root.style.setProperty('--radius-lg', borderRadius.lg);\n root.style.setProperty('--radius-xl', borderRadius.xl);\n\n // Animation (respect reduced motion)\n const motionMultiplier = prefersReducedMotion ? 0 : 1;\n root.style.setProperty('--animation-fast', prefersReducedMotion ? 'none' : animation.fast);\n root.style.setProperty('--animation-normal', prefersReducedMotion ? 'none' : animation.normal);\n root.style.setProperty('--animation-slow', prefersReducedMotion ? 'none' : animation.slow);\n root.style.setProperty('--duration-fast', `${animation.duration.fast * motionMultiplier}ms`);\n root.style.setProperty('--duration-normal', `${animation.duration.normal * motionMultiplier}ms`);\n root.style.setProperty('--duration-slow', `${animation.duration.slow * motionMultiplier}ms`);\n\n // Focus\n root.style.setProperty('--focus-ring-color', focus.color);\n root.style.setProperty('--focus-ring-width', focus.width);\n root.style.setProperty('--focus-ring-offset', focus.offset);\n\n // Shadows\n root.style.setProperty('--shadow-sm', shadows.sm);\n root.style.setProperty('--shadow-md', shadows.md);\n root.style.setProperty('--shadow-lg', shadows.lg);\n root.style.setProperty('--shadow-xl', shadows.xl);\n\n // Apply base styles to body\n document.body.style.backgroundColor = colors.background.base;\n document.body.style.color = colors.text.primary;\n document.body.style.fontFamily = typography.fontFamily.primary;\n }, [tokens, theme, mode, prefersReducedMotion]);\n\n // ──────────────────────────────────────────────────────────────────────────\n // Scrollbar Styles — thin, elegant, theme-connected, fully accessible\n // ──────────────────────────────────────────────────────────────────────────\n const scrollbarCSS = useMemo(() => buildScrollbarCSS(tokens, mode, prefersReducedMotion), [tokens, mode, prefersReducedMotion]);\n\n const value: ThemeContextValue = useMemo(\n () => ({\n theme,\n mode,\n tokens,\n prefersReducedMotion,\n accentColor,\n setTheme,\n setMode,\n toggleMode,\n setAccentColor,\n }),\n [theme, mode, tokens, prefersReducedMotion, accentColor]\n );\n\n // Global reduced-motion rule — ensures ALL animations and transitions across\n // every Zendir component respect the OS preference (WCAG 2.3.3).\n // Uses 0.01ms (not 0) so animation-end events still fire correctly.\n const reducedMotionCSS = `@media (prefers-reduced-motion: reduce) {\n *, *::before, *::after {\n animation-duration: 0.01ms !important;\n animation-iteration-count: 1 !important;\n transition-duration: 0.01ms !important;\n scroll-behavior: auto !important;\n }\n}`;\n\n return (\n <ThemeContext.Provider value={value}>\n <style data-zendir-scrollbar=\"\" dangerouslySetInnerHTML={{ __html: scrollbarCSS }} />\n <style data-zendir-reduced-motion=\"\" dangerouslySetInnerHTML={{ __html: reducedMotionCSS }} />\n {children}\n </ThemeContext.Provider>\n );\n}\n\n/**\n * useTheme - Access theme context\n * \n * @throws Error if used outside ThemeProvider\n * \n * @example\n * ```tsx\n * const { tokens, mode, toggleMode } = useTheme();\n * ```\n */\nexport function useTheme(): ThemeContextValue {\n const context = useContext(ThemeContext);\n if (!context) {\n throw new Error('useTheme must be used within a ThemeProvider');\n }\n return context;\n}\n\n/**\n * useThemeTokens - Access only theme tokens (for memoization)\n */\nexport function useThemeTokens(): ThemeTokens {\n return useTheme().tokens;\n}\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"ThemeProvider.js","sources":["../../../src/react/theme/ThemeProvider.tsx"],"sourcesContent":["/**\n * @zendir/ui - Theme Provider\n * \n * Enterprise-grade theming system supporting Astro UX Design System and Zendir themes.\n * Full Tier 3 compliance with official Astro tokens (dark + light themes).\n * \n * Features:\n * - Multiple theme variants (Zen Hybrid default, astro, purple-hue, transparent)\n * - Light and dark modes\n * - Animation tokens with reduced-motion support\n * - Focus ring styles for accessibility\n * - CSS variable injection\n * - Local storage persistence\n */\n\nimport React, { createContext, useContext, useState, useEffect, useMemo, ReactNode } from 'react';\n// Single source of truth — see src/tokens/index.ts. Importing here instead\n// of redeclaring prevents drift bugs (the recent Roboto + fontSize-scale\n// drift, fixed in commits b4e5a5b and 9c0c32a). Anything mode-invariant\n// (typography, spacing, radius) lives in tokens/index.ts; mode-specific\n// values (colors.background, .text, .border) stay structured in\n// ThemeProvider where light/dark/variant resolution happens.\nimport {\n typography as canonicalTypography,\n spacing as canonicalSpacing,\n borderRadius as canonicalBorderRadius,\n} from '../../tokens';\n\nfunction adjustHexBrightness(hex: string, factor: number): string {\n const normalized = hex.replace('#', '').trim();\n const expanded = normalized.length === 3\n ? normalized.split('').map(ch => ch + ch).join('')\n : normalized;\n\n if (!/^[0-9a-fA-F]{6}$/.test(expanded)) {\n return hex;\n }\n\n const clamp = (value: number) => Math.max(0, Math.min(255, Math.round(value)));\n const r = clamp(parseInt(expanded.slice(0, 2), 16) * factor);\n const g = clamp(parseInt(expanded.slice(2, 4), 16) * factor);\n const b = clamp(parseInt(expanded.slice(4, 6), 16) * factor);\n\n const toHex = (value: number) => value.toString(16).padStart(2, '0');\n return `#${toHex(r)}${toHex(g)}${toHex(b)}`;\n}\n\nexport type ThemeVariant = 'astro' | 'purple-hue' | 'hybrid' | 'transparent' | 'transparent-bold' | 'transparent-minimal';\nexport type ThemeMode = 'light' | 'dark';\n\nexport interface ThemeColors {\n background: {\n base: string;\n surface: string;\n elevated: string;\n overlay: string;\n };\n border: {\n default: string;\n muted: string;\n focus: string;\n /** Transparent theme: thin faded top-to-bottom border (box-shadow, use with border: 'none') */\n fadedBoxShadow?: string;\n /** Card border style (computed from theme – solid or transparent faded). Use ...tokens.colors.border.cardStyle */\n cardStyle?: React.CSSProperties;\n /** Card border for dashed/empty state. Use ...tokens.colors.border.cardStyleDashed */\n cardStyleDashed?: React.CSSProperties;\n /** Card border using default color (e.g. NavBall, EclipseTimer). Use ...tokens.colors.border.cardStyleDefault */\n cardStyleDefault?: React.CSSProperties;\n };\n text: {\n primary: string;\n secondary: string;\n tertiary: string;\n /** Muted/placeholder text (alias for tertiary when not set) */\n muted?: string;\n inverse: string;\n };\n status: {\n normal: string;\n standby: string;\n caution: string;\n serious: string;\n critical: string;\n off: string;\n };\n semantic: {\n success: string;\n warning: string;\n error: string;\n info: string;\n };\n accent: {\n primary: string;\n secondary: string;\n tertiary: string;\n };\n interactive: {\n default: string;\n hover: string;\n active: string;\n disabled: string;\n /** Transparent theme: button/control bg (low opacity purple) when not hovered */\n transparentDefault?: string;\n /** Transparent theme: button/control bg on hover (higher opacity purple) */\n transparentHover?: string;\n /** Transparent theme: input field bg (semi-transparent purple tint) */\n transparentInputBg?: string;\n };\n}\n\nexport interface ThemeAnimation {\n /** Fast transitions (hover states) - 150ms */\n fast: string;\n /** Normal transitions (most interactions) - 250ms */\n normal: string;\n /** Slow transitions (page/mode changes) - 400ms */\n slow: string;\n /** Spring animation for playful interactions */\n spring: string;\n /** Easing function */\n easing: {\n default: string;\n in: string;\n out: string;\n inOut: string;\n };\n /** Duration values in milliseconds */\n duration: {\n instant: number;\n fast: number;\n normal: number;\n slow: number;\n };\n}\n\nexport interface ThemeFocus {\n /** Focus ring color */\n color: string;\n /** Focus ring width */\n width: string;\n /** Focus ring offset */\n offset: string;\n /** Full focus ring CSS */\n ring: string;\n /** Focus ring style object */\n style: React.CSSProperties;\n}\n\n/**\n * Astro UX Typography Style Token\n * Based on https://www.astrouxds.com/foundations/typography/\n */\nexport interface TypographyStyle {\n fontSize: string;\n fontWeight: number;\n letterSpacing: string;\n lineHeight: string;\n}\n\nexport interface ThemeTokens {\n colors: ThemeColors;\n spacing: {\n /** 2px — micro adjustments, decorative offsets */\n xxs: string;\n /** 4px — tight gaps, icon padding */\n xs: string;\n /** 8px — compact spacing, small gaps */\n sm: string;\n /** 12px — form fields, compact cards (between sm and md) */\n smd: string;\n /** 16px — default card padding, section gaps */\n md: string;\n /** 20px — slightly generous spacing */\n mdl: string;\n /** 24px — generous spacing, content area padding */\n lg: string;\n /** 32px — large section gaps */\n xl: string;\n /** 48px — extra large spacing */\n xxl: string;\n };\n borderRadius: {\n none: string;\n /** 1px — very subtle rounding */\n xs: string;\n /** 2px — small elements, badges, tooltips */\n sm: string;\n /** 4px — inputs, selects, standard controls */\n md: string;\n /** 8px — cards, containers */\n lg: string;\n /** 12px — prominent surfaces, modals */\n xl: string;\n /** 9999px — pills, fully rounded */\n full: string;\n };\n /** Standardised element heights for interactive controls */\n elementSize: {\n /** 28px — compact controls */\n sm: string;\n /** 36px — default controls */\n md: string;\n /** 44px — prominent controls, touch targets */\n lg: string;\n };\n typography: {\n fontFamily: {\n primary: string;\n /** Optional heading face. Defaults to Sora, falls back to primary. */\n heading?: string;\n mono: string;\n };\n // Astro UX Typography Tokens (rem-based)\n // https://www.astrouxds.com/foundations/typography/\n display: {\n 1: TypographyStyle;\n 2: TypographyStyle;\n };\n heading: {\n 1: TypographyStyle;\n '1Bold': TypographyStyle;\n 2: TypographyStyle;\n 3: TypographyStyle;\n 4: TypographyStyle;\n 5: TypographyStyle;\n 6: TypographyStyle;\n };\n body: {\n 1: TypographyStyle;\n '1Bold': TypographyStyle;\n 2: TypographyStyle;\n '2Bold': TypographyStyle;\n 3: TypographyStyle;\n '3Bold': TypographyStyle;\n };\n control: {\n 1: TypographyStyle;\n '1Bold': TypographyStyle;\n };\n // Legacy/convenience tokens (keeping for backward compatibility)\n fontSize: {\n micro: string; // 9px - very tight spaces\n xxs: string; // 10px - compact labels\n xs: string;\n sm: string;\n base: string;\n md: string;\n lg: string;\n xl: string;\n xxl: string;\n xxxl: string;\n };\n fontWeight: {\n light: number;\n normal: number;\n medium: number;\n semibold: number;\n bold: number;\n };\n lineHeight: {\n tight: string;\n normal: string;\n relaxed: string;\n };\n letterSpacing: {\n tight: string;\n normal: string;\n wide: string;\n };\n };\n shadows: {\n none: string;\n sm: string;\n md: string;\n lg: string;\n xl: string;\n glow: (color: string) => string;\n };\n animation: ThemeAnimation;\n focus: ThemeFocus;\n /** Semantic layout tokens for consistent spacing across cards, forms, and sections */\n layout: LayoutTokens;\n /** Semantic border tokens for consistent element styling (derived from theme colors) */\n borders: BorderTokens;\n}\n\n// ============================================================================\n// Layout Tokens Interface\n// Semantic spacing for cards, sections, forms, and nested surfaces.\n// These decouple layout intent from raw pixel values, enabling consistent\n// spacing across all consuming applications.\n// ============================================================================\n\nexport interface LayoutTokens {\n /** Card / panel container spacing */\n card: {\n /** Internal padding of card containers */\n padding: string;\n /** Gap between cards in grid layouts (desktop) */\n gap: string;\n /** Gap between cards in grid layouts (mobile / compact) */\n gapCompact: string;\n /** Card heading: icon size, title typography, gap — single system for all cards/charts */\n heading: {\n iconSize: number;\n iconSizeCompact: number;\n titleFontSize: string;\n /** Smaller title for chart headers (e.g. Power Trends) so they match card visual weight */\n titleFontSizeChart: string;\n titleFontWeight: number;\n gap: number;\n };\n };\n /** Section heading spacing (heading + divider + content) */\n section: {\n /** Padding below heading text, above the divider border */\n headerPaddingBottom: string;\n /** Margin below the divider border, before content */\n headerMarginBottom: string;\n /** Padding above the first content element after heading */\n contentPaddingTop: string;\n };\n /** Form and input field spacing */\n form: {\n /** Vertical gap between stacked form fields */\n fieldGap: string;\n /** Horizontal gap between side-by-side inline fields */\n inlineGap: string;\n /** Padding inside grouped / nested field containers */\n groupPadding: string;\n /** Gap between items within a field group */\n groupGap: string;\n /** Gap between label text and helper icons (e.g. tooltip) */\n labelGap: string;\n };\n /** Nested surface / panel spacing (e.g. grouped fields within a card) */\n surface: {\n /** Internal padding of nested surfaces */\n padding: string;\n /** Border radius of nested surfaces */\n borderRadius: string;\n /** Gap between items in compact nested surfaces */\n gap: string;\n };\n}\n\n// ============================================================================\n// Border Tokens Interface\n// Standardised border widths, element borders, focus rings, and dividers.\n// These are DERIVED from theme colors at runtime via computeBorderTokens().\n// ============================================================================\n\nexport interface BorderTokens {\n /** Border widths — standardised across all interactive elements */\n width: {\n /** Thin (1px) — inputs, cards, dividers, dropdowns */\n thin: string;\n /** Medium (1.5px) — secondary buttons, emphasised inputs */\n medium: string;\n /** Thick (2px) — checkboxes, accent borders, active tabs */\n thick: string;\n };\n /** Pre-computed input / control borders (width + style + color) */\n input: {\n /** Default resting state */\n default: string;\n /** Hover state (accent at 50% opacity) */\n hover: string;\n /** Focused state (full accent) */\n focus: string;\n /** Error / critical state */\n error: string;\n };\n /** Focus ring box-shadow values for interactive elements */\n focusRing: {\n /** Standard ring for inputs and selects */\n default: string;\n /** Ring for buttons (double-ring pattern) */\n button: string;\n /** Subtle ring for checkboxes, tabs, toggles */\n subtle: string;\n };\n /** Section divider border (headings, dialog headers/footers, tab bars) */\n divider: string;\n /** Dropdown / popover container border */\n dropdown: string;\n /** Separator line inside elements (unit separators, table rows) */\n separator: string;\n}\n\n/**\n * Derives border tokens from a theme's resolved colors.\n * Called once per theme change inside the ThemeProvider useMemo.\n */\nfunction computeBorderTokens(colors: ThemeColors): BorderTokens {\n const thin = '1px';\n const medium = '1.5px';\n const thick = '2px';\n const accent = colors.accent.primary;\n const muted = colors.border.muted;\n const critical = colors.status.critical;\n const bgBase = colors.background.base;\n\n return {\n width: { thin, medium, thick },\n input: {\n default: `${thin} solid ${muted}`,\n hover: `${thin} solid ${accent}80`,\n focus: `${thin} solid ${accent}`,\n error: `${thin} solid ${critical}`,\n },\n focusRing: {\n // WCAG 2.4.13: focus indicators ≥ 3:1 contrast — raised from 20%/30% to 50%/60%\n default: `0 0 0 3px ${accent}50, 0 0 20px ${accent}15`,\n button: `0 0 0 2px ${bgBase}, 0 0 0 4px ${accent}`,\n subtle: `0 0 0 2px ${accent}60`,\n },\n divider: `${thin} solid ${muted}`,\n dropdown: `${thin} solid ${accent}30`,\n separator: `${thin} solid ${muted}`,\n };\n}\n\n/**\n * Base theme type used for static theme definitions.\n * `borders` is omitted because it is derived from colors at runtime\n * inside the ThemeProvider useMemo (see computeBorderTokens).\n */\ntype ThemeTokensBase = Omit<ThemeTokens, 'borders'>;\n\n// ============================================================================\n// Animation Tokens (Shared)\n// ============================================================================\n\nconst animationTokens: ThemeAnimation = {\n fast: 'all 150ms cubic-bezier(0.4, 0, 0.2, 1)',\n normal: 'all 250ms cubic-bezier(0.4, 0, 0.2, 1)',\n slow: 'all 400ms cubic-bezier(0.4, 0, 0.2, 1)',\n spring: 'all 300ms cubic-bezier(0.34, 1.56, 0.64, 1)',\n easing: {\n default: 'cubic-bezier(0.4, 0, 0.2, 1)',\n in: 'ease-in',\n out: 'ease-out',\n inOut: 'ease-in-out',\n },\n duration: {\n instant: 0,\n fast: 150,\n normal: 250,\n slow: 400,\n },\n};\n\n/**\n * Reduced-motion variant of `animationTokens`. Returned by `useThemeTokens()`\n * when the user has `prefers-reduced-motion: reduce` set. CSS animations\n * are also stripped globally via the @media rule below — this ensures\n * JS-driven animations (Framer Motion, react-spring, etc.) that read\n * `tokens.animation.*` directly also collapse, instead of relying on the\n * developer to manually branch on `useTheme().prefersReducedMotion`.\n *\n * Durations are 0.01ms (not 0) so animation-end events still fire.\n */\nconst animationTokensReduced: ThemeAnimation = {\n fast: 'all 0.01ms linear',\n normal: 'all 0.01ms linear',\n slow: 'all 0.01ms linear',\n spring: 'all 0.01ms linear',\n easing: animationTokens.easing,\n duration: { instant: 0, fast: 0, normal: 0, slow: 0 },\n};\n\n// ============================================================================\n// Focus Tokens (Dark)\n// ============================================================================\n\nconst focusTokensDark: ThemeFocus = {\n color: '#4dacff',\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #4dacff',\n style: {\n outline: '2px solid #4dacff',\n outlineOffset: '2px',\n },\n};\n\nconst focusTokensLight: ThemeFocus = {\n color: '#0066cc',\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #0066cc',\n style: {\n outline: '2px solid #0066cc',\n outlineOffset: '2px',\n },\n};\n\n// ============================================================================\n// Shadow Tokens\n// ============================================================================\n\nconst shadowsDark = {\n none: 'none',\n sm: '0 1px 2px rgba(0, 0, 0, 0.3)',\n md: '0 4px 8px rgba(0, 0, 0, 0.4)',\n lg: '0 8px 16px rgba(0, 0, 0, 0.5)',\n xl: '0 16px 32px rgba(0, 0, 0, 0.6)',\n glow: (color: string) => `0 0 20px ${color}40, 0 0 40px ${color}20`,\n};\n\nconst shadowsLight = {\n none: 'none',\n sm: '0 1px 2px rgba(0, 0, 0, 0.1)',\n md: '0 4px 8px rgba(0, 0, 0, 0.15)',\n lg: '0 8px 16px rgba(0, 0, 0, 0.2)',\n xl: '0 16px 32px rgba(0, 0, 0, 0.25)',\n glow: (color: string) => `0 0 20px ${color}30, 0 0 40px ${color}15`,\n};\n\n// ============================================================================\n// Layout Tokens (Shared)\n// Semantic layout spacing based on 4px grid. Shared across all themes.\n// ============================================================================\n\nconst layoutTokens: LayoutTokens = {\n card: {\n padding: '16px', // 4 × 4px\n gap: '24px', // 6 × 4px — desktop grid gap (matches outer padding)\n gapCompact: '16px', // 4 × 4px — mobile grid gap\n heading: {\n iconSize: 18, // default card header icon (px) — matches ISS/Telemetry card header size\n iconSizeCompact: 16, // compact (e.g. chart inline header)\n titleFontSize: '1rem', // card titles (ISS, Telemetry, Mission) — reduced from 1.125rem\n titleFontSizeChart: '0.875rem', // chart titles (Power Trends) — slightly smaller\n titleFontWeight: 500,\n gap: 8, // gap between icon and title (px)\n },\n },\n section: {\n headerPaddingBottom: '8px', // 2 × 4px\n headerMarginBottom: '0px',\n contentPaddingTop: '8px', // 2 × 4px\n },\n form: {\n fieldGap: '12px', // 3 × 4px — between stacked fields\n inlineGap: '12px', // 3 × 4px — between side-by-side fields\n groupPadding: '12px', // 3 × 4px — inside grouped containers\n groupGap: '12px', // 3 × 4px — within group items\n labelGap: '3px', // label text ↔ tooltip icon\n },\n surface: {\n padding: '12px', // 3 × 4px\n borderRadius: '12px', // 3 × 4px\n gap: '8px', // 2 × 4px\n },\n};\n\n// ============================================================================\n// Astro UX Dark Theme\n// ============================================================================\n\nconst astroThemeDark: ThemeTokensBase = {\n colors: {\n background: {\n base: '#101923',\n surface: '#1b2d3e',\n elevated: '#243b53',\n overlay: 'rgba(16, 25, 35, 0.95)',\n },\n border: {\n default: '#2b659b',\n muted: '#172635',\n focus: '#4dacff',\n },\n text: {\n primary: '#ffffff',\n secondary: '#b7c5d3',\n tertiary: '#8fa4b7',\n muted: '#8fa4b7',\n inverse: '#1b2d3e',\n },\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#4dacff',\n },\n accent: {\n primary: '#4dacff',\n secondary: '#56f000',\n tertiary: '#a371f7',\n },\n interactive: {\n default: '#4dacff',\n hover: '#92cbff',\n active: '#0066cc',\n disabled: '#6b7280',\n },\n },\n // spacing + borderRadius are sourced from tokens/index.ts (single source\n // of truth — see comment at the top of this file).\n spacing: { ...canonicalSpacing },\n borderRadius: { ...canonicalBorderRadius },\n elementSize: {\n sm: '28px',\n md: '36px',\n lg: '44px',\n },\n typography: {\n /**\n * Font families — sourced from src/tokens/index.ts. Canonical stacks:\n * primary: Public Sans → Roboto → system\n * heading: Sora → Roboto → system\n * mono: Roboto Mono → system mono\n * @see https://www.astrouxds.com/foundations/typography/\n */\n fontFamily: { ...canonicalTypography.fontFamily },\n // Astro UX Display Styles\n // https://www.astrouxds.com/foundations/typography/\n display: {\n 1: { fontSize: '3.75rem', fontWeight: 300, letterSpacing: '-0.5px', lineHeight: '4.375rem' },\n 2: { fontSize: '3rem', fontWeight: 400, letterSpacing: '0', lineHeight: '3.5rem' },\n },\n // Astro UX Heading Styles\n heading: {\n 1: { fontSize: '2.125rem', fontWeight: 400, letterSpacing: '0.25px', lineHeight: '2.5rem' },\n '1Bold': { fontSize: '2.125rem', fontWeight: 700, letterSpacing: '0.25px', lineHeight: '2rem' },\n 2: { fontSize: '1.5rem', fontWeight: 400, letterSpacing: '0', lineHeight: '1.75rem' },\n 3: { fontSize: '1.25rem', fontWeight: 500, letterSpacing: '0.15px', lineHeight: '1.5rem' },\n 4: { fontSize: '1.25rem', fontWeight: 300, letterSpacing: '0.15px', lineHeight: '1.5rem' },\n 5: { fontSize: '1.125rem', fontWeight: 400, letterSpacing: '0', lineHeight: '1.5rem' },\n 6: { fontSize: '1.125rem', fontWeight: 300, letterSpacing: '0', lineHeight: '1.5rem' },\n },\n // Astro UX Body Styles\n body: {\n 1: { fontSize: '1rem', fontWeight: 400, letterSpacing: '0.5px', lineHeight: '1.5rem' },\n '1Bold': { fontSize: '1rem', fontWeight: 700, letterSpacing: '0.5px', lineHeight: '1.5rem' },\n 2: { fontSize: '0.875rem', fontWeight: 400, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n '2Bold': { fontSize: '0.875rem', fontWeight: 700, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n 3: { fontSize: '0.75rem', fontWeight: 400, letterSpacing: '0', lineHeight: '1rem' },\n '3Bold': { fontSize: '0.75rem', fontWeight: 700, letterSpacing: '0', lineHeight: '1rem' },\n },\n // Astro UX Control Body Styles\n control: {\n 1: { fontSize: '1rem', fontWeight: 400, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n '1Bold': { fontSize: '1rem', fontWeight: 700, letterSpacing: '0.5px', lineHeight: '1.25rem' },\n },\n // Convenience numeric scale — sourced from tokens/index.ts. Prefer the\n // semantically-named display/heading/body/control style objects above\n // for new code; this scale is kept for the ~300 existing components\n // that consume `tokens.typography.fontSize.*` directly.\n fontSize: { ...canonicalTypography.fontSize },\n /**\n * Font weights per AstroUXDS specification\n * Only 300, 400, 500, 700 are officially supported\n * @see https://www.astrouxds.com/foundations/typography/\n */\n fontWeight: {\n light: 300,\n normal: 400,\n medium: 500,\n /** @deprecated Use medium (500) instead - Astro spec only supports 300, 400, 500, 700 */\n semibold: 500, // Changed from 600 to 500 for Astro compliance\n bold: 700,\n },\n lineHeight: {\n tight: '1.25rem',\n normal: '1.5rem',\n relaxed: '1.75rem',\n },\n letterSpacing: {\n tight: '-0.5px',\n normal: '0',\n wide: '0.5px',\n },\n },\n shadows: shadowsDark,\n animation: animationTokens,\n focus: focusTokensDark,\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Astro UX Light Theme\n// ============================================================================\n\nconst astroThemeLight: ThemeTokensBase = {\n colors: {\n background: {\n base: '#eaeef4',\n surface: '#f5f8fc',\n elevated: '#ffffff',\n overlay: 'rgba(234, 238, 244, 0.95)',\n },\n border: {\n default: '#2b659b',\n muted: '#d1d5db',\n focus: '#0066cc',\n },\n text: {\n primary: '#1b2d3e',\n secondary: '#51667c',\n tertiary: '#556a7c', // WCAG AA: 4.81:1 on #eaeef4, 5.24:1 on #f5f8fc, 5.61:1 on #fff (was #8fa4b7 → 2.25:1 FAIL)\n muted: '#556a7c', // WCAG AA compliant (was #8fa4b7)\n inverse: '#ffffff',\n },\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#4dacff',\n },\n accent: {\n primary: '#0066cc',\n secondary: '#56f000',\n tertiary: '#7c3aed',\n },\n interactive: {\n default: '#0066cc',\n hover: '#0052a3',\n active: '#003d7a',\n disabled: '#a4abb6',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: astroThemeDark.borderRadius,\n elementSize: astroThemeDark.elementSize,\n typography: astroThemeDark.typography,\n shadows: shadowsLight,\n animation: animationTokens,\n focus: focusTokensLight,\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Hybrid Theme (Zendir Background + Astro Status + Purple Accents)\n// Modern, award-winning design with Zendir purple personality\n// ============================================================================\n\nconst focusTokensPurpleHue: ThemeFocus = {\n color: '#8b5cf6',\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #8b5cf6',\n style: {\n outline: '2px solid #8b5cf6',\n outlineOffset: '2px',\n },\n};\n\nconst hybridTheme: ThemeTokensBase = {\n colors: {\n background: {\n base: '#09090b',\n surface: 'rgba(24, 24, 27, 0.98)',\n elevated: 'rgba(39, 39, 42, 0.98)',\n overlay: 'rgba(0, 0, 0, 0.85)',\n },\n border: {\n default: 'rgba(139, 92, 246, 0.3)', // Purple tint\n muted: 'rgba(63, 63, 70, 0.5)',\n focus: '#8b5cf6', // Zendir purple\n },\n text: {\n primary: '#ffffff',\n secondary: '#a1a1aa',\n tertiary: '#8e8e96', // WCAG AA: 6.12:1 on #09090b, 5.45:1 on #18181b, 4.58:1 on #27272a (was #71717a → 4.07:1 FAIL)\n muted: '#8e8e96', // WCAG AA compliant (was #71717a)\n inverse: '#09090b',\n },\n // Astro UXD Status Colors (unchanged for compliance)\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#8b5cf6', // Purple for info\n },\n // Zendir Purple Accent Palette\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6',\n hover: '#a78bfa',\n active: '#7c3aed',\n disabled: '#52525b',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '6px',\n lg: '10px',\n xl: '14px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n glow: (color: string) => `0 0 20px ${color}50, 0 0 40px ${color}25, 0 4px 12px ${color}20`,\n },\n animation: animationTokens,\n focus: focusTokensPurpleHue,\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Purple-Hue Theme - Full Purple Personality\n// Award-winning dark theme with purple accents\n// ============================================================================\n\nconst purpleHueTheme: ThemeTokensBase = {\n colors: {\n background: {\n base: '#0a0a0f',\n surface: 'rgba(18, 18, 28, 0.98)',\n elevated: 'rgba(28, 28, 42, 0.98)',\n overlay: 'rgba(0, 0, 0, 0.9)',\n },\n border: {\n default: 'rgba(139, 92, 246, 0.25)',\n muted: 'rgba(55, 48, 82, 0.5)',\n focus: '#8b5cf6',\n },\n text: {\n primary: '#fafafa',\n secondary: '#a1a1aa',\n tertiary: '#8e8e96', // WCAG AA: 6.08:1 on #0a0a0f, 5.02:1 on elevated (was #71717a → 4.05:1 FAIL)\n muted: '#8e8e96', // WCAG AA compliant (was #71717a)\n inverse: '#0a0a0f',\n },\n // Astro UXD Status Colors (for compliance)\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#a78bfa',\n },\n // Zendir Purple Accent Palette\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6',\n hover: '#a78bfa',\n active: '#7c3aed',\n disabled: '#52525b',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '6px',\n lg: '10px',\n xl: '14px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n // Same typography as hybrid (Astro UX font stack with full fallbacks) for consistent card font across themes\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n glow: (color: string) => `0 0 25px ${color}60, 0 0 50px ${color}30, 0 4px 16px ${color}25`,\n },\n animation: animationTokens,\n focus: {\n ...focusTokensPurpleHue,\n },\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Transparent Purple-Hue Theme - Glassmorphic Panels\n// Optimized for glass card overlays with transparent backgrounds\n// ============================================================================\n\nconst transparentTheme: ThemeTokensBase = {\n colors: {\n background: {\n // Highly transparent backgrounds for glass effect\n base: 'rgba(8, 12, 20, 0.95)',\n surface: 'rgba(15, 20, 35, 0.85)',\n elevated: 'rgba(25, 35, 55, 0.80)',\n overlay: 'rgba(0, 0, 0, 0.75)',\n },\n border: {\n // Purple accent borders (Zendir brand)\n default: 'rgba(139, 92, 246, 0.25)',\n muted: 'rgba(139, 92, 246, 0.15)',\n focus: '#8b5cf6',\n // Thin faded card border with purple hint\n fadedBoxShadow: 'inset 0 1px 0 rgba(139, 92, 246, 0.12), inset 0 -1px 0 rgba(0, 0, 0, 0.05), inset 1px 0 0 rgba(139, 92, 246, 0.06), inset -1px 0 0 rgba(139, 92, 246, 0.06)',\n },\n text: {\n primary: '#f0f4f8',\n secondary: '#94a3b8',\n tertiary: '#7b8da1', // WCAG AA: 5.71:1 on base, 5.34:1 on elevated (was #64748b → 3.66:1 FAIL)\n muted: '#7b8da1', // WCAG AA compliant (was #64748b)\n inverse: '#0f172a',\n },\n // Astro UXD Status Colors\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#8b5cf6',\n },\n // Purple accent (matching hybrid theme for consistency)\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6', // Purple for interactive elements\n hover: '#a78bfa', // Light purple on hover\n active: '#7c3aed', // Deep purple on active\n disabled: '#475569',\n // Purple-hue transparent backgrounds for buttons/inputs (transparent + transparent-bold)\n transparentDefault: 'rgba(139, 92, 246, 0.12)',\n transparentHover: 'rgba(139, 92, 246, 0.32)',\n transparentInputBg: 'rgba(139, 92, 246, 0.08)',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '8px',\n lg: '12px',\n xl: '16px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n // Same typography as hybrid for consistent card font across themes\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n // Enhanced glow shadows for glass effect\n glow: (color: string) => `0 0 30px ${color}50, 0 0 60px ${color}25, 0 4px 20px rgba(0,0,0,0.4)`,\n },\n animation: animationTokens,\n focus: {\n color: '#8b5cf6', // Purple focus ring\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #8b5cf6',\n style: {\n outline: '2px solid #8b5cf6',\n outlineOffset: '2px',\n },\n },\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Transparent Minimal Theme - Ultra-clean glassmorphic panels\n// Extremely subtle borders, minimal visual noise, clean dark surfaces\n// ============================================================================\n\nconst transparentMinimalTheme: ThemeTokensBase = {\n colors: {\n background: {\n // Same as transparent theme\n base: 'rgba(8, 12, 20, 0.95)',\n surface: 'rgba(15, 20, 35, 0.85)',\n elevated: 'rgba(25, 35, 55, 0.80)',\n overlay: 'rgba(0, 0, 0, 0.75)',\n },\n border: {\n // Accent-colored borders for minimal theme (Zendir purple)\n default: 'rgba(139, 92, 246, 0.25)',\n muted: 'rgba(139, 92, 246, 0.15)',\n focus: '#8b5cf6',\n // Subtle card border using accent color\n fadedBoxShadow: 'inset 0 1px 0 rgba(139, 92, 246, 0.1), inset 0 -1px 0 rgba(0, 0, 0, 0.1), inset 1px 0 0 rgba(139, 92, 246, 0.05), inset -1px 0 0 rgba(139, 92, 246, 0.05)',\n },\n text: {\n primary: '#f0f4f8',\n secondary: '#94a3b8',\n tertiary: '#7b8da1', // WCAG AA: 5.71:1 on base, 5.34:1 on elevated (was #64748b → 3.66:1 FAIL)\n muted: '#7b8da1', // WCAG AA compliant (was #64748b)\n inverse: '#0f172a',\n },\n // Astro UXD Status Colors\n status: {\n normal: '#56f000',\n standby: '#2dccff',\n caution: '#fce83a',\n serious: '#ffb302',\n critical: '#ff3838',\n off: '#a4abb6',\n },\n semantic: {\n success: '#56f000',\n warning: '#fce83a',\n error: '#ff3838',\n info: '#8b5cf6',\n },\n // Purple accent (matching transparent theme for consistency)\n accent: {\n primary: '#8b5cf6', // Vibrant purple\n secondary: '#a78bfa', // Light purple\n tertiary: '#7c3aed', // Deep purple\n },\n interactive: {\n default: '#8b5cf6', // Purple for interactive elements\n hover: '#a78bfa', // Light purple on hover\n active: '#7c3aed', // Deep purple on active\n disabled: '#475569',\n // Purple-hue transparent backgrounds for buttons/inputs (same as transparent theme)\n transparentDefault: 'rgba(139, 92, 246, 0.12)',\n transparentHover: 'rgba(139, 92, 246, 0.32)',\n transparentInputBg: 'rgba(139, 92, 246, 0.08)',\n },\n },\n spacing: astroThemeDark.spacing,\n borderRadius: {\n none: '0',\n xs: '2px',\n sm: '4px',\n md: '8px',\n lg: '12px',\n xl: '16px',\n full: '9999px',\n },\n elementSize: astroThemeDark.elementSize,\n // Same typography as hybrid for consistent card font across themes\n typography: astroThemeDark.typography,\n shadows: {\n ...shadowsDark,\n // Same glow as transparent theme\n glow: (color: string) => `0 0 30px ${color}50, 0 0 60px ${color}25, 0 4px 20px rgba(0,0,0,0.4)`,\n },\n animation: animationTokens,\n focus: {\n color: '#8b5cf6', // Purple focus ring\n width: '2px',\n offset: '2px',\n ring: '0 0 0 2px #8b5cf6',\n style: {\n outline: '2px solid #8b5cf6',\n outlineOffset: '2px',\n },\n },\n layout: layoutTokens,\n};\n\n// ============================================================================\n// Theme Context\n// ============================================================================\n\nexport interface ThemeContextValue {\n theme: ThemeVariant;\n mode: ThemeMode;\n tokens: ThemeTokens;\n prefersReducedMotion: boolean;\n /** Custom accent color override (null = use theme default) */\n accentColor: string | null;\n setTheme: (theme: ThemeVariant) => void;\n setMode: (mode: ThemeMode) => void;\n toggleMode: () => void;\n /** Override the accent color for the current theme */\n setAccentColor: (color: string | null) => void;\n}\n\nconst ThemeContext = createContext<ThemeContextValue | undefined>(undefined);\n\nexport interface ThemeProviderProps {\n children: ReactNode;\n /** Default theme variant */\n theme?: ThemeVariant;\n /** Alias for `theme` */\n defaultVariant?: ThemeVariant;\n /** Default color mode */\n mode?: ThemeMode;\n /** Alias for `mode` */\n defaultMode?: ThemeMode;\n /** Persist preferences to localStorage */\n persist?: boolean;\n}\n\n// ============================================================================\n// Scrollbar Styling System\n// ============================================================================\n// Generates global CSS for thin, elegant, theme-aware scrollbars.\n//\n// Features:\n// - Ultra-thin (6px) scrollbar tracks with soft border radius\n// - Accent-colored thumb with hover/active brightness shifts\n// - Semi-transparent track that blends into the surface\n// - Auto-hide behaviour on modern browsers (overlay mode)\n// - Firefox support via scrollbar-width / scrollbar-color\n// - WCAG 2.1 AA compliant: sufficient contrast, respects reduced-motion\n// - Smooth transitions (disabled when prefers-reduced-motion)\n// - Scoped .zendir-scroll utility class for individual containers\n// ============================================================================\n\n/**\n * Converts a hex color to an rgba string helper.\n * Accepts 3/6/8-char hex codes (#rgb, #rrggbb, #rrggbbaa).\n */\nfunction hexToRgba(hex: string, alpha: number): string {\n const clean = hex.replace('#', '');\n let r: number, g: number, b: number;\n if (clean.length === 3) {\n r = parseInt(clean[0] + clean[0], 16);\n g = parseInt(clean[1] + clean[1], 16);\n b = parseInt(clean[2] + clean[2], 16);\n } else {\n r = parseInt(clean.slice(0, 2), 16);\n g = parseInt(clean.slice(2, 4), 16);\n b = parseInt(clean.slice(4, 6), 16);\n }\n return `rgba(${r}, ${g}, ${b}, ${alpha})`;\n}\n\nfunction buildScrollbarCSS(tokens: ThemeTokens, mode: ThemeMode, reducedMotion: boolean): string {\n const accent = tokens.colors.accent.primary;\n const _surface = tokens.colors.background.surface; // eslint-disable-line @typescript-eslint/no-unused-vars -- reserved for future scrollbar track customisation\n const _base = tokens.colors.background.base; // eslint-disable-line @typescript-eslint/no-unused-vars -- reserved for future scrollbar corner styling\n const isLight = mode === 'light';\n\n // Thumb colors: accent-tinted, low opacity at rest, stronger on hover/active\n const thumbRest = isLight ? hexToRgba(accent, 0.28) : hexToRgba(accent, 0.35);\n const thumbHover = isLight ? hexToRgba(accent, 0.48) : hexToRgba(accent, 0.55);\n const thumbActive = isLight ? hexToRgba(accent, 0.64) : hexToRgba(accent, 0.72);\n\n // Track: nearly invisible surface blend\n const trackColor = isLight ? hexToRgba(accent, 0.04) : hexToRgba(accent, 0.06);\n const trackHover = isLight ? hexToRgba(accent, 0.08) : hexToRgba(accent, 0.10);\n\n const transition = reducedMotion ? 'none' : 'background-color 200ms ease, width 200ms ease, opacity 200ms ease';\n const thumbRadius = '9999px'; // pill shape\n\n // Firefox colors (two-value syntax: thumb track)\n const ffThumb = thumbRest;\n const ffTrack = trackColor;\n\n return `\n/* ═══════════════════════════════════════════════════════════════════════════\n Zendir Scrollbar System — Thin · Elegant · Theme-Connected\n ═══════════════════════════════════════════════════════════════════════════ */\n\n/* ── Firefox ────────────────────────────────────────────────────────────── */\n*,\n.zendir-scroll {\n scrollbar-width: thin;\n scrollbar-color: ${ffThumb} ${ffTrack};\n}\n\n/* ── Webkit / Blink (Chrome, Edge, Safari, Opera) ───────────────────────── */\n\n/* Global thin scrollbar */\n::-webkit-scrollbar {\n width: 6px;\n height: 6px;\n background: transparent;\n}\n\n/* Track — nearly invisible, subtle on hover */\n::-webkit-scrollbar-track {\n background: ${trackColor};\n border-radius: ${thumbRadius};\n margin: 2px;\n}\n::-webkit-scrollbar-track:hover {\n background: ${trackHover};\n}\n\n/* Thumb — accent-tinted pill */\n::-webkit-scrollbar-thumb {\n background: ${thumbRest};\n border-radius: ${thumbRadius};\n border: 1px solid transparent;\n background-clip: padding-box;\n transition: ${transition};\n min-height: 32px;\n}\n::-webkit-scrollbar-thumb:hover {\n background: ${thumbHover};\n border-color: transparent;\n background-clip: padding-box;\n}\n::-webkit-scrollbar-thumb:active {\n background: ${thumbActive};\n border-color: transparent;\n background-clip: padding-box;\n}\n\n/* Corner — where horizontal & vertical scrollbars meet */\n::-webkit-scrollbar-corner {\n background: transparent;\n}\n\n/* ── Utility class: zendir-scroll ────────────────────────────────────────\n Apply to individual containers for scoped thin scrollbar.\n Also enables overlay behaviour where supported.\n e.g. <div className=\"zendir-scroll\" style={{overflow:'auto'}}>…</div>\n ─────────────────────────────────────────────────────────────────────── */\n.zendir-scroll {\n overflow: auto;\n overflow: overlay; /* Chrome / Edge: auto-hide scrollbar */\n}\n.zendir-scroll::-webkit-scrollbar {\n width: 5px;\n height: 5px;\n}\n.zendir-scroll::-webkit-scrollbar-track {\n background: transparent;\n border-radius: ${thumbRadius};\n}\n.zendir-scroll::-webkit-scrollbar-thumb {\n background: ${thumbRest};\n border-radius: ${thumbRadius};\n transition: ${transition};\n}\n.zendir-scroll::-webkit-scrollbar-thumb:hover {\n background: ${thumbHover};\n}\n.zendir-scroll::-webkit-scrollbar-thumb:active {\n background: ${thumbActive};\n}\n\n/* ── Utility: zendir-scroll--hidden ─────────────────────────────────────\n Hides scrollbar visually but keeps scroll behaviour (accessible).\n Content remains scrollable via mouse wheel / touch / keyboard.\n ─────────────────────────────────────────────────────────────────────── */\n.zendir-scroll--hidden {\n overflow: auto;\n -ms-overflow-style: none;\n scrollbar-width: none;\n}\n.zendir-scroll--hidden::-webkit-scrollbar {\n display: none;\n}\n\n/* ── Focus-visible: keyboard users can still scroll with focus ─────────── */\n.zendir-scroll:focus-visible {\n outline: 2px solid ${accent};\n outline-offset: -2px;\n border-radius: ${tokens.borderRadius.md};\n}\n\n/* ── Reduced Motion: disable scrollbar transitions ─────────────────────── */\n@media (prefers-reduced-motion: reduce) {\n ::-webkit-scrollbar-thumb,\n .zendir-scroll::-webkit-scrollbar-thumb {\n transition: none !important;\n }\n}\n\n/* ── High Contrast Mode: ensure scrollbar is visible ───────────────────── */\n@media (forced-colors: active) {\n ::-webkit-scrollbar-thumb {\n background: ButtonText !important;\n }\n ::-webkit-scrollbar-track {\n background: Canvas !important;\n }\n}\n`.trim();\n}\n\n/**\n * useScrollbarStyles — returns inline React.CSSProperties for a\n * scrollable container that matches the current Zendir theme.\n *\n * Use when you need programmatic control rather than the\n * `zendir-scroll` CSS class.\n *\n * @example\n * ```tsx\n * const scrollStyle = useScrollbarStyles();\n * <div style={{ maxHeight: 400, ...scrollStyle }}>…</div>\n * ```\n */\nexport function useScrollbarStyles(): React.CSSProperties {\n const { tokens, mode } = useTheme();\n return useMemo((): React.CSSProperties => {\n const accent = tokens.colors.accent.primary;\n const isLight = mode === 'light';\n const thumbColor = isLight ? hexToRgba(accent, 0.28) : hexToRgba(accent, 0.35);\n const trackColor = isLight ? hexToRgba(accent, 0.04) : hexToRgba(accent, 0.06);\n return {\n // Firefox\n scrollbarWidth: 'thin' as const,\n scrollbarColor: `${thumbColor} ${trackColor}`,\n overflow: 'auto',\n };\n }, [tokens, mode]);\n}\n\n/**\n * ThemeProvider - Enterprise theme context provider\n * \n * Provides theme tokens, mode switching, and accessibility features\n * to all child components.\n * \n * @example\n * ```tsx\n * <ThemeProvider theme=\"hybrid\" mode=\"dark\"> // Zen (Hybrid) is the default\n * <App />\n * </ThemeProvider>\n * ```\n */\nexport function ThemeProvider({\n children,\n theme: themeProp,\n defaultVariant,\n mode: modeProp,\n defaultMode: defaultModeProp,\n persist = true,\n}: ThemeProviderProps): React.ReactElement {\n const defaultTheme = themeProp ?? defaultVariant ?? 'hybrid';\n const defaultMode = modeProp ?? defaultModeProp ?? 'dark';\n // Initialize theme from localStorage or defaults\n const [theme, setThemeState] = useState<ThemeVariant>(() => {\n if (persist && typeof window !== 'undefined') {\n const saved = localStorage.getItem('zendir-ui-theme');\n if (saved === 'astro' || saved === 'zendir' || saved === 'hybrid' || saved === 'transparent' || saved === 'transparent-bold' || saved === 'transparent-minimal') {\n return saved as ThemeVariant;\n }\n }\n return defaultTheme;\n });\n\n const [mode, setModeState] = useState<ThemeMode>(() => {\n if (persist && typeof window !== 'undefined') {\n const saved = localStorage.getItem('zendir-ui-mode');\n if (saved === 'light' || saved === 'dark') {\n return saved;\n }\n }\n return defaultMode;\n });\n\n // Detect reduced motion preference\n const [prefersReducedMotion, setPrefersReducedMotion] = useState(false);\n\n useEffect(() => {\n if (typeof window === 'undefined') return;\n\n const mediaQuery = window.matchMedia('(prefers-reduced-motion: reduce)');\n setPrefersReducedMotion(mediaQuery.matches);\n\n const handler = (e: MediaQueryListEvent) => setPrefersReducedMotion(e.matches);\n mediaQuery.addEventListener('change', handler);\n return () => mediaQuery.removeEventListener('change', handler);\n }, []);\n\n // Theme setters\n const setTheme = (newTheme: ThemeVariant) => {\n setThemeState(newTheme);\n if (persist && typeof window !== 'undefined') {\n localStorage.setItem('zendir-ui-theme', newTheme);\n }\n };\n\n const setMode = (newMode: ThemeMode) => {\n setModeState(newMode);\n if (persist && typeof window !== 'undefined') {\n localStorage.setItem('zendir-ui-mode', newMode);\n }\n };\n\n const toggleMode = () => {\n setMode(mode === 'dark' ? 'light' : 'dark');\n };\n\n // Accent color override\n const [accentColor, setAccentColorState] = useState<string | null>(() => {\n if (persist && typeof window !== 'undefined') {\n return localStorage.getItem('zendir-ui-accent') || null;\n }\n return null;\n });\n\n const setAccentColor = (color: string | null) => {\n setAccentColorState(color);\n if (persist && typeof window !== 'undefined') {\n if (color) {\n localStorage.setItem('zendir-ui-accent', color);\n } else {\n localStorage.removeItem('zendir-ui-accent');\n }\n }\n };\n\n // Get theme tokens (card border styles computed so all cards use theme-driven borders)\n const tokens = useMemo((): ThemeTokens => {\n // If light mode is selected, always use Astro Light theme (unified light mode)\n // regardless of the selected theme variant\n const base =\n mode === 'light'\n ? astroThemeLight\n : theme === 'astro'\n ? astroThemeDark\n : theme === 'hybrid'\n ? hybridTheme\n : theme === 'transparent' || theme === 'transparent-bold'\n ? transparentTheme\n : theme === 'transparent-minimal'\n ? transparentMinimalTheme\n : purpleHueTheme;\n const isTransparent =\n theme === 'transparent' || theme === 'transparent-bold' || theme === 'transparent-minimal';\n const isMinimal = theme === 'transparent-minimal';\n \n // For transparent-minimal: use visible accent-colored borders.\n // For other transparent themes: use subtle fadedBoxShadow.\n // Accent overrides are intentionally scoped to transparent variants only,\n // so switching to Astro/Hybrid/Purple-Hue restores each theme's native accent.\n const resolvedAccent = (isTransparent && accentColor)\n ? {\n primary: accentColor,\n secondary: adjustHexBrightness(accentColor, 1.2),\n tertiary: adjustHexBrightness(accentColor, 0.85),\n }\n : base.colors.accent;\n\n const accentPrimary = resolvedAccent.primary;\n const accentBorderMuted = `${accentPrimary}40`; // 25% opacity\n const accentBorderDefault = `${accentPrimary}66`; // 40% opacity\n const accentFadedBoxShadow = `inset 0 1px 0 ${hexToRgba(accentPrimary, 0.12)}, inset 0 -1px 0 rgba(0, 0, 0, 0.05), inset 1px 0 0 ${hexToRgba(accentPrimary, 0.06)}, inset -1px 0 0 ${hexToRgba(accentPrimary, 0.06)}`;\n \n const cardStyle = isMinimal\n ? { border: `1px solid ${accentBorderMuted}` }\n : isTransparent\n ? { border: 'none' as const, boxShadow: accentFadedBoxShadow }\n : { border: `1px solid ${base.colors.border.muted}` };\n const cardStyleDashed = isMinimal\n ? { border: `1px dashed ${accentBorderMuted}` }\n : isTransparent\n ? { border: 'none' as const, boxShadow: accentFadedBoxShadow }\n : { border: `1px dashed ${base.colors.border.muted}` };\n const cardStyleDefault = isMinimal\n ? { border: `1px solid ${accentBorderDefault}` }\n : isTransparent\n ? { border: 'none' as const, boxShadow: accentFadedBoxShadow }\n : { border: `1px solid ${base.colors.border.default}` };\n // Compute border tokens from the resolved theme colors\n const resolvedColors: ThemeColors = {\n ...base.colors,\n accent: resolvedAccent,\n border: {\n ...base.colors.border,\n cardStyle,\n cardStyleDashed,\n cardStyleDefault,\n },\n };\n\n return {\n ...base,\n colors: resolvedColors,\n borders: computeBorderTokens(resolvedColors),\n // Honor prefers-reduced-motion in the JS token surface too. The\n // global CSS @media rule below already strips CSS animations; this\n // ensures JS-driven animation libraries (Framer Motion etc.) that\n // read `tokens.animation.*` get the collapsed variant without each\n // developer having to branch on `useTheme().prefersReducedMotion`.\n animation: prefersReducedMotion ? animationTokensReduced : base.animation,\n } as ThemeTokens;\n }, [theme, mode, accentColor, prefersReducedMotion]);\n\n // Apply CSS variables to document\n useEffect(() => {\n if (typeof document === 'undefined') return;\n\n const root = document.documentElement;\n const { colors, typography, spacing, borderRadius, animation, focus, shadows } = tokens;\n\n // Theme mode attribute\n root.setAttribute('data-theme', mode);\n root.setAttribute('data-variant', theme);\n\n // Background colors\n root.style.setProperty('--color-background-base', colors.background.base);\n root.style.setProperty('--color-background-surface', colors.background.surface);\n root.style.setProperty('--color-background-elevated', colors.background.elevated);\n root.style.setProperty('--color-background-overlay', colors.background.overlay);\n\n // Border colors\n root.style.setProperty('--color-border', colors.border.default);\n root.style.setProperty('--color-border-muted', colors.border.muted);\n root.style.setProperty('--color-border-focus', colors.border.focus);\n\n // Text colors\n root.style.setProperty('--color-text-primary', colors.text.primary);\n root.style.setProperty('--color-text-secondary', colors.text.secondary);\n root.style.setProperty('--color-text-tertiary', colors.text.tertiary);\n root.style.setProperty('--color-text-inverse', colors.text.inverse);\n\n // Status colors\n root.style.setProperty('--color-status-normal', colors.status.normal);\n root.style.setProperty('--color-status-standby', colors.status.standby);\n root.style.setProperty('--color-status-caution', colors.status.caution);\n root.style.setProperty('--color-status-serious', colors.status.serious);\n root.style.setProperty('--color-status-critical', colors.status.critical);\n root.style.setProperty('--color-status-off', colors.status.off);\n\n // Interactive colors\n root.style.setProperty('--color-interactive-default', colors.interactive.default);\n root.style.setProperty('--color-interactive-hover', colors.interactive.hover);\n root.style.setProperty('--color-interactive-active', colors.interactive.active);\n root.style.setProperty('--color-interactive-disabled', colors.interactive.disabled);\n\n // Accent colors\n root.style.setProperty('--color-accent-primary', colors.accent.primary);\n root.style.setProperty('--color-accent-secondary', colors.accent.secondary);\n root.style.setProperty('--color-accent-tertiary', colors.accent.tertiary);\n\n // Typography. `heading` is optional on the type — fall back to `primary`\n // when the active theme doesn't define a separate heading face.\n root.style.setProperty('--font-family-heading', typography.fontFamily.heading || typography.fontFamily.primary);\n root.style.setProperty('--font-family-primary', typography.fontFamily.primary);\n root.style.setProperty('--font-family-mono', typography.fontFamily.mono);\n\n // Spacing (full scale so CSS and layout can pull from SDK)\n root.style.setProperty('--spacing-xxs', spacing.xxs);\n root.style.setProperty('--spacing-xs', spacing.xs);\n root.style.setProperty('--spacing-sm', spacing.sm);\n root.style.setProperty('--spacing-smd', spacing.smd);\n root.style.setProperty('--spacing-md', spacing.md);\n root.style.setProperty('--spacing-mdl', spacing.mdl);\n root.style.setProperty('--spacing-lg', spacing.lg);\n root.style.setProperty('--spacing-xl', spacing.xl);\n root.style.setProperty('--spacing-xxl', spacing.xxl);\n\n // Border radius\n root.style.setProperty('--radius-sm', borderRadius.sm);\n root.style.setProperty('--radius-md', borderRadius.md);\n root.style.setProperty('--radius-lg', borderRadius.lg);\n root.style.setProperty('--radius-xl', borderRadius.xl);\n\n // Animation (respect reduced motion)\n const motionMultiplier = prefersReducedMotion ? 0 : 1;\n root.style.setProperty('--animation-fast', prefersReducedMotion ? 'none' : animation.fast);\n root.style.setProperty('--animation-normal', prefersReducedMotion ? 'none' : animation.normal);\n root.style.setProperty('--animation-slow', prefersReducedMotion ? 'none' : animation.slow);\n root.style.setProperty('--duration-fast', `${animation.duration.fast * motionMultiplier}ms`);\n root.style.setProperty('--duration-normal', `${animation.duration.normal * motionMultiplier}ms`);\n root.style.setProperty('--duration-slow', `${animation.duration.slow * motionMultiplier}ms`);\n\n // Focus\n root.style.setProperty('--focus-ring-color', focus.color);\n root.style.setProperty('--focus-ring-width', focus.width);\n root.style.setProperty('--focus-ring-offset', focus.offset);\n\n // Shadows\n root.style.setProperty('--shadow-sm', shadows.sm);\n root.style.setProperty('--shadow-md', shadows.md);\n root.style.setProperty('--shadow-lg', shadows.lg);\n root.style.setProperty('--shadow-xl', shadows.xl);\n\n // Apply base styles to body\n document.body.style.backgroundColor = colors.background.base;\n document.body.style.color = colors.text.primary;\n document.body.style.fontFamily = typography.fontFamily.primary;\n }, [tokens, theme, mode, prefersReducedMotion]);\n\n // ──────────────────────────────────────────────────────────────────────────\n // Scrollbar Styles — thin, elegant, theme-connected, fully accessible\n // ──────────────────────────────────────────────────────────────────────────\n const scrollbarCSS = useMemo(() => buildScrollbarCSS(tokens, mode, prefersReducedMotion), [tokens, mode, prefersReducedMotion]);\n\n const value: ThemeContextValue = useMemo(\n () => ({\n theme,\n mode,\n tokens,\n prefersReducedMotion,\n accentColor,\n setTheme,\n setMode,\n toggleMode,\n setAccentColor,\n }),\n [theme, mode, tokens, prefersReducedMotion, accentColor]\n );\n\n // Global reduced-motion rule — ensures ALL animations and transitions across\n // every Zendir component respect the OS preference (WCAG 2.3.3).\n // Uses 0.01ms (not 0) so animation-end events still fire correctly.\n const reducedMotionCSS = `@media (prefers-reduced-motion: reduce) {\n *, *::before, *::after {\n animation-duration: 0.01ms !important;\n animation-iteration-count: 1 !important;\n transition-duration: 0.01ms !important;\n scroll-behavior: auto !important;\n }\n}`;\n\n return (\n <ThemeContext.Provider value={value}>\n <style data-zendir-scrollbar=\"\" dangerouslySetInnerHTML={{ __html: scrollbarCSS }} />\n <style data-zendir-reduced-motion=\"\" dangerouslySetInnerHTML={{ __html: reducedMotionCSS }} />\n {children}\n </ThemeContext.Provider>\n );\n}\n\n/**\n * useTheme - Access theme context\n * \n * @throws Error if used outside ThemeProvider\n * \n * @example\n * ```tsx\n * const { tokens, mode, toggleMode } = useTheme();\n * ```\n */\nexport function useTheme(): ThemeContextValue {\n const context = useContext(ThemeContext);\n if (!context) {\n throw new Error('useTheme must be used within a ThemeProvider');\n }\n return context;\n}\n\n/**\n * useThemeTokens - Access only theme tokens (for memoization)\n */\nexport function useThemeTokens(): ThemeTokens {\n return useTheme().tokens;\n}\n"],"names":["canonicalSpacing","canonicalBorderRadius","canonicalTypography","typography","spacing","borderRadius"],"mappings":"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"}
|
package/dist/react/types.d.ts
CHANGED
|
@@ -1,10 +1,15 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* @zendir/ui - Local Type Definitions
|
|
3
3
|
*
|
|
4
|
-
*
|
|
5
|
-
*
|
|
4
|
+
* UI-domain types (`Quaternion`, `Vector3D`, `PointingMode`, etc.) used by
|
|
5
|
+
* the visualization components. These live in `@zendir/ui` because they
|
|
6
|
+
* describe what the UI renders — they are NOT shipped by `zendir-ts`,
|
|
7
|
+
* which is REST-only and intentionally minimal.
|
|
6
8
|
*
|
|
7
|
-
*
|
|
9
|
+
* Helpers like `estimateOrbitalPeriod` / `estimateOrbitalVelocity` are
|
|
10
|
+
* pure client-side approximations used for quick UI estimates; for
|
|
11
|
+
* physics-accurate values, fetch the real properties from the engine
|
|
12
|
+
* via `client.getProperties(...)`.
|
|
8
13
|
*/
|
|
9
14
|
/**
|
|
10
15
|
* A named, colored data category — the SDK's shared concept for any secondary
|
|
@@ -80,6 +85,30 @@ export interface GroundTrackPoint {
|
|
|
80
85
|
altitude: number;
|
|
81
86
|
timestamp: string;
|
|
82
87
|
}
|
|
88
|
+
/**
|
|
89
|
+
* Developer-defined toggle in a Layers panel. The component that owns the
|
|
90
|
+
* panel only renders the toggle UI — it does NOT render the layer's
|
|
91
|
+
* content. The consumer app listens to the panel's `onChange` and
|
|
92
|
+
* renders/hides its own overlay accordingly.
|
|
93
|
+
*
|
|
94
|
+
* Used by both `<GroundTrackMap />` (2D map overlays) and the upcoming
|
|
95
|
+
* `<LayerControlPanel />` for the 3D viewer.
|
|
96
|
+
*/
|
|
97
|
+
export interface LayerDef {
|
|
98
|
+
/** Unique identifier (e.g. 'heatmap', 'coverage', 'spacecraft'). */
|
|
99
|
+
id: string;
|
|
100
|
+
/** Display label in the Layers panel. */
|
|
101
|
+
label: string;
|
|
102
|
+
/** Whether the layer is on by default (default true). */
|
|
103
|
+
defaultEnabled?: boolean;
|
|
104
|
+
/** Optional grouping bucket for the panel ('overlays', 'objects', etc.). */
|
|
105
|
+
category?: string;
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* @deprecated Use `LayerDef`. Alias kept for backward compatibility with
|
|
109
|
+
* existing `<GroundTrackMap />` consumers.
|
|
110
|
+
*/
|
|
111
|
+
export type MapLayerDef = LayerDef;
|
|
83
112
|
export interface AccessData {
|
|
84
113
|
stationId: string;
|
|
85
114
|
stationName: string;
|
package/dist/react/types.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.js","sources":["../../src/react/types.ts"],"sourcesContent":["/**\n * @zendir/ui - Local Type Definitions\n * \n * These types are provided so the UI library can be used without @zendir/sdk.\n * If you have @zendir/sdk installed, those types will be used instead.\n * \n * All types here mirror the @zendir/sdk interfaces.\n */\n\n// =============================================================================\n// Category / Group Primitive\n// =============================================================================\n\n/**\n * A named, colored data category — the SDK's shared concept for any secondary\n * grouping dimension (teams, squads, subsystems, mission phases, etc.).\n *\n * Components that support category-aware coloring accept `CategoryDef` arrays\n * and/or a `categoryId` field on individual data items. The `CategoryPalette`\n * utility resolves ids → colors/labels consistently across an entire dashboard.\n *\n * @example\n * const categories: CategoryDef[] = [\n * { id: 'alpha', label: 'Team Alpha', color: '#2dccff' },\n * { id: 'bravo', label: 'Team Bravo', color: '#fce83a' },\n * ];\n */\nexport interface CategoryDef {\n /** Stable unique identifier (string or numeric) */\n id: string | number;\n /** Human-readable display name */\n label: string;\n /** CSS color used for visual accents (stripes, rings, swatches, series lines) */\n color: string;\n /** Optional icon name from the Astro icon set */\n icon?: string;\n /** Arbitrary metadata for app-specific needs */\n meta?: Record<string, unknown>;\n}\n\n// =============================================================================\n// Spacecraft Types\n// =============================================================================\n\nexport interface SpacecraftPosition {\n /** Optional spacecraft id (when combined with Spacecraft) */\n id?: string;\n /** Optional display name */\n name?: string;\n /** Latitude in degrees (-90 to 90) */\n latitude: number;\n /** Longitude in degrees (-180 to 180) */\n longitude: number;\n /** Altitude in kilometers */\n altitude: number;\n /** Velocity in km/s */\n velocity?: number;\n /** Heading in degrees */\n heading?: number;\n /** Timestamp */\n timestamp?: string;\n /** Orbit type (e.g. LEO, GEO) */\n orbitType?: string;\n /** Orbital inclination in degrees */\n inclination?: number;\n /** Health/operational status using AstroUXDS 6-level system */\n status?: 'off' | 'standby' | 'normal' | 'caution' | 'serious' | 'critical';\n}\n\nexport interface Spacecraft {\n id: string;\n name: string;\n noradId?: number;\n type?: string;\n /** Status using AstroUXDS 6-level system */\n status?: 'off' | 'standby' | 'normal' | 'caution' | 'serious' | 'critical';\n}\n\n// =============================================================================\n// Ground Station Types\n// =============================================================================\n\nexport interface GroundStation {\n id: string;\n name: string;\n latitude: number;\n longitude: number;\n elevation?: number;\n /** Minimum elevation angle for visibility (degrees) */\n minElevation?: number;\n network?: string;\n}\n\nexport interface GroundTrackPoint {\n latitude: number;\n longitude: number;\n altitude: number;\n timestamp: string;\n}\n\n// =============================================================================\n// Access / Contact Types\n// =============================================================================\n\nexport interface AccessData {\n stationId: string;\n stationName: string;\n spacecraftId: string;\n aos: string; // Acquisition of Signal - ISO timestamp\n los: string; // Loss of Signal - ISO timestamp\n maxElevation: number;\n azimuthAos?: number;\n azimuthLos?: number;\n /** Whether link is currently accessible */\n isAccessible?: boolean;\n /** Current elevation angle (degrees) */\n elevation?: number;\n /** Current azimuth (degrees) */\n azimuth?: number;\n /** Slant range in km */\n slantRange?: number;\n /** Signal strength 0-100 */\n signalStrength?: number;\n /** Link budget details */\n linkBudget?: {\n receivedPower?: number;\n snr?: number;\n dataRate?: number;\n };\n}\n\nexport interface AccessWindow {\n aos: string;\n los: string;\n maxElevation: number;\n stationId: string;\n /** Start time (ISO or ms) */\n startTime?: string | number;\n /** Duration in seconds */\n duration?: number;\n}\n\n// =============================================================================\n// Telemetry Types\n// =============================================================================\n\nexport interface TelemetryData {\n timestamp: string;\n subsystems: {\n power?: {\n status: string;\n batteryLevel?: number;\n solarArrayOutput?: number;\n voltage?: number;\n solarGeneration?: number;\n consumption?: number;\n };\n thermal?: {\n status: string;\n cpuTemp?: number;\n batteryTemp?: number;\n };\n comms?: {\n status: string;\n signalStrength?: number;\n dataRate?: number;\n downlinkRate?: number;\n transmitterStatus?: string;\n packetsQueued?: number;\n bytesTransmitted?: number;\n packetsTransmitted?: number;\n };\n attitude?: {\n status: string;\n pointingError?: number;\n eulerAngles?: EulerAngles;\n targetMode?: string;\n };\n };\n /** Health status (convenience alias / extended) */\n health?: {\n status?: string;\n anomalies?: Array<{ id: string; message: string; severity?: string }>;\n };\n /** Power (convenience alias for subsystems.power / extended) */\n power?: {\n status?: string;\n batteryLevel?: number;\n solarArrayOutput?: number;\n voltage?: number;\n solarGeneration?: number;\n consumption?: number;\n };\n /** Attitude (convenience alias / extended) */\n attitude?: {\n status?: string;\n eulerAngles?: EulerAngles;\n targetMode?: string;\n pointingError?: number;\n };\n /** Communications (convenience alias / extended) */\n communications?: {\n status?: string;\n signalStrength?: number;\n dataRate?: number;\n downlinkRate?: number;\n transmitterStatus?: string;\n packetsQueued?: number;\n bytesTransmitted?: number;\n packetsTransmitted?: number;\n };\n alerts?: Array<{\n id: string;\n severity: string;\n message: string;\n timestamp: string;\n }>;\n}\n\n// =============================================================================\n// Orbital Types\n// =============================================================================\n\nexport interface OrbitalElements {\n semiMajorAxis: number;\n eccentricity: number;\n inclination: number;\n raan: number; // Right Ascension of Ascending Node\n argumentOfPerigee: number;\n trueAnomaly: number;\n period?: number;\n epoch?: string;\n}\n\n// =============================================================================\n// Attitude Types\n// =============================================================================\n\nexport interface Quaternion {\n x: number;\n y: number;\n z: number;\n w: number;\n}\n\nexport interface EulerAngles {\n roll: number;\n pitch: number;\n yaw: number;\n}\n\nexport interface AngularVelocity {\n x: number;\n y: number;\n z: number;\n}\n\nexport interface AttitudeData {\n quaternion: Quaternion;\n eulerAngles: EulerAngles;\n angularVelocity: AngularVelocity;\n targetMode?: string;\n /** Pointing error in degrees */\n pointingError?: number;\n /** Target direction vector [x, y, z] */\n targetDirection?: [number, number, number];\n}\n\nexport type PointingMode = 'nadir' | 'sun' | 'target' | 'inertial' | 'velocity';\n\n// =============================================================================\n// Eclipse Types\n// =============================================================================\n\nexport interface EclipseInfo {\n inEclipse: boolean;\n timeToEclipse: number; // seconds\n eclipseDuration: number; // seconds\n sunlightDuration: number; // seconds\n /** Umbra duration in seconds */\n umbraDuration?: number;\n /** Penumbra duration in seconds */\n penumbraDuration?: number;\n /** Seconds until sunlight (when in eclipse) */\n timeToSunlight?: number;\n}\n\n// =============================================================================\n// Link Budget Types\n// =============================================================================\n\nexport interface DetailedLinkBudget {\n spacecraftId: string;\n groundStationId: string;\n frequency: number;\n eirp: number;\n pathLoss: number;\n atmosphericLoss: number;\n receiverGain: number;\n systemNoiseTemp: number;\n cnoRequired: number;\n cnoActual: number;\n margin: number;\n dataRate: number;\n}\n\n// =============================================================================\n// Thermal Types\n// =============================================================================\n\nexport interface ThermalZone {\n id: string;\n name: string;\n temperature: number;\n minLimit: number;\n maxLimit: number;\n}\n\nexport interface ThermalData {\n timestamp: string;\n zones: ThermalZone[];\n averageTemp: number;\n hottest: { zone: string; temp: number };\n coldest: { zone: string; temp: number };\n}\n\n// =============================================================================\n// Propulsion Types\n// =============================================================================\n\nexport interface ThrusterStatus {\n id: string;\n name: string;\n status: 'ready' | 'firing' | 'disabled' | 'error';\n fuelFlow?: number;\n}\n\nexport interface PropulsionSummary {\n fuelRemaining: number;\n fuelCapacity: number;\n thrusterStatus: ThrusterStatus[];\n deltaVRemaining: number;\n deltaVUsed: number;\n lastManeuver?: string;\n}\n\n// =============================================================================\n// Reaction Wheel / Momentum Types\n// =============================================================================\n\nexport interface ReactionWheelData {\n /** Unique wheel identifier */\n id: string;\n /** Display name (e.g. \"RW-X\", \"RW-1\") */\n name: string;\n /** Mounted axis */\n axis: 'X' | 'Y' | 'Z' | 'skew';\n /** Current stored angular momentum in N·m·s */\n momentumNms: number;\n /** Maximum momentum capacity in N·m·s */\n maxMomentumNms: number;\n /** Current wheel speed in RPM */\n speedRpm: number;\n /** Maximum wheel speed in RPM */\n maxSpeedRpm: number;\n /** Operational status */\n status: 'nominal' | 'saturated' | 'desaturating' | 'off' | 'error';\n /** Power consumption in Watts */\n powerW?: number;\n /** Bearing temperature in °C */\n temperatureC?: number;\n}\n\n/** LVLH (Local Vertical Local Horizontal) frame vector */\nexport interface LVLHVector {\n /** Radial component (local vertical, positive away from Earth) */\n radial: number;\n /** In-track component (along velocity vector) */\n inTrack: number;\n /** Cross-track component (normal to orbital plane) */\n crossTrack: number;\n}\n\n/** Time-stamped LVLH state for trajectory plotting */\nexport interface LVLHState {\n /** Timestamp in milliseconds */\n time: number;\n /** Position in LVLH frame (km) */\n position: LVLHVector;\n /** Velocity in LVLH frame (km/s) */\n velocity: LVLHVector;\n}\n\n/** Thruster firing event */\nexport interface ThrusterFireEvent {\n /** Event timestamp in milliseconds */\n time: number;\n /** Thruster ID */\n thrusterId: string;\n /** Display name */\n name: string;\n /** Delta-V magnitude in m/s */\n deltaVMs: number;\n /** Delta-V direction in LVLH frame (unit vector) */\n deltaVDirection?: LVLHVector;\n /** Burn duration in seconds */\n durationSeconds: number;\n /** Event type */\n type: 'impulsive' | 'finite' | 'continuous';\n}\n\n// =============================================================================\n// Planet Types\n// =============================================================================\n\nexport type PlanetId = \n | 'sun' \n | 'mercury' \n | 'venus' \n | 'earth' \n | 'moon' \n | 'mars' \n | 'jupiter' \n | 'saturn' \n | 'uranus' \n | 'neptune' \n | 'pluto';\n\nexport interface PlanetInfo {\n id: PlanetId;\n name: string;\n radius: number; // km\n /** @deprecated Use radius */\n radiusKm?: number;\n mass: number; // kg\n semiMajorAxis: number; // AU\n /** Distance from Sun in AU (alias for semiMajorAxis) */\n distanceFromSunAU?: number;\n orbitalPeriod: number; // Earth days\n color: string;\n}\n\n// =============================================================================\n// Utility Functions (fallbacks when SDK not available)\n// =============================================================================\n\n/**\n * Estimate orbital period from altitude (simplified)\n */\nexport function estimateOrbitalPeriod(altitudeKm: number): number {\n const earthRadiusKm = 6371;\n const mu = 398600.4418; // km³/s² - Earth's gravitational parameter\n const semiMajorAxis = earthRadiusKm + altitudeKm;\n const period = 2 * Math.PI * Math.sqrt(Math.pow(semiMajorAxis, 3) / mu);\n return period / 60; // Return in minutes\n}\n\n/**\n * Estimate orbital velocity from altitude (simplified)\n */\nexport function estimateOrbitalVelocity(altitudeKm: number): number {\n const earthRadiusKm = 6371;\n const mu = 398600.4418; // km³/s²\n const radius = earthRadiusKm + altitudeKm;\n return Math.sqrt(mu / radius); // km/s\n}\n\n/**\n * Convert AU to kilometers\n */\nexport function auToKm(au: number): number {\n return au * 149597870.7;\n}\n\n/**\n * Normalize planet name to PlanetId\n */\nexport function normalizePlanetName(name: string): PlanetId | null {\n const normalized = name.toLowerCase().trim();\n const validPlanets: PlanetId[] = [\n 'sun', 'mercury', 'venus', 'earth', 'moon', \n 'mars', 'jupiter', 'saturn', 'uranus', 'neptune', 'pluto'\n ];\n return validPlanets.includes(normalized as PlanetId) \n ? (normalized as PlanetId) \n : null;\n}\n\n/**\n * Get planet info by ID\n */\nexport function getPlanet(id: PlanetId): PlanetInfo | undefined {\n return PLANETS.find(p => p.id === id);\n}\n\n/**\n * Planet data (simplified)\n */\nexport const PLANETS: PlanetInfo[] = [\n { id: 'sun', name: 'Sun', radius: 696340, mass: 1.989e30, semiMajorAxis: 0, orbitalPeriod: 0, color: '#ffcc00' },\n { id: 'mercury', name: 'Mercury', radius: 2439.7, mass: 3.3011e23, semiMajorAxis: 0.387, orbitalPeriod: 88, color: '#b5b5b5' },\n { id: 'venus', name: 'Venus', radius: 6051.8, mass: 4.8675e24, semiMajorAxis: 0.723, orbitalPeriod: 225, color: '#e6c87a' },\n { id: 'earth', name: 'Earth', radius: 6371, mass: 5.972e24, semiMajorAxis: 1.0, orbitalPeriod: 365, color: '#4da6ff' },\n { id: 'moon', name: 'Moon', radius: 1737.4, mass: 7.342e22, semiMajorAxis: 0.00257, orbitalPeriod: 27.3, color: '#c0c0c0' },\n { id: 'mars', name: 'Mars', radius: 3389.5, mass: 6.4171e23, semiMajorAxis: 1.524, orbitalPeriod: 687, color: '#e07b4a' },\n { id: 'jupiter', name: 'Jupiter', radius: 69911, mass: 1.8982e27, semiMajorAxis: 5.203, orbitalPeriod: 4333, color: '#d8ca9d' },\n { id: 'saturn', name: 'Saturn', radius: 58232, mass: 5.6834e26, semiMajorAxis: 9.537, orbitalPeriod: 10759, color: '#f4d59e' },\n { id: 'uranus', name: 'Uranus', radius: 25362, mass: 8.6810e25, semiMajorAxis: 19.19, orbitalPeriod: 30687, color: '#b3e0e6' },\n { id: 'neptune', name: 'Neptune', radius: 24622, mass: 1.02413e26, semiMajorAxis: 30.07, orbitalPeriod: 60190, color: '#5b7fff' },\n { id: 'pluto', name: 'Pluto', radius: 1188.3, mass: 1.303e22, semiMajorAxis: 39.48, orbitalPeriod: 90560, color: '#c9b79c' },\n];\n"],"names":[],"mappings":"AAicO,SAAS,sBAAsB,YAA4B;AAChE,QAAM,gBAAgB;AACtB,QAAM,KAAK;AACX,QAAM,gBAAgB,gBAAgB;AACtC,QAAM,SAAS,IAAI,KAAK,KAAK,KAAK,KAAK,KAAK,IAAI,eAAe,CAAC,IAAI,EAAE;AACtE,SAAO,SAAS;AAClB;AAKO,SAAS,wBAAwB,YAA4B;AAClE,QAAM,gBAAgB;AACtB,QAAM,KAAK;AACX,QAAM,SAAS,gBAAgB;AAC/B,SAAO,KAAK,KAAK,KAAK,MAAM;AAC9B;AAKO,SAAS,OAAO,IAAoB;AACzC,SAAO,KAAK;AACd;AAKO,SAAS,oBAAoB,MAA+B;AACjE,QAAM,aAAa,KAAK,YAAA,EAAc,KAAA;AACtC,QAAM,eAA2B;AAAA,IAC/B;AAAA,IAAO;AAAA,IAAW;AAAA,IAAS;AAAA,IAAS;AAAA,IACpC;AAAA,IAAQ;AAAA,IAAW;AAAA,IAAU;AAAA,IAAU;AAAA,IAAW;AAAA,EAAA;AAEpD,SAAO,aAAa,SAAS,UAAsB,IAC9C,aACD;AACN;AAKO,SAAS,UAAU,IAAsC;AAC9D,SAAO,QAAQ,KAAK,CAAA,MAAK,EAAE,OAAO,EAAE;AACtC;AAKO,MAAM,UAAwB;AAAA,EACnC,EAAE,IAAI,OAAO,MAAM,OAAO,QAAQ,QAAQ,MAAM,SAAU,eAAe,GAAG,eAAe,GAAG,OAAO,UAAA;AAAA,EACrG,EAAE,IAAI,WAAW,MAAM,WAAW,QAAQ,QAAQ,MAAM,UAAW,eAAe,OAAO,eAAe,IAAI,OAAO,UAAA;AAAA,EACnH,EAAE,IAAI,SAAS,MAAM,SAAS,QAAQ,QAAQ,MAAM,UAAW,eAAe,OAAO,eAAe,KAAK,OAAO,UAAA;AAAA,EAChH,EAAE,IAAI,SAAS,MAAM,SAAS,QAAQ,MAAM,MAAM,SAAU,eAAe,GAAK,eAAe,KAAK,OAAO,UAAA;AAAA,EAC3G,EAAE,IAAI,QAAQ,MAAM,QAAQ,QAAQ,QAAQ,MAAM,SAAU,eAAe,QAAS,eAAe,MAAM,OAAO,UAAA;AAAA,EAChH,EAAE,IAAI,QAAQ,MAAM,QAAQ,QAAQ,QAAQ,MAAM,UAAW,eAAe,OAAO,eAAe,KAAK,OAAO,UAAA;AAAA,EAC9G,EAAE,IAAI,WAAW,MAAM,WAAW,QAAQ,OAAO,MAAM,UAAW,eAAe,OAAO,eAAe,MAAM,OAAO,UAAA;AAAA,EACpH,EAAE,IAAI,UAAU,MAAM,UAAU,QAAQ,OAAO,MAAM,UAAW,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AAAA,EACnH,EAAE,IAAI,UAAU,MAAM,UAAU,QAAQ,OAAO,MAAM,SAAW,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AAAA,EACnH,EAAE,IAAI,WAAW,MAAM,WAAW,QAAQ,OAAO,MAAM,WAAY,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AAAA,EACtH,EAAE,IAAI,SAAS,MAAM,SAAS,QAAQ,QAAQ,MAAM,SAAU,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AACnH;"}
|
|
1
|
+
{"version":3,"file":"types.js","sources":["../../src/react/types.ts"],"sourcesContent":["/**\r\n * @zendir/ui - Local Type Definitions\r\n *\r\n * UI-domain types (`Quaternion`, `Vector3D`, `PointingMode`, etc.) used by\r\n * the visualization components. These live in `@zendir/ui` because they\r\n * describe what the UI renders — they are NOT shipped by `zendir-ts`,\r\n * which is REST-only and intentionally minimal.\r\n *\r\n * Helpers like `estimateOrbitalPeriod` / `estimateOrbitalVelocity` are\r\n * pure client-side approximations used for quick UI estimates; for\r\n * physics-accurate values, fetch the real properties from the engine\r\n * via `client.getProperties(...)`.\r\n */\r\n\r\n// =============================================================================\r\n// Category / Group Primitive\r\n// =============================================================================\r\n\r\n/**\r\n * A named, colored data category — the SDK's shared concept for any secondary\r\n * grouping dimension (teams, squads, subsystems, mission phases, etc.).\r\n *\r\n * Components that support category-aware coloring accept `CategoryDef` arrays\r\n * and/or a `categoryId` field on individual data items. The `CategoryPalette`\r\n * utility resolves ids → colors/labels consistently across an entire dashboard.\r\n *\r\n * @example\r\n * const categories: CategoryDef[] = [\r\n * { id: 'alpha', label: 'Team Alpha', color: '#2dccff' },\r\n * { id: 'bravo', label: 'Team Bravo', color: '#fce83a' },\r\n * ];\r\n */\r\nexport interface CategoryDef {\r\n /** Stable unique identifier (string or numeric) */\r\n id: string | number;\r\n /** Human-readable display name */\r\n label: string;\r\n /** CSS color used for visual accents (stripes, rings, swatches, series lines) */\r\n color: string;\r\n /** Optional icon name from the Astro icon set */\r\n icon?: string;\r\n /** Arbitrary metadata for app-specific needs */\r\n meta?: Record<string, unknown>;\r\n}\r\n\r\n// =============================================================================\r\n// Spacecraft Types\r\n// =============================================================================\r\n\r\nexport interface SpacecraftPosition {\r\n /** Optional spacecraft id (when combined with Spacecraft) */\r\n id?: string;\r\n /** Optional display name */\r\n name?: string;\r\n /** Latitude in degrees (-90 to 90) */\r\n latitude: number;\r\n /** Longitude in degrees (-180 to 180) */\r\n longitude: number;\r\n /** Altitude in kilometers */\r\n altitude: number;\r\n /** Velocity in km/s */\r\n velocity?: number;\r\n /** Heading in degrees */\r\n heading?: number;\r\n /** Timestamp */\r\n timestamp?: string;\r\n /** Orbit type (e.g. LEO, GEO) */\r\n orbitType?: string;\r\n /** Orbital inclination in degrees */\r\n inclination?: number;\r\n /** Health/operational status using AstroUXDS 6-level system */\r\n status?: 'off' | 'standby' | 'normal' | 'caution' | 'serious' | 'critical';\r\n}\r\n\r\nexport interface Spacecraft {\r\n id: string;\r\n name: string;\r\n noradId?: number;\r\n type?: string;\r\n /** Status using AstroUXDS 6-level system */\r\n status?: 'off' | 'standby' | 'normal' | 'caution' | 'serious' | 'critical';\r\n}\r\n\r\n// =============================================================================\r\n// Ground Station Types\r\n// =============================================================================\r\n\r\nexport interface GroundStation {\r\n id: string;\r\n name: string;\r\n latitude: number;\r\n longitude: number;\r\n elevation?: number;\r\n /** Minimum elevation angle for visibility (degrees) */\r\n minElevation?: number;\r\n network?: string;\r\n}\r\n\r\nexport interface GroundTrackPoint {\r\n latitude: number;\r\n longitude: number;\r\n altitude: number;\r\n timestamp: string;\r\n}\r\n\r\n// =============================================================================\r\n// Layer Definition (shared by 2D map + 3D viewer panels)\r\n// =============================================================================\r\n\r\n/**\r\n * Developer-defined toggle in a Layers panel. The component that owns the\r\n * panel only renders the toggle UI — it does NOT render the layer's\r\n * content. The consumer app listens to the panel's `onChange` and\r\n * renders/hides its own overlay accordingly.\r\n *\r\n * Used by both `<GroundTrackMap />` (2D map overlays) and the upcoming\r\n * `<LayerControlPanel />` for the 3D viewer.\r\n */\r\nexport interface LayerDef {\r\n /** Unique identifier (e.g. 'heatmap', 'coverage', 'spacecraft'). */\r\n id: string;\r\n /** Display label in the Layers panel. */\r\n label: string;\r\n /** Whether the layer is on by default (default true). */\r\n defaultEnabled?: boolean;\r\n /** Optional grouping bucket for the panel ('overlays', 'objects', etc.). */\r\n category?: string;\r\n}\r\n\r\n/**\r\n * @deprecated Use `LayerDef`. Alias kept for backward compatibility with\r\n * existing `<GroundTrackMap />` consumers.\r\n */\r\nexport type MapLayerDef = LayerDef;\r\n\r\n// =============================================================================\r\n// Access / Contact Types\r\n// =============================================================================\r\n\r\nexport interface AccessData {\r\n stationId: string;\r\n stationName: string;\r\n spacecraftId: string;\r\n aos: string; // Acquisition of Signal - ISO timestamp\r\n los: string; // Loss of Signal - ISO timestamp\r\n maxElevation: number;\r\n azimuthAos?: number;\r\n azimuthLos?: number;\r\n /** Whether link is currently accessible */\r\n isAccessible?: boolean;\r\n /** Current elevation angle (degrees) */\r\n elevation?: number;\r\n /** Current azimuth (degrees) */\r\n azimuth?: number;\r\n /** Slant range in km */\r\n slantRange?: number;\r\n /** Signal strength 0-100 */\r\n signalStrength?: number;\r\n /** Link budget details */\r\n linkBudget?: {\r\n receivedPower?: number;\r\n snr?: number;\r\n dataRate?: number;\r\n };\r\n}\r\n\r\nexport interface AccessWindow {\r\n aos: string;\r\n los: string;\r\n maxElevation: number;\r\n stationId: string;\r\n /** Start time (ISO or ms) */\r\n startTime?: string | number;\r\n /** Duration in seconds */\r\n duration?: number;\r\n}\r\n\r\n// =============================================================================\r\n// Telemetry Types\r\n// =============================================================================\r\n\r\nexport interface TelemetryData {\r\n timestamp: string;\r\n subsystems: {\r\n power?: {\r\n status: string;\r\n batteryLevel?: number;\r\n solarArrayOutput?: number;\r\n voltage?: number;\r\n solarGeneration?: number;\r\n consumption?: number;\r\n };\r\n thermal?: {\r\n status: string;\r\n cpuTemp?: number;\r\n batteryTemp?: number;\r\n };\r\n comms?: {\r\n status: string;\r\n signalStrength?: number;\r\n dataRate?: number;\r\n downlinkRate?: number;\r\n transmitterStatus?: string;\r\n packetsQueued?: number;\r\n bytesTransmitted?: number;\r\n packetsTransmitted?: number;\r\n };\r\n attitude?: {\r\n status: string;\r\n pointingError?: number;\r\n eulerAngles?: EulerAngles;\r\n targetMode?: string;\r\n };\r\n };\r\n /** Health status (convenience alias / extended) */\r\n health?: {\r\n status?: string;\r\n anomalies?: Array<{ id: string; message: string; severity?: string }>;\r\n };\r\n /** Power (convenience alias for subsystems.power / extended) */\r\n power?: {\r\n status?: string;\r\n batteryLevel?: number;\r\n solarArrayOutput?: number;\r\n voltage?: number;\r\n solarGeneration?: number;\r\n consumption?: number;\r\n };\r\n /** Attitude (convenience alias / extended) */\r\n attitude?: {\r\n status?: string;\r\n eulerAngles?: EulerAngles;\r\n targetMode?: string;\r\n pointingError?: number;\r\n };\r\n /** Communications (convenience alias / extended) */\r\n communications?: {\r\n status?: string;\r\n signalStrength?: number;\r\n dataRate?: number;\r\n downlinkRate?: number;\r\n transmitterStatus?: string;\r\n packetsQueued?: number;\r\n bytesTransmitted?: number;\r\n packetsTransmitted?: number;\r\n };\r\n alerts?: Array<{\r\n id: string;\r\n severity: string;\r\n message: string;\r\n timestamp: string;\r\n }>;\r\n}\r\n\r\n// =============================================================================\r\n// Orbital Types\r\n// =============================================================================\r\n\r\nexport interface OrbitalElements {\r\n semiMajorAxis: number;\r\n eccentricity: number;\r\n inclination: number;\r\n raan: number; // Right Ascension of Ascending Node\r\n argumentOfPerigee: number;\r\n trueAnomaly: number;\r\n period?: number;\r\n epoch?: string;\r\n}\r\n\r\n// =============================================================================\r\n// Attitude Types\r\n// =============================================================================\r\n\r\nexport interface Quaternion {\r\n x: number;\r\n y: number;\r\n z: number;\r\n w: number;\r\n}\r\n\r\nexport interface EulerAngles {\r\n roll: number;\r\n pitch: number;\r\n yaw: number;\r\n}\r\n\r\nexport interface AngularVelocity {\r\n x: number;\r\n y: number;\r\n z: number;\r\n}\r\n\r\nexport interface AttitudeData {\r\n quaternion: Quaternion;\r\n eulerAngles: EulerAngles;\r\n angularVelocity: AngularVelocity;\r\n targetMode?: string;\r\n /** Pointing error in degrees */\r\n pointingError?: number;\r\n /** Target direction vector [x, y, z] */\r\n targetDirection?: [number, number, number];\r\n}\r\n\r\nexport type PointingMode = 'nadir' | 'sun' | 'target' | 'inertial' | 'velocity';\r\n\r\n// =============================================================================\r\n// Eclipse Types\r\n// =============================================================================\r\n\r\nexport interface EclipseInfo {\r\n inEclipse: boolean;\r\n timeToEclipse: number; // seconds\r\n eclipseDuration: number; // seconds\r\n sunlightDuration: number; // seconds\r\n /** Umbra duration in seconds */\r\n umbraDuration?: number;\r\n /** Penumbra duration in seconds */\r\n penumbraDuration?: number;\r\n /** Seconds until sunlight (when in eclipse) */\r\n timeToSunlight?: number;\r\n}\r\n\r\n// =============================================================================\r\n// Link Budget Types\r\n// =============================================================================\r\n\r\nexport interface DetailedLinkBudget {\r\n spacecraftId: string;\r\n groundStationId: string;\r\n frequency: number;\r\n eirp: number;\r\n pathLoss: number;\r\n atmosphericLoss: number;\r\n receiverGain: number;\r\n systemNoiseTemp: number;\r\n cnoRequired: number;\r\n cnoActual: number;\r\n margin: number;\r\n dataRate: number;\r\n}\r\n\r\n// =============================================================================\r\n// Thermal Types\r\n// =============================================================================\r\n\r\nexport interface ThermalZone {\r\n id: string;\r\n name: string;\r\n temperature: number;\r\n minLimit: number;\r\n maxLimit: number;\r\n}\r\n\r\nexport interface ThermalData {\r\n timestamp: string;\r\n zones: ThermalZone[];\r\n averageTemp: number;\r\n hottest: { zone: string; temp: number };\r\n coldest: { zone: string; temp: number };\r\n}\r\n\r\n// =============================================================================\r\n// Propulsion Types\r\n// =============================================================================\r\n\r\nexport interface ThrusterStatus {\r\n id: string;\r\n name: string;\r\n status: 'ready' | 'firing' | 'disabled' | 'error';\r\n fuelFlow?: number;\r\n}\r\n\r\nexport interface PropulsionSummary {\r\n fuelRemaining: number;\r\n fuelCapacity: number;\r\n thrusterStatus: ThrusterStatus[];\r\n deltaVRemaining: number;\r\n deltaVUsed: number;\r\n lastManeuver?: string;\r\n}\r\n\r\n// =============================================================================\r\n// Reaction Wheel / Momentum Types\r\n// =============================================================================\r\n\r\nexport interface ReactionWheelData {\r\n /** Unique wheel identifier */\r\n id: string;\r\n /** Display name (e.g. \"RW-X\", \"RW-1\") */\r\n name: string;\r\n /** Mounted axis */\r\n axis: 'X' | 'Y' | 'Z' | 'skew';\r\n /** Current stored angular momentum in N·m·s */\r\n momentumNms: number;\r\n /** Maximum momentum capacity in N·m·s */\r\n maxMomentumNms: number;\r\n /** Current wheel speed in RPM */\r\n speedRpm: number;\r\n /** Maximum wheel speed in RPM */\r\n maxSpeedRpm: number;\r\n /** Operational status */\r\n status: 'nominal' | 'saturated' | 'desaturating' | 'off' | 'error';\r\n /** Power consumption in Watts */\r\n powerW?: number;\r\n /** Bearing temperature in °C */\r\n temperatureC?: number;\r\n}\r\n\r\n/** LVLH (Local Vertical Local Horizontal) frame vector */\r\nexport interface LVLHVector {\r\n /** Radial component (local vertical, positive away from Earth) */\r\n radial: number;\r\n /** In-track component (along velocity vector) */\r\n inTrack: number;\r\n /** Cross-track component (normal to orbital plane) */\r\n crossTrack: number;\r\n}\r\n\r\n/** Time-stamped LVLH state for trajectory plotting */\r\nexport interface LVLHState {\r\n /** Timestamp in milliseconds */\r\n time: number;\r\n /** Position in LVLH frame (km) */\r\n position: LVLHVector;\r\n /** Velocity in LVLH frame (km/s) */\r\n velocity: LVLHVector;\r\n}\r\n\r\n/** Thruster firing event */\r\nexport interface ThrusterFireEvent {\r\n /** Event timestamp in milliseconds */\r\n time: number;\r\n /** Thruster ID */\r\n thrusterId: string;\r\n /** Display name */\r\n name: string;\r\n /** Delta-V magnitude in m/s */\r\n deltaVMs: number;\r\n /** Delta-V direction in LVLH frame (unit vector) */\r\n deltaVDirection?: LVLHVector;\r\n /** Burn duration in seconds */\r\n durationSeconds: number;\r\n /** Event type */\r\n type: 'impulsive' | 'finite' | 'continuous';\r\n}\r\n\r\n// =============================================================================\r\n// Planet Types\r\n// =============================================================================\r\n\r\nexport type PlanetId = \r\n | 'sun' \r\n | 'mercury' \r\n | 'venus' \r\n | 'earth' \r\n | 'moon' \r\n | 'mars' \r\n | 'jupiter' \r\n | 'saturn' \r\n | 'uranus' \r\n | 'neptune' \r\n | 'pluto';\r\n\r\nexport interface PlanetInfo {\r\n id: PlanetId;\r\n name: string;\r\n radius: number; // km\r\n /** @deprecated Use radius */\r\n radiusKm?: number;\r\n mass: number; // kg\r\n semiMajorAxis: number; // AU\r\n /** Distance from Sun in AU (alias for semiMajorAxis) */\r\n distanceFromSunAU?: number;\r\n orbitalPeriod: number; // Earth days\r\n color: string;\r\n}\r\n\r\n// =============================================================================\r\n// Utility Functions (fallbacks when SDK not available)\r\n// =============================================================================\r\n\r\n/**\r\n * Estimate orbital period from altitude (simplified)\r\n */\r\nexport function estimateOrbitalPeriod(altitudeKm: number): number {\r\n const earthRadiusKm = 6371;\r\n const mu = 398600.4418; // km³/s² - Earth's gravitational parameter\r\n const semiMajorAxis = earthRadiusKm + altitudeKm;\r\n const period = 2 * Math.PI * Math.sqrt(Math.pow(semiMajorAxis, 3) / mu);\r\n return period / 60; // Return in minutes\r\n}\r\n\r\n/**\r\n * Estimate orbital velocity from altitude (simplified)\r\n */\r\nexport function estimateOrbitalVelocity(altitudeKm: number): number {\r\n const earthRadiusKm = 6371;\r\n const mu = 398600.4418; // km³/s²\r\n const radius = earthRadiusKm + altitudeKm;\r\n return Math.sqrt(mu / radius); // km/s\r\n}\r\n\r\n/**\r\n * Convert AU to kilometers\r\n */\r\nexport function auToKm(au: number): number {\r\n return au * 149597870.7;\r\n}\r\n\r\n/**\r\n * Normalize planet name to PlanetId\r\n */\r\nexport function normalizePlanetName(name: string): PlanetId | null {\r\n const normalized = name.toLowerCase().trim();\r\n const validPlanets: PlanetId[] = [\r\n 'sun', 'mercury', 'venus', 'earth', 'moon', \r\n 'mars', 'jupiter', 'saturn', 'uranus', 'neptune', 'pluto'\r\n ];\r\n return validPlanets.includes(normalized as PlanetId) \r\n ? (normalized as PlanetId) \r\n : null;\r\n}\r\n\r\n/**\r\n * Get planet info by ID\r\n */\r\nexport function getPlanet(id: PlanetId): PlanetInfo | undefined {\r\n return PLANETS.find(p => p.id === id);\r\n}\r\n\r\n/**\r\n * Planet data (simplified)\r\n */\r\nexport const PLANETS: PlanetInfo[] = [\r\n { id: 'sun', name: 'Sun', radius: 696340, mass: 1.989e30, semiMajorAxis: 0, orbitalPeriod: 0, color: '#ffcc00' },\r\n { id: 'mercury', name: 'Mercury', radius: 2439.7, mass: 3.3011e23, semiMajorAxis: 0.387, orbitalPeriod: 88, color: '#b5b5b5' },\r\n { id: 'venus', name: 'Venus', radius: 6051.8, mass: 4.8675e24, semiMajorAxis: 0.723, orbitalPeriod: 225, color: '#e6c87a' },\r\n { id: 'earth', name: 'Earth', radius: 6371, mass: 5.972e24, semiMajorAxis: 1.0, orbitalPeriod: 365, color: '#4da6ff' },\r\n { id: 'moon', name: 'Moon', radius: 1737.4, mass: 7.342e22, semiMajorAxis: 0.00257, orbitalPeriod: 27.3, color: '#c0c0c0' },\r\n { id: 'mars', name: 'Mars', radius: 3389.5, mass: 6.4171e23, semiMajorAxis: 1.524, orbitalPeriod: 687, color: '#e07b4a' },\r\n { id: 'jupiter', name: 'Jupiter', radius: 69911, mass: 1.8982e27, semiMajorAxis: 5.203, orbitalPeriod: 4333, color: '#d8ca9d' },\r\n { id: 'saturn', name: 'Saturn', radius: 58232, mass: 5.6834e26, semiMajorAxis: 9.537, orbitalPeriod: 10759, color: '#f4d59e' },\r\n { id: 'uranus', name: 'Uranus', radius: 25362, mass: 8.6810e25, semiMajorAxis: 19.19, orbitalPeriod: 30687, color: '#b3e0e6' },\r\n { id: 'neptune', name: 'Neptune', radius: 24622, mass: 1.02413e26, semiMajorAxis: 30.07, orbitalPeriod: 60190, color: '#5b7fff' },\r\n { id: 'pluto', name: 'Pluto', radius: 1188.3, mass: 1.303e22, semiMajorAxis: 39.48, orbitalPeriod: 90560, color: '#c9b79c' },\r\n];\r\n"],"names":[],"mappings":"AAoeO,SAAS,sBAAsB,YAA4B;AAChE,QAAM,gBAAgB;AACtB,QAAM,KAAK;AACX,QAAM,gBAAgB,gBAAgB;AACtC,QAAM,SAAS,IAAI,KAAK,KAAK,KAAK,KAAK,KAAK,IAAI,eAAe,CAAC,IAAI,EAAE;AACtE,SAAO,SAAS;AAClB;AAKO,SAAS,wBAAwB,YAA4B;AAClE,QAAM,gBAAgB;AACtB,QAAM,KAAK;AACX,QAAM,SAAS,gBAAgB;AAC/B,SAAO,KAAK,KAAK,KAAK,MAAM;AAC9B;AAKO,SAAS,OAAO,IAAoB;AACzC,SAAO,KAAK;AACd;AAKO,SAAS,oBAAoB,MAA+B;AACjE,QAAM,aAAa,KAAK,YAAA,EAAc,KAAA;AACtC,QAAM,eAA2B;AAAA,IAC/B;AAAA,IAAO;AAAA,IAAW;AAAA,IAAS;AAAA,IAAS;AAAA,IACpC;AAAA,IAAQ;AAAA,IAAW;AAAA,IAAU;AAAA,IAAU;AAAA,IAAW;AAAA,EAAA;AAEpD,SAAO,aAAa,SAAS,UAAsB,IAC9C,aACD;AACN;AAKO,SAAS,UAAU,IAAsC;AAC9D,SAAO,QAAQ,KAAK,CAAA,MAAK,EAAE,OAAO,EAAE;AACtC;AAKO,MAAM,UAAwB;AAAA,EACnC,EAAE,IAAI,OAAO,MAAM,OAAO,QAAQ,QAAQ,MAAM,SAAU,eAAe,GAAG,eAAe,GAAG,OAAO,UAAA;AAAA,EACrG,EAAE,IAAI,WAAW,MAAM,WAAW,QAAQ,QAAQ,MAAM,UAAW,eAAe,OAAO,eAAe,IAAI,OAAO,UAAA;AAAA,EACnH,EAAE,IAAI,SAAS,MAAM,SAAS,QAAQ,QAAQ,MAAM,UAAW,eAAe,OAAO,eAAe,KAAK,OAAO,UAAA;AAAA,EAChH,EAAE,IAAI,SAAS,MAAM,SAAS,QAAQ,MAAM,MAAM,SAAU,eAAe,GAAK,eAAe,KAAK,OAAO,UAAA;AAAA,EAC3G,EAAE,IAAI,QAAQ,MAAM,QAAQ,QAAQ,QAAQ,MAAM,SAAU,eAAe,QAAS,eAAe,MAAM,OAAO,UAAA;AAAA,EAChH,EAAE,IAAI,QAAQ,MAAM,QAAQ,QAAQ,QAAQ,MAAM,UAAW,eAAe,OAAO,eAAe,KAAK,OAAO,UAAA;AAAA,EAC9G,EAAE,IAAI,WAAW,MAAM,WAAW,QAAQ,OAAO,MAAM,UAAW,eAAe,OAAO,eAAe,MAAM,OAAO,UAAA;AAAA,EACpH,EAAE,IAAI,UAAU,MAAM,UAAU,QAAQ,OAAO,MAAM,UAAW,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AAAA,EACnH,EAAE,IAAI,UAAU,MAAM,UAAU,QAAQ,OAAO,MAAM,SAAW,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AAAA,EACnH,EAAE,IAAI,WAAW,MAAM,WAAW,QAAQ,OAAO,MAAM,WAAY,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AAAA,EACtH,EAAE,IAAI,SAAS,MAAM,SAAS,QAAQ,QAAQ,MAAM,SAAU,eAAe,OAAO,eAAe,OAAO,OAAO,UAAA;AACnH;"}
|