@zendir/ui 0.1.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.
Files changed (359) hide show
  1. package/CHANGELOG.md +19 -0
  2. package/LICENSE +21 -0
  3. package/README.md +589 -0
  4. package/dist/index.d.ts +8 -0
  5. package/dist/index.js +421 -0
  6. package/dist/index.js.map +1 -0
  7. package/dist/react/3d/EarthViewer.d.ts +46 -0
  8. package/dist/react/3d/EarthViewer.js +836 -0
  9. package/dist/react/3d/EarthViewer.js.map +1 -0
  10. package/dist/react/3d/SolarSystemViewer.d.ts +43 -0
  11. package/dist/react/3d/SolarSystemViewer.js +372 -0
  12. package/dist/react/3d/SolarSystemViewer.js.map +1 -0
  13. package/dist/react/3d/ZenSpace3D.d.ts +16 -0
  14. package/dist/react/3d/ZenSpace3D.js +1253 -0
  15. package/dist/react/3d/ZenSpace3D.js.map +1 -0
  16. package/dist/react/3d/ZenSpace3DCesium.d.ts +9 -0
  17. package/dist/react/3d/ZenSpace3DCesium.js +186 -0
  18. package/dist/react/3d/ZenSpace3DCesium.js.map +1 -0
  19. package/dist/react/3d/ZenSpace3DShaders.d.ts +78 -0
  20. package/dist/react/3d/ZenSpace3DShaders.js +94 -0
  21. package/dist/react/3d/ZenSpace3DShaders.js.map +1 -0
  22. package/dist/react/3d/ZenSpace3DTypes.d.ts +614 -0
  23. package/dist/react/3d/ZenSpace3DUtils.d.ts +183 -0
  24. package/dist/react/3d/ZenSpace3DUtils.js +213 -0
  25. package/dist/react/3d/ZenSpace3DUtils.js.map +1 -0
  26. package/dist/react/3d/index.d.ts +23 -0
  27. package/dist/react/3d/threeLoader.d.ts +22 -0
  28. package/dist/react/3d/threeLoader.js +18 -0
  29. package/dist/react/3d/threeLoader.js.map +1 -0
  30. package/dist/react/astro/ClassificationBanner.d.ts +25 -0
  31. package/dist/react/astro/ClassificationBanner.js +83 -0
  32. package/dist/react/astro/ClassificationBanner.js.map +1 -0
  33. package/dist/react/astro/GlobalStatusBar.d.ts +42 -0
  34. package/dist/react/astro/GlobalStatusBar.js +165 -0
  35. package/dist/react/astro/GlobalStatusBar.js.map +1 -0
  36. package/dist/react/astro/MissionClock.d.ts +169 -0
  37. package/dist/react/astro/MissionClock.js +411 -0
  38. package/dist/react/astro/MissionClock.js.map +1 -0
  39. package/dist/react/astro/MonitoringIcon.d.ts +60 -0
  40. package/dist/react/astro/MonitoringIcon.js +369 -0
  41. package/dist/react/astro/MonitoringIcon.js.map +1 -0
  42. package/dist/react/astro/Notification.d.ts +42 -0
  43. package/dist/react/astro/Notification.js +156 -0
  44. package/dist/react/astro/Notification.js.map +1 -0
  45. package/dist/react/astro/Progress.d.ts +39 -0
  46. package/dist/react/astro/Progress.js +149 -0
  47. package/dist/react/astro/Progress.js.map +1 -0
  48. package/dist/react/astro/SimulationControls.d.ts +136 -0
  49. package/dist/react/astro/SimulationControls.js +668 -0
  50. package/dist/react/astro/SimulationControls.js.map +1 -0
  51. package/dist/react/astro/StatusIndicator.d.ts +34 -0
  52. package/dist/react/astro/StatusIndicator.js +189 -0
  53. package/dist/react/astro/StatusIndicator.js.map +1 -0
  54. package/dist/react/astro/UnifiedTimeline.d.ts +106 -0
  55. package/dist/react/astro/UnifiedTimeline.js +1768 -0
  56. package/dist/react/astro/UnifiedTimeline.js.map +1 -0
  57. package/dist/react/astro/index.d.ts +63 -0
  58. package/dist/react/cards/AccessCard.d.ts +37 -0
  59. package/dist/react/cards/AccessCard.js +410 -0
  60. package/dist/react/cards/AccessCard.js.map +1 -0
  61. package/dist/react/cards/OrbitCard.d.ts +31 -0
  62. package/dist/react/cards/OrbitCard.js +372 -0
  63. package/dist/react/cards/OrbitCard.js.map +1 -0
  64. package/dist/react/cards/SpacecraftCard.d.ts +54 -0
  65. package/dist/react/cards/SpacecraftCard.js +941 -0
  66. package/dist/react/cards/SpacecraftCard.js.map +1 -0
  67. package/dist/react/cards/TelemetryCard.d.ts +40 -0
  68. package/dist/react/cards/TelemetryCard.js +742 -0
  69. package/dist/react/cards/TelemetryCard.js.map +1 -0
  70. package/dist/react/cards/TelemetryStreamCard.d.ts +59 -0
  71. package/dist/react/cards/TelemetryStreamCard.js +309 -0
  72. package/dist/react/cards/TelemetryStreamCard.js.map +1 -0
  73. package/dist/react/cards/index.d.ts +13 -0
  74. package/dist/react/charts/GroundTrackMap.d.ts +112 -0
  75. package/dist/react/charts/GroundTrackMap.js +1123 -0
  76. package/dist/react/charts/GroundTrackMap.js.map +1 -0
  77. package/dist/react/charts/GroundTrackMapLeaflet.d.ts +26 -0
  78. package/dist/react/charts/GroundTrackMapLeaflet.js +571 -0
  79. package/dist/react/charts/GroundTrackMapLeaflet.js.map +1 -0
  80. package/dist/react/charts/groundTrackMapLeafletTiles.d.ts +22 -0
  81. package/dist/react/charts/groundTrackMapLeafletTiles.js +11 -0
  82. package/dist/react/charts/groundTrackMapLeafletTiles.js.map +1 -0
  83. package/dist/react/charts/groundTrackMapLeafletUtils.d.ts +24 -0
  84. package/dist/react/charts/groundTrackMapLeafletUtils.js +109 -0
  85. package/dist/react/charts/groundTrackMapLeafletUtils.js.map +1 -0
  86. package/dist/react/charts/index.d.ts +10 -0
  87. package/dist/react/charts/unified/AstroChart.d.ts +24 -0
  88. package/dist/react/charts/unified/AstroChart.js +1405 -0
  89. package/dist/react/charts/unified/AstroChart.js.map +1 -0
  90. package/dist/react/charts/unified/PowerOverviewChart.d.ts +73 -0
  91. package/dist/react/charts/unified/PowerOverviewChart.js +488 -0
  92. package/dist/react/charts/unified/PowerOverviewChart.js.map +1 -0
  93. package/dist/react/charts/unified/domain.d.ts +845 -0
  94. package/dist/react/charts/unified/domain.js +3168 -0
  95. package/dist/react/charts/unified/domain.js.map +1 -0
  96. package/dist/react/charts/unified/generators.d.ts +276 -0
  97. package/dist/react/charts/unified/generators.js +518 -0
  98. package/dist/react/charts/unified/generators.js.map +1 -0
  99. package/dist/react/charts/unified/index.d.ts +55 -0
  100. package/dist/react/charts/unified/presets.d.ts +290 -0
  101. package/dist/react/charts/unified/presets.js +999 -0
  102. package/dist/react/charts/unified/presets.js.map +1 -0
  103. package/dist/react/charts/unified/sync.d.ts +69 -0
  104. package/dist/react/charts/unified/sync.js +219 -0
  105. package/dist/react/charts/unified/sync.js.map +1 -0
  106. package/dist/react/charts/unified/theme.d.ts +447 -0
  107. package/dist/react/charts/unified/theme.js +562 -0
  108. package/dist/react/charts/unified/theme.js.map +1 -0
  109. package/dist/react/charts/unified/types.d.ts +826 -0
  110. package/dist/react/charts/unified/useChartStream.d.ts +58 -0
  111. package/dist/react/charts/unified/useChartStream.js +226 -0
  112. package/dist/react/charts/unified/useChartStream.js.map +1 -0
  113. package/dist/react/chatgpt/AppCard.d.ts +59 -0
  114. package/dist/react/chatgpt/AppCard.js +306 -0
  115. package/dist/react/chatgpt/AppCard.js.map +1 -0
  116. package/dist/react/chatgpt/ChatGPTCard.d.ts +6 -0
  117. package/dist/react/chatgpt/index.d.ts +167 -0
  118. package/dist/react/chatgpt/index.js +166 -0
  119. package/dist/react/chatgpt/index.js.map +1 -0
  120. package/dist/react/context/DisplaySettingsContext.d.ts +107 -0
  121. package/dist/react/context/DisplaySettingsContext.js +169 -0
  122. package/dist/react/context/DisplaySettingsContext.js.map +1 -0
  123. package/dist/react/context/index.d.ts +5 -0
  124. package/dist/react/core/ActivityPlanner.d.ts +45 -0
  125. package/dist/react/core/ActivityPlanner.js +532 -0
  126. package/dist/react/core/ActivityPlanner.js.map +1 -0
  127. package/dist/react/core/AppBar.d.ts +71 -0
  128. package/dist/react/core/AppBar.js +817 -0
  129. package/dist/react/core/AppBar.js.map +1 -0
  130. package/dist/react/core/AstroIcon.d.ts +84 -0
  131. package/dist/react/core/AstroIcon.js +1243 -0
  132. package/dist/react/core/AstroIcon.js.map +1 -0
  133. package/dist/react/core/Badge.d.ts +27 -0
  134. package/dist/react/core/Badge.js +134 -0
  135. package/dist/react/core/Badge.js.map +1 -0
  136. package/dist/react/core/Button.d.ts +26 -0
  137. package/dist/react/core/Button.js +306 -0
  138. package/dist/react/core/Button.js.map +1 -0
  139. package/dist/react/core/CardHeader.d.ts +34 -0
  140. package/dist/react/core/CardHeader.js +316 -0
  141. package/dist/react/core/CardHeader.js.map +1 -0
  142. package/dist/react/core/ChatPanel.d.ts +627 -0
  143. package/dist/react/core/ChatPanel.js +1144 -0
  144. package/dist/react/core/ChatPanel.js.map +1 -0
  145. package/dist/react/core/Checkbox.d.ts +26 -0
  146. package/dist/react/core/Checkbox.js +130 -0
  147. package/dist/react/core/Checkbox.js.map +1 -0
  148. package/dist/react/core/ColorPickerPanel.d.ts +25 -0
  149. package/dist/react/core/ColorPickerPanel.js +293 -0
  150. package/dist/react/core/ColorPickerPanel.js.map +1 -0
  151. package/dist/react/core/CommandBuilder.d.ts +74 -0
  152. package/dist/react/core/CommandBuilder.js +518 -0
  153. package/dist/react/core/CommandBuilder.js.map +1 -0
  154. package/dist/react/core/ConfirmDialog.d.ts +45 -0
  155. package/dist/react/core/ConfirmDialog.js +315 -0
  156. package/dist/react/core/ConfirmDialog.js.map +1 -0
  157. package/dist/react/core/ConnectionForm.d.ts +57 -0
  158. package/dist/react/core/ConnectionForm.js +496 -0
  159. package/dist/react/core/ConnectionForm.js.map +1 -0
  160. package/dist/react/core/Container.d.ts +51 -0
  161. package/dist/react/core/Container.js +670 -0
  162. package/dist/react/core/Container.js.map +1 -0
  163. package/dist/react/core/CopyButton.d.ts +39 -0
  164. package/dist/react/core/CopyButton.js +105 -0
  165. package/dist/react/core/CopyButton.js.map +1 -0
  166. package/dist/react/core/DataTable.d.ts +113 -0
  167. package/dist/react/core/DataTable.js +446 -0
  168. package/dist/react/core/DataTable.js.map +1 -0
  169. package/dist/react/core/DataValue.d.ts +64 -0
  170. package/dist/react/core/DataValue.js +545 -0
  171. package/dist/react/core/DataValue.js.map +1 -0
  172. package/dist/react/core/Dialog.d.ts +32 -0
  173. package/dist/react/core/Dialog.js +201 -0
  174. package/dist/react/core/Dialog.js.map +1 -0
  175. package/dist/react/core/FileExplorer.d.ts +65 -0
  176. package/dist/react/core/FileExplorer.js +520 -0
  177. package/dist/react/core/FileExplorer.js.map +1 -0
  178. package/dist/react/core/GlassCard.d.ts +129 -0
  179. package/dist/react/core/GlassCard.js +375 -0
  180. package/dist/react/core/GlassCard.js.map +1 -0
  181. package/dist/react/core/HeaderIconWithStatus.d.ts +39 -0
  182. package/dist/react/core/HeaderIconWithStatus.js +157 -0
  183. package/dist/react/core/HeaderIconWithStatus.js.map +1 -0
  184. package/dist/react/core/HexViewer.d.ts +143 -0
  185. package/dist/react/core/HexViewer.js +1106 -0
  186. package/dist/react/core/HexViewer.js.map +1 -0
  187. package/dist/react/core/Icon.d.ts +32 -0
  188. package/dist/react/core/Icon.js +142 -0
  189. package/dist/react/core/Icon.js.map +1 -0
  190. package/dist/react/core/ImageGallery.d.ts +41 -0
  191. package/dist/react/core/ImageGallery.js +320 -0
  192. package/dist/react/core/ImageGallery.js.map +1 -0
  193. package/dist/react/core/Input.d.ts +38 -0
  194. package/dist/react/core/Input.js +288 -0
  195. package/dist/react/core/Input.js.map +1 -0
  196. package/dist/react/core/LimitsBar.d.ts +51 -0
  197. package/dist/react/core/LimitsBar.js +200 -0
  198. package/dist/react/core/LimitsBar.js.map +1 -0
  199. package/dist/react/core/LogViewer.d.ts +61 -0
  200. package/dist/react/core/LogViewer.js +599 -0
  201. package/dist/react/core/LogViewer.js.map +1 -0
  202. package/dist/react/core/MessageStream.d.ts +58 -0
  203. package/dist/react/core/MessageStream.js +455 -0
  204. package/dist/react/core/MessageStream.js.map +1 -0
  205. package/dist/react/core/MissionCalendar.d.ts +81 -0
  206. package/dist/react/core/MissionCalendar.js +1049 -0
  207. package/dist/react/core/MissionCalendar.js.map +1 -0
  208. package/dist/react/core/NumberInput.d.ts +85 -0
  209. package/dist/react/core/NumberInput.js +507 -0
  210. package/dist/react/core/NumberInput.js.map +1 -0
  211. package/dist/react/core/PacketViewer.d.ts +73 -0
  212. package/dist/react/core/PacketViewer.js +431 -0
  213. package/dist/react/core/PacketViewer.js.map +1 -0
  214. package/dist/react/core/Pagination.d.ts +30 -0
  215. package/dist/react/core/Pagination.js +190 -0
  216. package/dist/react/core/Pagination.js.map +1 -0
  217. package/dist/react/core/PinInput.d.ts +41 -0
  218. package/dist/react/core/PinInput.js +210 -0
  219. package/dist/react/core/PinInput.js.map +1 -0
  220. package/dist/react/core/Popover.d.ts +55 -0
  221. package/dist/react/core/Popover.js +288 -0
  222. package/dist/react/core/Popover.js.map +1 -0
  223. package/dist/react/core/Select.d.ts +42 -0
  224. package/dist/react/core/Select.js +303 -0
  225. package/dist/react/core/Select.js.map +1 -0
  226. package/dist/react/core/SideNav.d.ts +103 -0
  227. package/dist/react/core/SideNav.js +551 -0
  228. package/dist/react/core/SideNav.js.map +1 -0
  229. package/dist/react/core/SidePanel.d.ts +33 -0
  230. package/dist/react/core/SidePanel.js +199 -0
  231. package/dist/react/core/SidePanel.js.map +1 -0
  232. package/dist/react/core/Tabs.d.ts +47 -0
  233. package/dist/react/core/Tabs.js +129 -0
  234. package/dist/react/core/Tabs.js.map +1 -0
  235. package/dist/react/core/Toast.d.ts +56 -0
  236. package/dist/react/core/Toast.js +229 -0
  237. package/dist/react/core/Toast.js.map +1 -0
  238. package/dist/react/core/Toggle.d.ts +22 -0
  239. package/dist/react/core/Toggle.js +151 -0
  240. package/dist/react/core/Toggle.js.map +1 -0
  241. package/dist/react/core/Tooltip.d.ts +19 -0
  242. package/dist/react/core/Tooltip.js +179 -0
  243. package/dist/react/core/Tooltip.js.map +1 -0
  244. package/dist/react/core/Typography.d.ts +127 -0
  245. package/dist/react/core/Typography.js +187 -0
  246. package/dist/react/core/Typography.js.map +1 -0
  247. package/dist/react/core/index.d.ts +108 -0
  248. package/dist/react/core/layout/Box.d.ts +77 -0
  249. package/dist/react/core/layout/Box.js +126 -0
  250. package/dist/react/core/layout/Box.js.map +1 -0
  251. package/dist/react/core/layout/Center.d.ts +20 -0
  252. package/dist/react/core/layout/Center.js +34 -0
  253. package/dist/react/core/layout/Center.js.map +1 -0
  254. package/dist/react/core/layout/Divider.d.ts +16 -0
  255. package/dist/react/core/layout/Divider.js +108 -0
  256. package/dist/react/core/layout/Divider.js.map +1 -0
  257. package/dist/react/core/layout/Flex.d.ts +30 -0
  258. package/dist/react/core/layout/Flex.js +128 -0
  259. package/dist/react/core/layout/Flex.js.map +1 -0
  260. package/dist/react/core/layout/Grid.d.ts +36 -0
  261. package/dist/react/core/layout/Grid.js +142 -0
  262. package/dist/react/core/layout/Grid.js.map +1 -0
  263. package/dist/react/core/layout/Spacer.d.ts +8 -0
  264. package/dist/react/core/layout/Spacer.js +31 -0
  265. package/dist/react/core/layout/Spacer.js.map +1 -0
  266. package/dist/react/core/layout/Stack.d.ts +54 -0
  267. package/dist/react/core/layout/Stack.js +74 -0
  268. package/dist/react/core/layout/Stack.js.map +1 -0
  269. package/dist/react/core/layout/index.d.ts +38 -0
  270. package/dist/react/core/layout/responsive.d.ts +23 -0
  271. package/dist/react/core/layout/responsive.js +26 -0
  272. package/dist/react/core/layout/responsive.js.map +1 -0
  273. package/dist/react/core/layout/useBreakpoint.d.ts +77 -0
  274. package/dist/react/core/layout/useBreakpoint.js +73 -0
  275. package/dist/react/core/layout/useBreakpoint.js.map +1 -0
  276. package/dist/react/core/propertyConfig.d.ts +443 -0
  277. package/dist/react/core/propertyConfig.js +399 -0
  278. package/dist/react/core/propertyConfig.js.map +1 -0
  279. package/dist/react/hooks/index.d.ts +21 -0
  280. package/dist/react/hooks/useAccessWindows.d.ts +66 -0
  281. package/dist/react/hooks/useCompactMode.d.ts +82 -0
  282. package/dist/react/hooks/useCompactMode.js +62 -0
  283. package/dist/react/hooks/useCompactMode.js.map +1 -0
  284. package/dist/react/hooks/useLiveSelection.d.ts +57 -0
  285. package/dist/react/hooks/useSimulationPlayback.d.ts +97 -0
  286. package/dist/react/hooks/useSimulationTime.d.ts +61 -0
  287. package/dist/react/hooks/useSpacecraftPosition.d.ts +50 -0
  288. package/dist/react/hooks/useSpacecraftPosition.js +89 -0
  289. package/dist/react/hooks/useSpacecraftPosition.js.map +1 -0
  290. package/dist/react/hooks/useTelemetry.d.ts +55 -0
  291. package/dist/react/hooks/useTelemetry.js +73 -0
  292. package/dist/react/hooks/useTelemetry.js.map +1 -0
  293. package/dist/react/hooks/useZendirSession.d.ts +109 -0
  294. package/dist/react/hooks/useZendirSession.js +148 -0
  295. package/dist/react/hooks/useZendirSession.js.map +1 -0
  296. package/dist/react/index.d.ts +74 -0
  297. package/dist/react/shared/ErrorBoundary.d.ts +63 -0
  298. package/dist/react/shared/ErrorBoundary.js +142 -0
  299. package/dist/react/shared/ErrorBoundary.js.map +1 -0
  300. package/dist/react/shared/Skeleton.d.ts +110 -0
  301. package/dist/react/shared/Skeleton.js +324 -0
  302. package/dist/react/shared/Skeleton.js.map +1 -0
  303. package/dist/react/shared/index.d.ts +12 -0
  304. package/dist/react/theme/ThemeProvider.d.ts +385 -0
  305. package/dist/react/theme/ThemeProvider.js +1096 -0
  306. package/dist/react/theme/ThemeProvider.js.map +1 -0
  307. package/dist/react/theme/astro-tokens.d.ts +153 -0
  308. package/dist/react/theme/cardAccent.d.ts +75 -0
  309. package/dist/react/theme/cardAccent.js +137 -0
  310. package/dist/react/theme/cardAccent.js.map +1 -0
  311. package/dist/react/theme/config.d.ts +39 -0
  312. package/dist/react/theme/index.d.ts +9 -0
  313. package/dist/react/types.d.ts +360 -0
  314. package/dist/react/types.js +58 -0
  315. package/dist/react/types.js.map +1 -0
  316. package/dist/react/utils/index.d.ts +247 -0
  317. package/dist/react/utils/index.js +423 -0
  318. package/dist/react/utils/index.js.map +1 -0
  319. package/dist/react/visualizations/EclipseTimerCard.d.ts +17 -0
  320. package/dist/react/visualizations/EclipseTimerCard.js +250 -0
  321. package/dist/react/visualizations/EclipseTimerCard.js.map +1 -0
  322. package/dist/react/visualizations/LinkBudgetCard.d.ts +50 -0
  323. package/dist/react/visualizations/LinkBudgetCard.js +444 -0
  324. package/dist/react/visualizations/LinkBudgetCard.js.map +1 -0
  325. package/dist/react/visualizations/NavBallCard.d.ts +17 -0
  326. package/dist/react/visualizations/NavBallCard.js +243 -0
  327. package/dist/react/visualizations/NavBallCard.js.map +1 -0
  328. package/dist/react/visualizations/PropulsionCard.d.ts +37 -0
  329. package/dist/react/visualizations/PropulsionCard.js +298 -0
  330. package/dist/react/visualizations/PropulsionCard.js.map +1 -0
  331. package/dist/react/visualizations/SensorFootprintCard.d.ts +33 -0
  332. package/dist/react/visualizations/SensorFootprintCard.js +326 -0
  333. package/dist/react/visualizations/SensorFootprintCard.js.map +1 -0
  334. package/dist/react/visualizations/ThermalHeatmapCard.d.ts +38 -0
  335. package/dist/react/visualizations/ThermalHeatmapCard.js +372 -0
  336. package/dist/react/visualizations/ThermalHeatmapCard.js.map +1 -0
  337. package/dist/react/visualizations/index.d.ts +17 -0
  338. package/dist/react.d.ts +1 -0
  339. package/dist/react.js +421 -0
  340. package/dist/react.js.map +1 -0
  341. package/dist/shaders/atmosphere.frag.js +5 -0
  342. package/dist/shaders/atmosphere.frag.js.map +1 -0
  343. package/dist/shaders/atmosphere.vert.js +5 -0
  344. package/dist/shaders/atmosphere.vert.js.map +1 -0
  345. package/dist/shaders/stars.frag.js +5 -0
  346. package/dist/shaders/stars.frag.js.map +1 -0
  347. package/dist/shaders/stars.vert.js +5 -0
  348. package/dist/shaders/stars.vert.js.map +1 -0
  349. package/dist/style.css +143 -0
  350. package/dist/tokens/index.d.ts +296 -0
  351. package/dist/tokens/index.js +263 -0
  352. package/dist/tokens/index.js.map +1 -0
  353. package/dist/tokens/tokens.css +155 -0
  354. package/dist/types/index.d.ts +23 -0
  355. package/dist/types.d.ts +1 -0
  356. package/dist/types.js +2 -0
  357. package/dist/types.js.map +1 -0
  358. package/package.json +220 -0
  359. package/sdk-stub.js +22 -0
@@ -0,0 +1,518 @@
1
+ const SPEED_OF_LIGHT = 299792458;
2
+ const EARTH_RADIUS = 6371;
3
+ const MU_EARTH = 398600.4418;
4
+ function generatePowerData(config = {}) {
5
+ const {
6
+ orbitPeriod = 5400,
7
+ eclipseDuration = 2100,
8
+ solarArrayPower = 85,
9
+ loadPower = 45,
10
+ batteryCapacity = 300,
11
+ initialSoC = 80,
12
+ eclipseLoadReduction = 0.7,
13
+ orbits = 2,
14
+ resolution = 60
15
+ } = config;
16
+ const data = [];
17
+ const totalDuration = orbitPeriod * orbits;
18
+ let batterySoC = initialSoC;
19
+ const batteryEfficiency = 0.9;
20
+ const eclipseStart = 0.4;
21
+ const eclipseEnd = eclipseStart + eclipseDuration / orbitPeriod;
22
+ for (let t = 0; t < totalDuration; t += resolution) {
23
+ const orbitPhase = t % orbitPeriod / orbitPeriod;
24
+ const inEclipse = orbitPhase >= eclipseStart && orbitPhase < eclipseEnd;
25
+ const pointingFactor = 0.9 + 0.1 * Math.sin(orbitPhase * Math.PI * 2);
26
+ const solarPower = inEclipse ? 0 : solarArrayPower * pointingFactor;
27
+ const consumption = inEclipse ? loadPower * eclipseLoadReduction : loadPower;
28
+ const netPower = solarPower - consumption;
29
+ const energyDelta = netPower * resolution / 3600;
30
+ const efficiencyFactor = netPower > 0 ? batteryEfficiency : 1 / batteryEfficiency;
31
+ batterySoC += energyDelta * efficiencyFactor / batteryCapacity * 100;
32
+ batterySoC = Math.max(0, Math.min(100, batterySoC));
33
+ data.push({
34
+ time: Date.now() + t * 1e3,
35
+ solarPower: Math.round(solarPower * 10) / 10,
36
+ consumption: Math.round(consumption * 10) / 10,
37
+ netPower: Math.round(netPower * 10) / 10,
38
+ batteryPercent: Math.round(batterySoC * 10) / 10,
39
+ eclipse: inEclipse
40
+ });
41
+ }
42
+ return data;
43
+ }
44
+ function generateAttitudeData(config = {}) {
45
+ const {
46
+ initialAttitude = [0, -45, 90],
47
+ targetAttitude = [0, 30, 180],
48
+ slewRate = 0.5,
49
+ settleTime = 30,
50
+ jitter = 0.02,
51
+ samplingRate = 1,
52
+ duration = 300
53
+ } = config;
54
+ const data = [];
55
+ const dt = 1 / samplingRate;
56
+ const deltaAttitude = targetAttitude.map((t, i) => t - initialAttitude[i]);
57
+ const totalAngle = Math.sqrt(deltaAttitude.reduce((sum, d) => sum + d * d, 0));
58
+ const slewDuration = totalAngle / slewRate;
59
+ deltaAttitude.map((d) => d / totalAngle);
60
+ const gaussRandom = () => {
61
+ const u1 = Math.random();
62
+ const u2 = Math.random();
63
+ return Math.sqrt(-2 * Math.log(u1)) * Math.cos(2 * Math.PI * u2);
64
+ };
65
+ let prevRoll = initialAttitude[0];
66
+ let prevPitch = initialAttitude[1];
67
+ let prevYaw = initialAttitude[2];
68
+ for (let t = 0; t < duration; t += dt) {
69
+ let roll, pitch, yaw;
70
+ let pointingError;
71
+ if (t < slewDuration) {
72
+ const progress = t / slewDuration;
73
+ const smoothProgress = 0.5 - 0.5 * Math.cos(progress * Math.PI);
74
+ roll = initialAttitude[0] + deltaAttitude[0] * smoothProgress;
75
+ pitch = initialAttitude[1] + deltaAttitude[1] * smoothProgress;
76
+ yaw = initialAttitude[2] + deltaAttitude[2] * smoothProgress;
77
+ pointingError = totalAngle * (1 - progress);
78
+ } else if (t < slewDuration + settleTime) {
79
+ const settleProgress = (t - slewDuration) / settleTime;
80
+ const decayFactor = Math.exp(-3 * settleProgress);
81
+ const overshoot = 0.05 * Math.sin(settleProgress * Math.PI * 4) * decayFactor;
82
+ roll = targetAttitude[0] + overshoot + jitter * gaussRandom();
83
+ pitch = targetAttitude[1] + overshoot + jitter * gaussRandom();
84
+ yaw = targetAttitude[2] + overshoot + jitter * gaussRandom();
85
+ pointingError = Math.abs(overshoot) + jitter * Math.abs(gaussRandom());
86
+ } else {
87
+ roll = targetAttitude[0] + jitter * gaussRandom();
88
+ pitch = targetAttitude[1] + jitter * gaussRandom();
89
+ yaw = targetAttitude[2] + jitter * gaussRandom();
90
+ pointingError = jitter * Math.sqrt(
91
+ Math.pow(gaussRandom(), 2) + Math.pow(gaussRandom(), 2) + Math.pow(gaussRandom(), 2)
92
+ );
93
+ }
94
+ const angularVelocity = Math.sqrt(
95
+ Math.pow((roll - prevRoll) / dt, 2) + Math.pow((pitch - prevPitch) / dt, 2) + Math.pow((yaw - prevYaw) / dt, 2)
96
+ );
97
+ data.push({
98
+ time: Date.now() + t * 1e3,
99
+ roll: Math.round(roll * 100) / 100,
100
+ pitch: Math.round(pitch * 100) / 100,
101
+ yaw: Math.round(yaw * 100) / 100,
102
+ pointingError: Math.round(pointingError * 1e3) / 1e3,
103
+ angularVelocity: Math.round(angularVelocity * 1e3) / 1e3
104
+ });
105
+ prevRoll = roll;
106
+ prevPitch = pitch;
107
+ prevYaw = yaw;
108
+ }
109
+ return data;
110
+ }
111
+ function generateDopplerData(config = {}) {
112
+ const {
113
+ carrierFrequency = 437e6,
114
+ // 437 MHz (UHF amateur)
115
+ altitude = 400,
116
+ // km (ISS altitude)
117
+ maxElevation = 75,
118
+ passDuration = 600,
119
+ // 10 minutes
120
+ resolution = 1,
121
+ measurementNoise = 50,
122
+ // Hz
123
+ satelliteName = "ISS",
124
+ groundStation: _groundStation = "Home Station"
125
+ } = config;
126
+ const predicted = [];
127
+ const measured = [];
128
+ const orbitRadius = EARTH_RADIUS + altitude;
129
+ const orbitalVelocity = Math.sqrt(MU_EARTH / orbitRadius);
130
+ const maxDoppler = carrierFrequency * (orbitalVelocity * 1e3 / SPEED_OF_LIGHT);
131
+ const elevationRad = maxElevation * Math.PI / 180;
132
+ const passHalfAngle = Math.PI / 2 - elevationRad;
133
+ const now = Date.now();
134
+ const aos = now;
135
+ const los = now + passDuration * 1e3;
136
+ const tca = now + passDuration / 2 * 1e3;
137
+ const gaussRandom = () => {
138
+ const u1 = Math.random();
139
+ const u2 = Math.random();
140
+ return Math.sqrt(-2 * Math.log(u1)) * Math.cos(2 * Math.PI * u2);
141
+ };
142
+ for (let t = 0; t <= passDuration; t += resolution) {
143
+ const time = now + t * 1e3;
144
+ const passPhase = t / passDuration;
145
+ const angle = (passPhase - 0.5) * 2 * passHalfAngle;
146
+ const dopplerShift = maxDoppler * Math.sin(angle);
147
+ const elevation = maxElevation * Math.cos(angle);
148
+ predicted.push({
149
+ time,
150
+ dopplerShift: Math.round(dopplerShift),
151
+ elevation: Math.round(elevation * 10) / 10
152
+ });
153
+ const measuredShift = dopplerShift + measurementNoise * gaussRandom();
154
+ measured.push({
155
+ time,
156
+ dopplerShift: Math.round(measuredShift),
157
+ elevation: Math.round(elevation * 10) / 10
158
+ });
159
+ }
160
+ const passInfo = {
161
+ name: satelliteName,
162
+ aos: new Date(aos),
163
+ los: new Date(los),
164
+ tca: new Date(tca),
165
+ maxElevation
166
+ };
167
+ return { predicted, measured, passInfo };
168
+ }
169
+ function generateWaterfallData(config = {}) {
170
+ const {
171
+ centerFrequency = 437e6,
172
+ bandwidth = 1e5,
173
+ fftBins = 512,
174
+ noiseFloor = -120,
175
+ timeSamples = 100,
176
+ signals = [
177
+ { offset: -3e4, power: -80, bandwidth: 500, type: "cw" },
178
+ { offset: 0, power: -75, bandwidth: 2e4, type: "digital" },
179
+ { offset: 25e3, power: -90, bandwidth: 3e3, type: "ssb" }
180
+ ]
181
+ } = config;
182
+ const data = [];
183
+ const freqStep = bandwidth / fftBins;
184
+ const noiseStdDev = 3;
185
+ const gaussRandom = () => {
186
+ const u1 = Math.random();
187
+ const u2 = Math.random();
188
+ return Math.sqrt(-2 * Math.log(u1)) * Math.cos(2 * Math.PI * u2);
189
+ };
190
+ const frequencies = [];
191
+ for (let f = 0; f < fftBins; f++) {
192
+ frequencies.push(centerFrequency + (f - fftBins / 2) * freqStep);
193
+ }
194
+ for (let t = 0; t < timeSamples; t++) {
195
+ const powers = [];
196
+ const time = Date.now() - (timeSamples - t) * 100;
197
+ for (let f = 0; f < fftBins; f++) {
198
+ const freqOffset = (f - fftBins / 2) * freqStep;
199
+ let power = noiseFloor + noiseStdDev * gaussRandom();
200
+ for (const signal of signals) {
201
+ const signalOffset = signal.offset + (signal.dopplerDrift || 0) * (t / 10);
202
+ const distance = Math.abs(freqOffset - signalOffset);
203
+ if (signal.type === "cw") {
204
+ if (distance < signal.bandwidth * 2) {
205
+ const shape = Math.pow(Math.sin(Math.PI * distance / signal.bandwidth) / (Math.PI * distance / signal.bandwidth + 1e-3), 2);
206
+ power = Math.max(power, signal.power + 10 * Math.log10(shape + 1e-3) + gaussRandom());
207
+ }
208
+ } else if (signal.type === "ssb") {
209
+ if (freqOffset >= signalOffset && freqOffset <= signalOffset + signal.bandwidth) {
210
+ const modulation = 0.5 + 0.5 * Math.sin(t * 0.3 + f * 0.1);
211
+ power = Math.max(power, signal.power - 3 + 6 * modulation + 2 * gaussRandom());
212
+ }
213
+ } else if (signal.type === "digital") {
214
+ if (distance < signal.bandwidth / 2) {
215
+ power = Math.max(power, signal.power + gaussRandom());
216
+ } else if (distance < signal.bandwidth) {
217
+ const rolloff = 1 - (distance - signal.bandwidth / 2) / (signal.bandwidth / 2);
218
+ power = Math.max(power, signal.power - 20 * (1 - rolloff) + gaussRandom());
219
+ }
220
+ } else if (signal.type === "beacon") {
221
+ if (distance < signal.bandwidth) {
222
+ power = Math.max(power, signal.power);
223
+ }
224
+ }
225
+ }
226
+ powers.push(Math.round(power * 10) / 10);
227
+ }
228
+ data.push({ time, frequencies: [...frequencies], powers });
229
+ }
230
+ return data;
231
+ }
232
+ function generateContactWindows(config = {}) {
233
+ const {
234
+ groundStations = [
235
+ { name: "Svalbard (SG)", latitude: 78.2, longitude: 15.4 },
236
+ { name: "McMurdo (MCM)", latitude: -77.8, longitude: 166.7 },
237
+ { name: "Hawaii (HBK)", latitude: 21.5, longitude: -158 },
238
+ { name: "Kiruna (KIR)", latitude: 67.9, longitude: 20.9 }
239
+ ],
240
+ altitude = 400,
241
+ inclination = 51.6,
242
+ duration = 24,
243
+ minElevation = 5
244
+ } = config;
245
+ const result = [];
246
+ const now = Date.now();
247
+ const orbitPeriod = 2 * Math.PI * Math.sqrt(Math.pow(EARTH_RADIUS + altitude, 3) / MU_EARTH);
248
+ const passesPerDay = Math.floor(duration * 3600 / orbitPeriod);
249
+ for (const gs of groundStations) {
250
+ const windows = [];
251
+ const isVisible = Math.abs(gs.latitude) <= inclination + 15;
252
+ if (!isVisible) continue;
253
+ const passesForStation = Math.floor(passesPerDay * (0.3 + 0.7 * Math.cos(gs.latitude * Math.PI / 180)));
254
+ for (let i = 0; i < passesForStation; i++) {
255
+ const passStart = now + i * duration * 3600 * 1e3 / passesForStation;
256
+ const passStart2 = passStart + (Math.random() - 0.5) * orbitPeriod * 500;
257
+ const maxEl = minElevation + Math.random() * (85 - minElevation);
258
+ const passDuration = 10 * 60 * 1e3 * (1 - maxEl / 180);
259
+ const signalQuality = maxEl > 45 ? "good" : maxEl > 20 ? "marginal" : "poor";
260
+ windows.push({
261
+ start: passStart2,
262
+ end: passStart2 + passDuration,
263
+ label: `Pass ${i + 1}`,
264
+ signalQuality,
265
+ elevation: Math.round(maxEl)
266
+ });
267
+ }
268
+ result.push({ id: gs.name, name: gs.name, windows });
269
+ }
270
+ return result;
271
+ }
272
+ function generateHeliocentricOrbits(config = {}) {
273
+ const {
274
+ bodies = [
275
+ { name: "Earth", type: "planet", semiMajorAxis: 1, eccentricity: 0.017, inclination: 0, argumentOfPerihelion: 102, trueAnomaly: 0, color: "#3498db" },
276
+ { name: "Mars", type: "planet", semiMajorAxis: 1.52, eccentricity: 0.093, inclination: 1.85, argumentOfPerihelion: 286, trueAnomaly: 45, color: "#e74c3c" },
277
+ { name: "Voyager 1", type: "spacecraft", semiMajorAxis: 40, eccentricity: 0.99, inclination: 35, argumentOfPerihelion: 0, trueAnomaly: 170, color: "#3E3CFF" }
278
+ ],
279
+ orbitPoints = 72
280
+ } = config;
281
+ return bodies.map((body) => {
282
+ const { semiMajorAxis: a, eccentricity: e, inclination: i, argumentOfPerihelion: omega, trueAnomaly: nu } = body;
283
+ const orbitPath = [];
284
+ for (let j = 0; j <= orbitPoints; j++) {
285
+ const angle = j / orbitPoints * 2 * Math.PI;
286
+ const r2 = a * (1 - e * e) / (1 + e * Math.cos(angle));
287
+ const xOrbit2 = r2 * Math.cos(angle);
288
+ const yOrbit2 = r2 * Math.sin(angle);
289
+ const omegaRad2 = omega * Math.PI / 180;
290
+ const iRad2 = i * Math.PI / 180;
291
+ const x2 = xOrbit2 * Math.cos(omegaRad2) - yOrbit2 * Math.cos(iRad2) * Math.sin(omegaRad2);
292
+ const y2 = xOrbit2 * Math.sin(omegaRad2) + yOrbit2 * Math.cos(iRad2) * Math.cos(omegaRad2);
293
+ const z2 = yOrbit2 * Math.sin(iRad2);
294
+ orbitPath.push({ x: x2, y: y2, z: z2 });
295
+ }
296
+ const nuRad = nu * Math.PI / 180;
297
+ const r = a * (1 - e * e) / (1 + e * Math.cos(nuRad));
298
+ const xOrbit = r * Math.cos(nuRad);
299
+ const yOrbit = r * Math.sin(nuRad);
300
+ const omegaRad = omega * Math.PI / 180;
301
+ const iRad = i * Math.PI / 180;
302
+ const x = xOrbit * Math.cos(omegaRad) - yOrbit * Math.cos(iRad) * Math.sin(omegaRad);
303
+ const y = xOrbit * Math.sin(omegaRad) + yOrbit * Math.cos(iRad) * Math.cos(omegaRad);
304
+ const z = yOrbit * Math.sin(iRad);
305
+ const period = 365.25 * Math.pow(a, 1.5);
306
+ return {
307
+ name: body.name,
308
+ type: body.type,
309
+ x,
310
+ y,
311
+ z,
312
+ orbitPath,
313
+ color: body.color,
314
+ semiMajorAxis: a,
315
+ period,
316
+ inclination: i,
317
+ size: body.type === "planet" ? 20 : 12
318
+ };
319
+ });
320
+ }
321
+ function generateSpectrum(config = {}) {
322
+ const {
323
+ centerFrequency = 22e8,
324
+ bandwidth = 1e7,
325
+ points = 256,
326
+ noiseFloor = -120,
327
+ signals = [
328
+ { frequency: 22e8, power: -75, bandwidth: 1e5 },
329
+ { frequency: 21975e5, power: -80, bandwidth: 1e4 },
330
+ { frequency: 22025e5, power: -95, bandwidth: 5e4 }
331
+ ]
332
+ } = config;
333
+ const data = [];
334
+ const freqMin = centerFrequency - bandwidth / 2;
335
+ const freqStep = bandwidth / points;
336
+ for (let i = 0; i < points; i++) {
337
+ const freq = freqMin + i * freqStep;
338
+ let power = noiseFloor + 3 * (Math.random() - 0.5);
339
+ for (const signal of signals) {
340
+ const distance = Math.abs(freq - signal.frequency);
341
+ if (distance < signal.bandwidth * 2) {
342
+ const attenuation = 20 * Math.log10(1 + distance / (signal.bandwidth / 2));
343
+ power = Math.max(power, signal.power - attenuation);
344
+ }
345
+ }
346
+ data.push({ frequency: freq, power: Math.round(power * 10) / 10 });
347
+ }
348
+ return data;
349
+ }
350
+ function generateEclipseTimeline(config = {}) {
351
+ const {
352
+ orbitPeriod = 5400,
353
+ eclipseFraction = 0.38,
354
+ orbits = 5,
355
+ includePenumbra = true
356
+ } = config;
357
+ const data = [];
358
+ const now = Date.now();
359
+ const penumbraDuration = includePenumbra ? 60 : 0;
360
+ for (let orbit = 0; orbit < orbits; orbit++) {
361
+ const orbitStart = now + orbit * orbitPeriod * 1e3;
362
+ const eclipseStart = orbitStart + orbitPeriod * 0.4 * 1e3;
363
+ const eclipseEnd = eclipseStart + orbitPeriod * eclipseFraction * 1e3;
364
+ data.push({
365
+ start: orbitStart,
366
+ end: eclipseStart - penumbraDuration * 1e3,
367
+ type: "sunlight",
368
+ orbitNumber: orbit + 1
369
+ });
370
+ if (includePenumbra) {
371
+ data.push({
372
+ start: eclipseStart - penumbraDuration * 1e3,
373
+ end: eclipseStart,
374
+ type: "penumbra",
375
+ orbitNumber: orbit + 1
376
+ });
377
+ }
378
+ data.push({
379
+ start: eclipseStart,
380
+ end: eclipseEnd,
381
+ type: "umbra",
382
+ orbitNumber: orbit + 1
383
+ });
384
+ if (includePenumbra) {
385
+ data.push({
386
+ start: eclipseEnd,
387
+ end: eclipseEnd + penumbraDuration * 1e3,
388
+ type: "penumbra",
389
+ orbitNumber: orbit + 1
390
+ });
391
+ }
392
+ data.push({
393
+ start: eclipseEnd + penumbraDuration * 1e3,
394
+ end: orbitStart + orbitPeriod * 1e3,
395
+ type: "sunlight",
396
+ orbitNumber: orbit + 1
397
+ });
398
+ }
399
+ return data;
400
+ }
401
+ function generateConstellationCoverage(config = {}) {
402
+ const {
403
+ latResolution = 5,
404
+ lonResolution = 5,
405
+ planes = 6,
406
+ satsPerPlane = 22,
407
+ altitude = 550,
408
+ inclination = 53,
409
+ metric = "visibility"
410
+ } = config;
411
+ const latitudes = [];
412
+ const longitudes = [];
413
+ const coverage = [];
414
+ for (let lat = -90; lat <= 90; lat += latResolution) latitudes.push(lat);
415
+ for (let lon = -180; lon < 180; lon += lonResolution) longitudes.push(lon);
416
+ const earthRadius = 6371;
417
+ const footprintAngle = Math.acos(earthRadius / (earthRadius + altitude)) * (180 / Math.PI);
418
+ const totalSats = planes * satsPerPlane;
419
+ for (let li = 0; li < latitudes.length; li++) {
420
+ const row = [];
421
+ const lat = latitudes[li];
422
+ for (let lj = 0; lj < longitudes.length; lj++) {
423
+ let value;
424
+ const latFactor = Math.abs(lat) <= inclination ? 1 : Math.max(0, 1 - (Math.abs(lat) - inclination) / 20);
425
+ const densityFactor = totalSats / 100;
426
+ const footprintOverlap = Math.min(1, densityFactor * (footprintAngle / 30));
427
+ if (metric === "visibility") {
428
+ value = Math.min(100, footprintOverlap * latFactor * 100);
429
+ if (Math.abs(lat) > 60) value = Math.min(100, value * 1.2);
430
+ value += (Math.random() - 0.5) * 5;
431
+ value = Math.max(0, Math.min(100, value));
432
+ } else if (metric === "revisitTime") {
433
+ const baseRevisit = 90 / (footprintOverlap * latFactor + 0.01);
434
+ value = Math.max(1, Math.min(120, baseRevisit + (Math.random() - 0.5) * 10));
435
+ } else {
436
+ const slantRange = altitude / Math.sin(Math.max(0.1, latFactor * Math.PI / 4));
437
+ value = slantRange / 300 + 2 + Math.random() * 2;
438
+ }
439
+ row.push(Math.round(value * 10) / 10);
440
+ }
441
+ coverage.push(row);
442
+ }
443
+ const units = { visibility: "%", revisitTime: "min", latency: "ms" };
444
+ return { latitudes, longitudes, coverage, unit: units[metric] || "%" };
445
+ }
446
+ function generateConjunctionEvents(config = {}) {
447
+ const {
448
+ primaryName = "SAT-ALPHA",
449
+ eventCount = 12,
450
+ timeRangeHours = 72,
451
+ includeHighRisk = true
452
+ } = config;
453
+ const secondaries = [
454
+ "COSMOS-2251 DEB",
455
+ "FENGYUN-1C DEB",
456
+ "SL-16 R/B",
457
+ "IRIDIUM 33 DEB",
458
+ "NOAA 17 DEB",
459
+ "ATLAS 5 CENTAUR R/B",
460
+ "BREEZE-M DEB",
461
+ "CZ-3B R/B",
462
+ "UNKNOWN DEB",
463
+ "MICROSAT-R DEB",
464
+ "DELTA 2 R/B",
465
+ "H-2A R/B"
466
+ ];
467
+ const now = Date.now();
468
+ const events = [];
469
+ for (let i = 0; i < eventCount; i++) {
470
+ const tca = now + Math.random() * timeRangeHours * 36e5;
471
+ let missDistance;
472
+ if (includeHighRisk && i < 2) {
473
+ missDistance = 0.1 + Math.random() * 0.9;
474
+ } else if (i < 5) {
475
+ missDistance = 1 + Math.random() * 4;
476
+ } else {
477
+ missDistance = 5 + Math.random() * 15;
478
+ }
479
+ const probability = Math.pow(10, -3 - missDistance * 1.5 + Math.random());
480
+ const relativeVelocity = 5 + Math.random() * 12;
481
+ const angle = Math.random() * 2 * Math.PI;
482
+ const incAngle = (Math.random() - 0.5) * Math.PI;
483
+ const radius = 6800 + Math.random() * 600;
484
+ events.push({
485
+ id: `CDM-${String(i + 1).padStart(3, "0")}`,
486
+ primary: primaryName,
487
+ secondary: secondaries[i % secondaries.length],
488
+ tca,
489
+ missDistance: Math.round(missDistance * 1e3) / 1e3,
490
+ probability: Math.max(1e-10, probability),
491
+ relativeVelocity: Math.round(relativeVelocity * 100) / 100,
492
+ position: {
493
+ x: Math.round(radius * Math.cos(angle) * Math.cos(incAngle)),
494
+ y: Math.round(radius * Math.sin(angle) * Math.cos(incAngle)),
495
+ z: Math.round(radius * Math.sin(incAngle))
496
+ },
497
+ uncertainty: {
498
+ rx: 20 + Math.random() * 100,
499
+ ry: 50 + Math.random() * 300,
500
+ rz: 10 + Math.random() * 60
501
+ }
502
+ });
503
+ }
504
+ return events.sort((a, b) => a.tca - b.tca);
505
+ }
506
+ export {
507
+ generateAttitudeData,
508
+ generateConjunctionEvents,
509
+ generateConstellationCoverage,
510
+ generateContactWindows,
511
+ generateDopplerData,
512
+ generateEclipseTimeline,
513
+ generateHeliocentricOrbits,
514
+ generatePowerData,
515
+ generateSpectrum,
516
+ generateWaterfallData
517
+ };
518
+ //# sourceMappingURL=generators.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generators.js","sources":["../../../../src/react/charts/unified/generators.ts"],"sourcesContent":["/**\r\n * @zendir/ui - Realistic Space Data Generators\r\n * \r\n * Generate realistic demo data for space-specific charts.\r\n * All generators produce physically accurate data following orbital mechanics\r\n * and RF/signal propagation principles.\r\n */\r\n\r\nimport type {\r\n PowerChartData,\r\n AttitudeChartData,\r\n WaterfallDataPoint,\r\n DopplerTrackPoint,\r\n SatellitePassInfo,\r\n SpectrumDataPoint,\r\n ContactWindowData,\r\n EclipseTimelineData,\r\n HeliocentricOrbitData,\r\n} from './types';\r\nimport type { ConjunctionEvent as _ConjunctionEvent } from './domain';\r\n\r\n// =============================================================================\r\n// Constants\r\n// =============================================================================\r\n\r\nconst SPEED_OF_LIGHT = 299792458; // m/s\r\nconst EARTH_RADIUS = 6371; // km\r\nconst MU_EARTH = 398600.4418; // km³/s² (Earth gravitational parameter)\r\n\r\n// =============================================================================\r\n// Power System Generator\r\n// =============================================================================\r\n\r\nexport interface PowerGeneratorConfig {\r\n /** Orbit period in seconds (default: 5400 = 90 min LEO) */\r\n orbitPeriod?: number;\r\n /** Eclipse duration in seconds (default: 2100 = 35 min) */\r\n eclipseDuration?: number;\r\n /** Solar array power output at optimal pointing (W) */\r\n solarArrayPower?: number;\r\n /** Average load power (W) */\r\n loadPower?: number;\r\n /** Battery capacity (Wh) */\r\n batteryCapacity?: number;\r\n /** Initial state of charge (%) */\r\n initialSoC?: number;\r\n /** Load reduction factor during eclipse (0-1) */\r\n eclipseLoadReduction?: number;\r\n /** Simulation duration in orbits */\r\n orbits?: number;\r\n /** Time resolution in seconds */\r\n resolution?: number;\r\n}\r\n\r\n/**\r\n * Generate realistic LEO satellite power system telemetry\r\n * \r\n * Models:\r\n * - Solar array power variation with orbit phase (cosine pointing loss)\r\n * - Eclipse periods with zero solar input\r\n * - Battery charge/discharge with 90% round-trip efficiency\r\n * - Load shedding during eclipse\r\n */\r\nexport function generatePowerData(config: PowerGeneratorConfig = {}): PowerChartData[] {\r\n const {\r\n orbitPeriod = 5400,\r\n eclipseDuration = 2100,\r\n solarArrayPower = 85,\r\n loadPower = 45,\r\n batteryCapacity = 300,\r\n initialSoC = 80,\r\n eclipseLoadReduction = 0.7,\r\n orbits = 2,\r\n resolution = 60,\r\n } = config;\r\n\r\n const data: PowerChartData[] = [];\r\n const totalDuration = orbitPeriod * orbits;\r\n let batterySoC = initialSoC;\r\n const batteryEfficiency = 0.9;\r\n \r\n // Eclipse starts at 40% of orbit, ends at 78% (typical LEO)\r\n const eclipseStart = 0.4;\r\n const eclipseEnd = eclipseStart + eclipseDuration / orbitPeriod;\r\n\r\n for (let t = 0; t < totalDuration; t += resolution) {\r\n const orbitPhase = (t % orbitPeriod) / orbitPeriod;\r\n const inEclipse = orbitPhase >= eclipseStart && orbitPhase < eclipseEnd;\r\n \r\n // Solar power with pointing variation (±10% sinusoidal)\r\n const pointingFactor = 0.9 + 0.1 * Math.sin(orbitPhase * Math.PI * 2);\r\n const solarPower = inEclipse ? 0 : solarArrayPower * pointingFactor;\r\n \r\n // Load power (reduced during eclipse for power conservation)\r\n const consumption = inEclipse ? loadPower * eclipseLoadReduction : loadPower;\r\n \r\n // Net power (positive = charging, negative = discharging)\r\n const netPower = solarPower - consumption;\r\n \r\n // Battery state of charge update\r\n const energyDelta = (netPower * resolution) / 3600; // Wh\r\n const efficiencyFactor = netPower > 0 ? batteryEfficiency : 1 / batteryEfficiency;\r\n batterySoC += (energyDelta * efficiencyFactor / batteryCapacity) * 100;\r\n batterySoC = Math.max(0, Math.min(100, batterySoC));\r\n \r\n data.push({\r\n time: Date.now() + t * 1000,\r\n solarPower: Math.round(solarPower * 10) / 10,\r\n consumption: Math.round(consumption * 10) / 10,\r\n netPower: Math.round(netPower * 10) / 10,\r\n batteryPercent: Math.round(batterySoC * 10) / 10,\r\n eclipse: inEclipse,\r\n });\r\n }\r\n\r\n return data;\r\n}\r\n\r\n// =============================================================================\r\n// Attitude Generator\r\n// =============================================================================\r\n\r\nexport interface AttitudeGeneratorConfig {\r\n /** Initial attitude [roll, pitch, yaw] in degrees */\r\n initialAttitude?: [number, number, number];\r\n /** Target attitude [roll, pitch, yaw] in degrees */\r\n targetAttitude?: [number, number, number];\r\n /** Slew rate in deg/s */\r\n slewRate?: number;\r\n /** Settling time in seconds */\r\n settleTime?: number;\r\n /** Pointing jitter RMS in degrees */\r\n jitter?: number;\r\n /** Sampling rate in Hz */\r\n samplingRate?: number;\r\n /** Duration in seconds */\r\n duration?: number;\r\n}\r\n\r\n/**\r\n * Generate realistic spacecraft attitude telemetry\r\n * \r\n * Models:\r\n * - Slew maneuver with trapezoidal velocity profile\r\n * - Fine pointing settling with exponential decay\r\n * - Random jitter (Gaussian distribution)\r\n * - Angular velocity from finite differences\r\n */\r\nexport function generateAttitudeData(config: AttitudeGeneratorConfig = {}): AttitudeChartData[] {\r\n const {\r\n initialAttitude = [0, -45, 90],\r\n targetAttitude = [0, 30, 180],\r\n slewRate = 0.5,\r\n settleTime = 30,\r\n jitter = 0.02,\r\n samplingRate = 1,\r\n duration = 300,\r\n } = config;\r\n\r\n const data: AttitudeChartData[] = [];\r\n const dt = 1 / samplingRate;\r\n \r\n // Calculate slew geometry\r\n const deltaAttitude = targetAttitude.map((t, i) => t - initialAttitude[i]);\r\n const totalAngle = Math.sqrt(deltaAttitude.reduce((sum, d) => sum + d * d, 0));\r\n const slewDuration = totalAngle / slewRate;\r\n const _unitVector = deltaAttitude.map(d => d / totalAngle);\r\n\r\n // Gaussian random for jitter\r\n const gaussRandom = () => {\r\n const u1 = Math.random();\r\n const u2 = Math.random();\r\n return Math.sqrt(-2 * Math.log(u1)) * Math.cos(2 * Math.PI * u2);\r\n };\r\n\r\n let prevRoll = initialAttitude[0];\r\n let prevPitch = initialAttitude[1];\r\n let prevYaw = initialAttitude[2];\r\n\r\n for (let t = 0; t < duration; t += dt) {\r\n let roll: number, pitch: number, yaw: number;\r\n let pointingError: number;\r\n\r\n if (t < slewDuration) {\r\n // During slew\r\n const progress = t / slewDuration;\r\n const smoothProgress = 0.5 - 0.5 * Math.cos(progress * Math.PI); // Smooth ease\r\n roll = initialAttitude[0] + deltaAttitude[0] * smoothProgress;\r\n pitch = initialAttitude[1] + deltaAttitude[1] * smoothProgress;\r\n yaw = initialAttitude[2] + deltaAttitude[2] * smoothProgress;\r\n pointingError = totalAngle * (1 - progress);\r\n } else if (t < slewDuration + settleTime) {\r\n // Settling phase\r\n const settleProgress = (t - slewDuration) / settleTime;\r\n const decayFactor = Math.exp(-3 * settleProgress);\r\n const overshoot = 0.05 * Math.sin(settleProgress * Math.PI * 4) * decayFactor;\r\n \r\n roll = targetAttitude[0] + overshoot + jitter * gaussRandom();\r\n pitch = targetAttitude[1] + overshoot + jitter * gaussRandom();\r\n yaw = targetAttitude[2] + overshoot + jitter * gaussRandom();\r\n pointingError = Math.abs(overshoot) + jitter * Math.abs(gaussRandom());\r\n } else {\r\n // Fine pointing\r\n roll = targetAttitude[0] + jitter * gaussRandom();\r\n pitch = targetAttitude[1] + jitter * gaussRandom();\r\n yaw = targetAttitude[2] + jitter * gaussRandom();\r\n pointingError = jitter * Math.sqrt(\r\n Math.pow(gaussRandom(), 2) + Math.pow(gaussRandom(), 2) + Math.pow(gaussRandom(), 2)\r\n );\r\n }\r\n\r\n // Angular velocity from finite differences\r\n const angularVelocity = Math.sqrt(\r\n Math.pow((roll - prevRoll) / dt, 2) +\r\n Math.pow((pitch - prevPitch) / dt, 2) +\r\n Math.pow((yaw - prevYaw) / dt, 2)\r\n );\r\n\r\n data.push({\r\n time: Date.now() + t * 1000,\r\n roll: Math.round(roll * 100) / 100,\r\n pitch: Math.round(pitch * 100) / 100,\r\n yaw: Math.round(yaw * 100) / 100,\r\n pointingError: Math.round(pointingError * 1000) / 1000,\r\n angularVelocity: Math.round(angularVelocity * 1000) / 1000,\r\n });\r\n\r\n prevRoll = roll;\r\n prevPitch = pitch;\r\n prevYaw = yaw;\r\n }\r\n\r\n return data;\r\n}\r\n\r\n// =============================================================================\r\n// Doppler Track Generator\r\n// =============================================================================\r\n\r\nexport interface DopplerGeneratorConfig {\r\n /** Carrier frequency in Hz */\r\n carrierFrequency?: number;\r\n /** Satellite altitude in km */\r\n altitude?: number;\r\n /** Maximum elevation angle in degrees */\r\n maxElevation?: number;\r\n /** Pass duration in seconds */\r\n passDuration?: number;\r\n /** Time resolution in seconds */\r\n resolution?: number;\r\n /** Add measurement noise (Hz RMS) */\r\n measurementNoise?: number;\r\n /** Satellite name */\r\n satelliteName?: string;\r\n /** Ground station name */\r\n groundStation?: string;\r\n}\r\n\r\n/**\r\n * Generate realistic Doppler tracking data for a satellite pass\r\n * \r\n * Uses simplified orbital mechanics:\r\n * - Circular orbit at specified altitude\r\n * - Pass geometry from maximum elevation\r\n * - Doppler shift from velocity projection\r\n */\r\nexport function generateDopplerData(config: DopplerGeneratorConfig = {}): {\r\n predicted: DopplerTrackPoint[];\r\n measured: DopplerTrackPoint[];\r\n passInfo: SatellitePassInfo;\r\n} {\r\n const {\r\n carrierFrequency = 437e6, // 437 MHz (UHF amateur)\r\n altitude = 400, // km (ISS altitude)\r\n maxElevation = 75,\r\n passDuration = 600, // 10 minutes\r\n resolution = 1,\r\n measurementNoise = 50, // Hz\r\n satelliteName = 'ISS',\r\n groundStation: _groundStation = 'Home Station',\r\n } = config;\r\n\r\n const predicted: DopplerTrackPoint[] = [];\r\n const measured: DopplerTrackPoint[] = [];\r\n\r\n // Orbital velocity\r\n const orbitRadius = EARTH_RADIUS + altitude;\r\n const orbitalVelocity = Math.sqrt(MU_EARTH / orbitRadius); // km/s\r\n\r\n // Maximum Doppler shift\r\n const maxDoppler = carrierFrequency * (orbitalVelocity * 1000 / SPEED_OF_LIGHT);\r\n\r\n // Calculate pass geometry\r\n const elevationRad = (maxElevation * Math.PI) / 180;\r\n const passHalfAngle = Math.PI / 2 - elevationRad;\r\n\r\n const now = Date.now();\r\n const aos = now;\r\n const los = now + passDuration * 1000;\r\n const tca = now + (passDuration / 2) * 1000;\r\n\r\n // Gaussian random for noise\r\n const gaussRandom = () => {\r\n const u1 = Math.random();\r\n const u2 = Math.random();\r\n return Math.sqrt(-2 * Math.log(u1)) * Math.cos(2 * Math.PI * u2);\r\n };\r\n\r\n for (let t = 0; t <= passDuration; t += resolution) {\r\n const time = now + t * 1000;\r\n const passPhase = t / passDuration; // 0 to 1\r\n const angle = (passPhase - 0.5) * 2 * passHalfAngle; // -halfAngle to +halfAngle\r\n\r\n // Doppler shift (maximum at edges, zero at TCA)\r\n const dopplerShift = maxDoppler * Math.sin(angle);\r\n\r\n // Elevation angle (maximum at TCA)\r\n const elevation = maxElevation * Math.cos(angle);\r\n\r\n predicted.push({\r\n time,\r\n dopplerShift: Math.round(dopplerShift),\r\n elevation: Math.round(elevation * 10) / 10,\r\n });\r\n\r\n // Simulated measured data with noise and bias\r\n const measuredShift = dopplerShift + measurementNoise * gaussRandom();\r\n measured.push({\r\n time,\r\n dopplerShift: Math.round(measuredShift),\r\n elevation: Math.round(elevation * 10) / 10,\r\n });\r\n }\r\n\r\n const passInfo: SatellitePassInfo = {\r\n name: satelliteName,\r\n aos: new Date(aos),\r\n los: new Date(los),\r\n tca: new Date(tca),\r\n maxElevation,\r\n };\r\n\r\n return { predicted, measured, passInfo };\r\n}\r\n\r\n// =============================================================================\r\n// Waterfall/Spectrum Generator\r\n// =============================================================================\r\n\r\nexport interface WaterfallGeneratorConfig {\r\n /** Center frequency in Hz */\r\n centerFrequency?: number;\r\n /** Bandwidth in Hz */\r\n bandwidth?: number;\r\n /** Number of FFT bins */\r\n fftBins?: number;\r\n /** Noise floor in dBm */\r\n noiseFloor?: number;\r\n /** Number of time samples */\r\n timeSamples?: number;\r\n /** Signals to include */\r\n signals?: Array<{\r\n /** Offset from center frequency (Hz) */\r\n offset: number;\r\n /** Signal power (dBm) */\r\n power: number;\r\n /** Signal bandwidth (Hz) */\r\n bandwidth: number;\r\n /** Signal type */\r\n type: 'cw' | 'ssb' | 'digital' | 'beacon';\r\n /** Doppler drift (Hz/s) - for moving signals */\r\n dopplerDrift?: number;\r\n }>;\r\n}\r\n\r\n/**\r\n * Generate realistic RF waterfall data\r\n * \r\n * Models:\r\n * - Gaussian noise floor\r\n * - CW signals (narrow spikes)\r\n * - SSB signals (offset sideband with modulation)\r\n * - Digital signals (wider, flatter spectrum)\r\n * - Time-varying Doppler drift\r\n */\r\nexport function generateWaterfallData(config: WaterfallGeneratorConfig = {}): WaterfallDataPoint[] {\r\n const {\r\n centerFrequency = 437e6,\r\n bandwidth = 100e3,\r\n fftBins = 512,\r\n noiseFloor = -120,\r\n timeSamples = 100,\r\n signals = [\r\n { offset: -30e3, power: -80, bandwidth: 500, type: 'cw' as const },\r\n { offset: 0, power: -75, bandwidth: 20e3, type: 'digital' as const },\r\n { offset: 25e3, power: -90, bandwidth: 3e3, type: 'ssb' as const },\r\n ],\r\n } = config;\r\n\r\n const data: WaterfallDataPoint[] = [];\r\n const freqStep = bandwidth / fftBins;\r\n const noiseStdDev = 3; // dB variation\r\n\r\n // Gaussian random\r\n const gaussRandom = () => {\r\n const u1 = Math.random();\r\n const u2 = Math.random();\r\n return Math.sqrt(-2 * Math.log(u1)) * Math.cos(2 * Math.PI * u2);\r\n };\r\n\r\n // Pre-compute frequency bins (constant for all time slices)\r\n const frequencies: number[] = [];\r\n for (let f = 0; f < fftBins; f++) {\r\n frequencies.push(centerFrequency + (f - fftBins / 2) * freqStep);\r\n }\r\n\r\n for (let t = 0; t < timeSamples; t++) {\r\n const powers: number[] = [];\r\n const time = Date.now() - (timeSamples - t) * 100; // 100ms between samples\r\n\r\n for (let f = 0; f < fftBins; f++) {\r\n const freqOffset = (f - fftBins / 2) * freqStep;\r\n let power = noiseFloor + noiseStdDev * gaussRandom();\r\n\r\n // Add signals\r\n for (const signal of signals) {\r\n const signalOffset = signal.offset + (signal.dopplerDrift || 0) * (t / 10);\r\n const distance = Math.abs(freqOffset - signalOffset);\r\n \r\n if (signal.type === 'cw') {\r\n // Narrow CW signal (sinc-like)\r\n if (distance < signal.bandwidth * 2) {\r\n const shape = Math.pow(Math.sin(Math.PI * distance / signal.bandwidth) / \r\n (Math.PI * distance / signal.bandwidth + 0.001), 2);\r\n power = Math.max(power, signal.power + 10 * Math.log10(shape + 0.001) + gaussRandom());\r\n }\r\n } else if (signal.type === 'ssb') {\r\n // SSB with voice-like modulation\r\n if (freqOffset >= signalOffset && freqOffset <= signalOffset + signal.bandwidth) {\r\n const modulation = 0.5 + 0.5 * Math.sin(t * 0.3 + f * 0.1);\r\n power = Math.max(power, signal.power - 3 + 6 * modulation + 2 * gaussRandom());\r\n }\r\n } else if (signal.type === 'digital') {\r\n // Digital signal (flat-topped with rolloff)\r\n if (distance < signal.bandwidth / 2) {\r\n power = Math.max(power, signal.power + gaussRandom());\r\n } else if (distance < signal.bandwidth) {\r\n const rolloff = 1 - (distance - signal.bandwidth / 2) / (signal.bandwidth / 2);\r\n power = Math.max(power, signal.power - 20 * (1 - rolloff) + gaussRandom());\r\n }\r\n } else if (signal.type === 'beacon') {\r\n // Beacon (stable CW)\r\n if (distance < signal.bandwidth) {\r\n power = Math.max(power, signal.power);\r\n }\r\n }\r\n }\r\n\r\n powers.push(Math.round(power * 10) / 10);\r\n }\r\n\r\n data.push({ time, frequencies: [...frequencies], powers });\r\n }\r\n\r\n return data;\r\n}\r\n\r\n// =============================================================================\r\n// Contact Window Generator\r\n// =============================================================================\r\n\r\nexport interface ContactGeneratorConfig {\r\n /** Ground stations */\r\n groundStations?: Array<{\r\n name: string;\r\n latitude: number;\r\n longitude: number;\r\n }>;\r\n /** Satellite altitude (km) */\r\n altitude?: number;\r\n /** Orbit inclination (degrees) */\r\n inclination?: number;\r\n /** Simulation duration (hours) */\r\n duration?: number;\r\n /** Minimum elevation (degrees) */\r\n minElevation?: number;\r\n}\r\n\r\n/**\r\n * Generate realistic ground station contact windows\r\n * \r\n * Uses simplified visibility calculation based on:\r\n * - Orbital period from altitude\r\n * - Swath width from elevation angle\r\n * - Ground track precession\r\n */\r\nexport function generateContactWindows(config: ContactGeneratorConfig = {}): ContactWindowData[] {\r\n const {\r\n groundStations = [\r\n { name: 'Svalbard (SG)', latitude: 78.2, longitude: 15.4 },\r\n { name: 'McMurdo (MCM)', latitude: -77.8, longitude: 166.7 },\r\n { name: 'Hawaii (HBK)', latitude: 21.5, longitude: -158.0 },\r\n { name: 'Kiruna (KIR)', latitude: 67.9, longitude: 20.9 },\r\n ],\r\n altitude = 400,\r\n inclination = 51.6,\r\n duration = 24,\r\n minElevation = 5,\r\n } = config;\r\n\r\n const result: ContactWindowData[] = [];\r\n const now = Date.now();\r\n const orbitPeriod = 2 * Math.PI * Math.sqrt(Math.pow(EARTH_RADIUS + altitude, 3) / MU_EARTH);\r\n const passesPerDay = Math.floor((duration * 3600) / orbitPeriod);\r\n\r\n for (const gs of groundStations) {\r\n const windows: ContactWindowData['windows'] = [];\r\n \r\n // Determine visibility based on latitude and inclination\r\n const isVisible = Math.abs(gs.latitude) <= inclination + 15;\r\n if (!isVisible) continue;\r\n\r\n // Generate passes based on ground station latitude\r\n const passesForStation = Math.floor(passesPerDay * (0.3 + 0.7 * Math.cos((gs.latitude * Math.PI) / 180)));\r\n \r\n for (let i = 0; i < passesForStation; i++) {\r\n const passStart = now + (i * duration * 3600 * 1000) / passesForStation;\r\n const passStart2 = passStart + (Math.random() - 0.5) * orbitPeriod * 500; // Jitter\r\n \r\n // Pass duration depends on elevation (higher el = shorter duration but better signal)\r\n const maxEl = minElevation + Math.random() * (85 - minElevation);\r\n const passDuration = 10 * 60 * 1000 * (1 - maxEl / 180); // 5-10 minutes\r\n \r\n const signalQuality = maxEl > 45 ? 'good' : maxEl > 20 ? 'marginal' : 'poor';\r\n \r\n windows.push({\r\n start: passStart2,\r\n end: passStart2 + passDuration,\r\n label: `Pass ${i + 1}`,\r\n signalQuality: signalQuality as 'good' | 'marginal' | 'poor',\r\n elevation: Math.round(maxEl),\r\n });\r\n }\r\n\r\n result.push({ id: gs.name, name: gs.name, windows });\r\n }\r\n\r\n return result;\r\n}\r\n\r\n// =============================================================================\r\n// Heliocentric Orbit Generator\r\n// =============================================================================\r\n\r\nexport interface HeliocentricGeneratorConfig {\r\n /** Bodies to include */\r\n bodies?: Array<{\r\n name: string;\r\n type: 'planet' | 'spacecraft' | 'asteroid' | 'comet';\r\n semiMajorAxis: number; // AU\r\n eccentricity: number;\r\n inclination: number; // degrees\r\n argumentOfPerihelion: number; // degrees\r\n trueAnomaly: number; // degrees (current position)\r\n color?: string;\r\n }>;\r\n /** Number of points per orbit */\r\n orbitPoints?: number;\r\n}\r\n\r\n/**\r\n * Generate heliocentric orbit visualization data\r\n */\r\nexport function generateHeliocentricOrbits(config: HeliocentricGeneratorConfig = {}): HeliocentricOrbitData[] {\r\n const {\r\n bodies = [\r\n { name: 'Earth', type: 'planet' as const, semiMajorAxis: 1.0, eccentricity: 0.017, inclination: 0, argumentOfPerihelion: 102, trueAnomaly: 0, color: '#3498db' },\r\n { name: 'Mars', type: 'planet' as const, semiMajorAxis: 1.52, eccentricity: 0.093, inclination: 1.85, argumentOfPerihelion: 286, trueAnomaly: 45, color: '#e74c3c' },\r\n { name: 'Voyager 1', type: 'spacecraft' as const, semiMajorAxis: 40, eccentricity: 0.99, inclination: 35, argumentOfPerihelion: 0, trueAnomaly: 170, color: '#3E3CFF' },\r\n ],\r\n orbitPoints = 72,\r\n } = config;\r\n\r\n return bodies.map(body => {\r\n const { semiMajorAxis: a, eccentricity: e, inclination: i, argumentOfPerihelion: omega, trueAnomaly: nu } = body;\r\n \r\n // Generate orbit path\r\n const orbitPath: { x: number; y: number; z: number }[] = [];\r\n for (let j = 0; j <= orbitPoints; j++) {\r\n const angle = (j / orbitPoints) * 2 * Math.PI;\r\n const r = (a * (1 - e * e)) / (1 + e * Math.cos(angle));\r\n \r\n // Rotate by argument of perihelion and inclination\r\n const xOrbit = r * Math.cos(angle);\r\n const yOrbit = r * Math.sin(angle);\r\n \r\n const omegaRad = (omega * Math.PI) / 180;\r\n const iRad = (i * Math.PI) / 180;\r\n \r\n const x = xOrbit * Math.cos(omegaRad) - yOrbit * Math.cos(iRad) * Math.sin(omegaRad);\r\n const y = xOrbit * Math.sin(omegaRad) + yOrbit * Math.cos(iRad) * Math.cos(omegaRad);\r\n const z = yOrbit * Math.sin(iRad);\r\n \r\n orbitPath.push({ x, y, z });\r\n }\r\n\r\n // Current position\r\n const nuRad = (nu * Math.PI) / 180;\r\n const r = (a * (1 - e * e)) / (1 + e * Math.cos(nuRad));\r\n const xOrbit = r * Math.cos(nuRad);\r\n const yOrbit = r * Math.sin(nuRad);\r\n \r\n const omegaRad = (omega * Math.PI) / 180;\r\n const iRad = (i * Math.PI) / 180;\r\n \r\n const x = xOrbit * Math.cos(omegaRad) - yOrbit * Math.cos(iRad) * Math.sin(omegaRad);\r\n const y = xOrbit * Math.sin(omegaRad) + yOrbit * Math.cos(iRad) * Math.cos(omegaRad);\r\n const z = yOrbit * Math.sin(iRad);\r\n\r\n // Orbital period in days\r\n const period = 365.25 * Math.pow(a, 1.5);\r\n\r\n return {\r\n name: body.name,\r\n type: body.type,\r\n x,\r\n y,\r\n z,\r\n orbitPath,\r\n color: body.color,\r\n semiMajorAxis: a,\r\n period,\r\n inclination: i,\r\n size: body.type === 'planet' ? 20 : 12,\r\n };\r\n });\r\n}\r\n\r\n// =============================================================================\r\n// Spectrum Generator (single FFT snapshot)\r\n// =============================================================================\r\n\r\nexport interface SpectrumGeneratorConfig {\r\n /** Center frequency (Hz) */\r\n centerFrequency?: number;\r\n /** Bandwidth (Hz) */\r\n bandwidth?: number;\r\n /** Number of points */\r\n points?: number;\r\n /** Noise floor (dBm) */\r\n noiseFloor?: number;\r\n /** Signals */\r\n signals?: Array<{\r\n frequency: number;\r\n power: number;\r\n bandwidth: number;\r\n }>;\r\n}\r\n\r\n/**\r\n * Generate a single RF spectrum snapshot\r\n */\r\nexport function generateSpectrum(config: SpectrumGeneratorConfig = {}): SpectrumDataPoint[] {\r\n const {\r\n centerFrequency = 2200e6,\r\n bandwidth = 10e6,\r\n points = 256,\r\n noiseFloor = -120,\r\n signals = [\r\n { frequency: 2200e6, power: -75, bandwidth: 100e3 },\r\n { frequency: 2197.5e6, power: -80, bandwidth: 10e3 },\r\n { frequency: 2202.5e6, power: -95, bandwidth: 50e3 },\r\n ],\r\n } = config;\r\n\r\n const data: SpectrumDataPoint[] = [];\r\n const freqMin = centerFrequency - bandwidth / 2;\r\n const freqStep = bandwidth / points;\r\n\r\n for (let i = 0; i < points; i++) {\r\n const freq = freqMin + i * freqStep;\r\n let power = noiseFloor + 3 * (Math.random() - 0.5);\r\n\r\n for (const signal of signals) {\r\n const distance = Math.abs(freq - signal.frequency);\r\n if (distance < signal.bandwidth * 2) {\r\n const attenuation = 20 * Math.log10(1 + distance / (signal.bandwidth / 2));\r\n power = Math.max(power, signal.power - attenuation);\r\n }\r\n }\r\n\r\n data.push({ frequency: freq, power: Math.round(power * 10) / 10 });\r\n }\r\n\r\n return data;\r\n}\r\n\r\n// =============================================================================\r\n// Eclipse Timeline Generator\r\n// =============================================================================\r\n\r\nexport interface EclipseGeneratorConfig {\r\n /** Orbit period (seconds) */\r\n orbitPeriod?: number;\r\n /** Eclipse fraction (0-1) */\r\n eclipseFraction?: number;\r\n /** Number of orbits */\r\n orbits?: number;\r\n /** Include penumbra */\r\n includePenumbra?: boolean;\r\n}\r\n\r\n/**\r\n * Generate eclipse timeline data\r\n */\r\nexport function generateEclipseTimeline(config: EclipseGeneratorConfig = {}): EclipseTimelineData[] {\r\n const {\r\n orbitPeriod = 5400,\r\n eclipseFraction = 0.38,\r\n orbits = 5,\r\n includePenumbra = true,\r\n } = config;\r\n\r\n const data: EclipseTimelineData[] = [];\r\n const now = Date.now();\r\n const penumbraDuration = includePenumbra ? 60 : 0; // 1 minute penumbra\r\n\r\n for (let orbit = 0; orbit < orbits; orbit++) {\r\n const orbitStart = now + orbit * orbitPeriod * 1000;\r\n const eclipseStart = orbitStart + orbitPeriod * 0.4 * 1000;\r\n const eclipseEnd = eclipseStart + orbitPeriod * eclipseFraction * 1000;\r\n\r\n // Sunlight before eclipse\r\n data.push({\r\n start: orbitStart,\r\n end: eclipseStart - penumbraDuration * 1000,\r\n type: 'sunlight',\r\n orbitNumber: orbit + 1,\r\n });\r\n\r\n // Penumbra entry\r\n if (includePenumbra) {\r\n data.push({\r\n start: eclipseStart - penumbraDuration * 1000,\r\n end: eclipseStart,\r\n type: 'penumbra',\r\n orbitNumber: orbit + 1,\r\n });\r\n }\r\n\r\n // Umbra\r\n data.push({\r\n start: eclipseStart,\r\n end: eclipseEnd,\r\n type: 'umbra',\r\n orbitNumber: orbit + 1,\r\n });\r\n\r\n // Penumbra exit\r\n if (includePenumbra) {\r\n data.push({\r\n start: eclipseEnd,\r\n end: eclipseEnd + penumbraDuration * 1000,\r\n type: 'penumbra',\r\n orbitNumber: orbit + 1,\r\n });\r\n }\r\n\r\n // Sunlight after eclipse\r\n data.push({\r\n start: eclipseEnd + penumbraDuration * 1000,\r\n end: orbitStart + orbitPeriod * 1000,\r\n type: 'sunlight',\r\n orbitNumber: orbit + 1,\r\n });\r\n }\r\n\r\n return data;\r\n}\r\n\r\n// =============================================================================\r\n// Constellation Coverage Generator\r\n// =============================================================================\r\n\r\nexport interface CoverageGeneratorConfig {\r\n /** Latitude resolution (degrees) */\r\n latResolution?: number;\r\n /** Longitude resolution (degrees) */\r\n lonResolution?: number;\r\n /** Number of orbital planes */\r\n planes?: number;\r\n /** Satellites per plane */\r\n satsPerPlane?: number;\r\n /** Orbit altitude (km) */\r\n altitude?: number;\r\n /** Orbit inclination (degrees) */\r\n inclination?: number;\r\n /** Metric to compute: 'visibility' | 'revisitTime' | 'latency' */\r\n metric?: 'visibility' | 'revisitTime' | 'latency';\r\n}\r\n\r\n/**\r\n * Generate constellation coverage grid data\r\n * \r\n * Models simplified visibility based on satellite altitude\r\n * and ground footprint overlap from Walker constellation geometry.\r\n */\r\nexport function generateConstellationCoverage(config: CoverageGeneratorConfig = {}): {\r\n latitudes: number[];\r\n longitudes: number[];\r\n coverage: number[][];\r\n unit: string;\r\n} {\r\n const {\r\n latResolution = 5,\r\n lonResolution = 5,\r\n planes = 6,\r\n satsPerPlane = 22,\r\n altitude = 550,\r\n inclination = 53,\r\n metric = 'visibility',\r\n } = config;\r\n\r\n const latitudes: number[] = [];\r\n const longitudes: number[] = [];\r\n const coverage: number[][] = [];\r\n\r\n for (let lat = -90; lat <= 90; lat += latResolution) latitudes.push(lat);\r\n for (let lon = -180; lon < 180; lon += lonResolution) longitudes.push(lon);\r\n\r\n // Footprint half-angle from altitude\r\n const earthRadius = 6371;\r\n const footprintAngle = Math.acos(earthRadius / (earthRadius + altitude)) * (180 / Math.PI);\r\n const totalSats = planes * satsPerPlane;\r\n\r\n for (let li = 0; li < latitudes.length; li++) {\r\n const row: number[] = [];\r\n const lat = latitudes[li];\r\n const _latRad = (lat * Math.PI) / 180;\r\n\r\n for (let lj = 0; lj < longitudes.length; lj++) {\r\n let value: number;\r\n\r\n // Coverage depends on latitude vs inclination\r\n const latFactor = Math.abs(lat) <= inclination\r\n ? 1.0\r\n : Math.max(0, 1.0 - (Math.abs(lat) - inclination) / 20);\r\n\r\n // Base coverage from constellation density\r\n const densityFactor = totalSats / 100;\r\n const footprintOverlap = Math.min(1, densityFactor * (footprintAngle / 30));\r\n\r\n if (metric === 'visibility') {\r\n // Percentage of time with at least 1 satellite visible\r\n value = Math.min(100, footprintOverlap * latFactor * 100);\r\n // Add polar convergence bonus\r\n if (Math.abs(lat) > 60) value = Math.min(100, value * 1.2);\r\n // Add noise\r\n value += (Math.random() - 0.5) * 5;\r\n value = Math.max(0, Math.min(100, value));\r\n } else if (metric === 'revisitTime') {\r\n // Minutes between passes (lower is better)\r\n const baseRevisit = 90 / (footprintOverlap * latFactor + 0.01);\r\n value = Math.max(1, Math.min(120, baseRevisit + (Math.random() - 0.5) * 10));\r\n } else {\r\n // Latency in ms (one-way)\r\n const slantRange = altitude / Math.sin(Math.max(0.1, latFactor * Math.PI / 4));\r\n value = (slantRange / 300) + 2 + Math.random() * 2; // ms\r\n }\r\n\r\n row.push(Math.round(value * 10) / 10);\r\n }\r\n coverage.push(row);\r\n }\r\n\r\n const units: Record<string, string> = { visibility: '%', revisitTime: 'min', latency: 'ms' };\r\n return { latitudes, longitudes, coverage, unit: units[metric] || '%' };\r\n}\r\n\r\n// =============================================================================\r\n// Conjunction Event Generator\r\n// =============================================================================\r\n\r\nexport interface ConjunctionGeneratorConfig {\r\n /** Primary object name */\r\n primaryName?: string;\r\n /** Number of events to generate */\r\n eventCount?: number;\r\n /** Time range in hours */\r\n timeRangeHours?: number;\r\n /** Include high-risk events */\r\n includeHighRisk?: boolean;\r\n}\r\n\r\n/**\r\n * Generate realistic conjunction/close approach events\r\n */\r\nexport function generateConjunctionEvents(config: ConjunctionGeneratorConfig = {}): Array<{\r\n id: string;\r\n primary: string;\r\n secondary: string;\r\n tca: number;\r\n missDistance: number;\r\n probability: number;\r\n relativeVelocity: number;\r\n position: { x: number; y: number; z: number };\r\n uncertainty?: { rx: number; ry: number; rz: number };\r\n}> {\r\n const {\r\n primaryName = 'SAT-ALPHA',\r\n eventCount = 12,\r\n timeRangeHours = 72,\r\n includeHighRisk = true,\r\n } = config;\r\n\r\n const secondaries = [\r\n 'COSMOS-2251 DEB', 'FENGYUN-1C DEB', 'SL-16 R/B', 'IRIDIUM 33 DEB',\r\n 'NOAA 17 DEB', 'ATLAS 5 CENTAUR R/B', 'BREEZE-M DEB', 'CZ-3B R/B',\r\n 'UNKNOWN DEB', 'MICROSAT-R DEB', 'DELTA 2 R/B', 'H-2A R/B',\r\n ];\r\n\r\n const now = Date.now();\r\n const events = [];\r\n\r\n for (let i = 0; i < eventCount; i++) {\r\n const tca = now + Math.random() * timeRangeHours * 3600000;\r\n let missDistance: number;\r\n\r\n if (includeHighRisk && i < 2) {\r\n missDistance = 0.1 + Math.random() * 0.9; // Critical: < 1 km\r\n } else if (i < 5) {\r\n missDistance = 1 + Math.random() * 4; // Warning: 1-5 km\r\n } else {\r\n missDistance = 5 + Math.random() * 15; // Safe: > 5 km\r\n }\r\n\r\n const probability = Math.pow(10, -3 - missDistance * 1.5 + Math.random());\r\n const relativeVelocity = 5 + Math.random() * 12;\r\n\r\n const angle = Math.random() * 2 * Math.PI;\r\n const incAngle = (Math.random() - 0.5) * Math.PI;\r\n const radius = 6800 + Math.random() * 600;\r\n\r\n events.push({\r\n id: `CDM-${String(i + 1).padStart(3, '0')}`,\r\n primary: primaryName,\r\n secondary: secondaries[i % secondaries.length],\r\n tca,\r\n missDistance: Math.round(missDistance * 1000) / 1000,\r\n probability: Math.max(1e-10, probability),\r\n relativeVelocity: Math.round(relativeVelocity * 100) / 100,\r\n position: {\r\n x: Math.round(radius * Math.cos(angle) * Math.cos(incAngle)),\r\n y: Math.round(radius * Math.sin(angle) * Math.cos(incAngle)),\r\n z: Math.round(radius * Math.sin(incAngle)),\r\n },\r\n uncertainty: {\r\n rx: 20 + Math.random() * 100,\r\n ry: 50 + Math.random() * 300,\r\n rz: 10 + Math.random() * 60,\r\n },\r\n });\r\n }\r\n\r\n return events.sort((a, b) => a.tca - b.tca);\r\n}\r\n\r\nexport default {\r\n generatePowerData,\r\n generateAttitudeData,\r\n generateDopplerData,\r\n generateWaterfallData,\r\n generateContactWindows,\r\n generateHeliocentricOrbits,\r\n generateSpectrum,\r\n generateEclipseTimeline,\r\n generateConstellationCoverage,\r\n generateConjunctionEvents,\r\n};\r\n"],"names":["r","xOrbit","yOrbit","omegaRad","iRad","x","y","z"],"mappings":"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"}