@zendir/ui 0.2.0 → 0.2.1
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.
|
@@ -68,25 +68,37 @@ function drawCoverageEllipse(ctx, centerLat, centerLon, radiusDeg, W, H, lonToX,
|
|
|
68
68
|
}
|
|
69
69
|
ctx.closePath();
|
|
70
70
|
}
|
|
71
|
-
function
|
|
71
|
+
function calculateTerminatorContinuous(date, depressionDeg = 0, numPoints = 360) {
|
|
72
72
|
const dayOfYear = Math.floor((date.getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 864e5);
|
|
73
73
|
const declination = -23.44 * Math.cos(2 * Math.PI / 365 * (dayOfYear + 10));
|
|
74
74
|
const decRad = declination * Math.PI / 180;
|
|
75
75
|
const hourAngle = (date.getUTCHours() + date.getUTCMinutes() / 60) / 24 * 360 - 180;
|
|
76
76
|
const depRad = depressionDeg * Math.PI / 180;
|
|
77
|
-
const
|
|
77
|
+
const sunset = [];
|
|
78
|
+
const sunrise = [];
|
|
78
79
|
for (let i = 0; i <= numPoints; i++) {
|
|
79
|
-
const
|
|
80
|
+
const lat = -89.999 + i / numPoints * 179.998;
|
|
81
|
+
const latRad = lat * (Math.PI / 180);
|
|
80
82
|
const cosH = -(Math.sin(depRad) + Math.sin(latRad) * Math.sin(decRad)) / (Math.cos(latRad) * Math.cos(decRad));
|
|
81
|
-
|
|
82
|
-
if (cosH
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
83
|
+
if (cosH < -1) ;
|
|
84
|
+
else if (cosH > 1) {
|
|
85
|
+
sunset.push([lat, hourAngle]);
|
|
86
|
+
sunrise.push([lat, hourAngle + 360]);
|
|
87
|
+
} else {
|
|
88
|
+
const H = Math.acos(cosH) * 180 / Math.PI;
|
|
89
|
+
sunset.push([lat, hourAngle + H]);
|
|
90
|
+
sunrise.push([lat, hourAngle + 360 - H]);
|
|
91
|
+
}
|
|
88
92
|
}
|
|
89
|
-
return
|
|
93
|
+
return { sunset, sunrise };
|
|
94
|
+
}
|
|
95
|
+
function buildNightPolygon(terminator) {
|
|
96
|
+
const { sunset, sunrise } = terminator;
|
|
97
|
+
if (!sunset.length || !sunrise.length) return [];
|
|
98
|
+
const poly = [];
|
|
99
|
+
poly.push(...sunset);
|
|
100
|
+
poly.push(...[...sunrise].reverse());
|
|
101
|
+
return poly;
|
|
90
102
|
}
|
|
91
103
|
function GroundTrackMap({
|
|
92
104
|
groundTrack,
|
|
@@ -202,63 +214,52 @@ function GroundTrackMap({
|
|
|
202
214
|
ctx.fillRect(0, 0, W, H);
|
|
203
215
|
if (showTerminator) {
|
|
204
216
|
const now = terminatorTime ?? /* @__PURE__ */ new Date();
|
|
205
|
-
const
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
{ depression:
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
ctx.beginPath();
|
|
223
|
-
if (isSunrise) {
|
|
224
|
-
ctx.moveTo(0, 0);
|
|
225
|
-
ctx.lineTo(0, H);
|
|
226
|
-
} else {
|
|
227
|
-
ctx.moveTo(W, 0);
|
|
228
|
-
ctx.lineTo(W, H);
|
|
217
|
+
const BAND_STEP = 2;
|
|
218
|
+
const MAX_DEP = 24;
|
|
219
|
+
const NIGHT_ALPHA = 0.55;
|
|
220
|
+
const bandSteps = [];
|
|
221
|
+
for (let d = 0; d <= MAX_DEP; d += BAND_STEP) {
|
|
222
|
+
const t = Math.min(d / 18, 1);
|
|
223
|
+
const alpha = t * t * (NIGHT_ALPHA * 0.82) + (d > 18 ? (d - 18) / 6 * (NIGHT_ALPHA * 0.18) : 0);
|
|
224
|
+
bandSteps.push({ depression: d, alpha: Math.min(alpha, NIGHT_ALPHA) });
|
|
225
|
+
}
|
|
226
|
+
bandSteps.push({ depression: 90, alpha: NIGHT_ALPHA });
|
|
227
|
+
let prevAlpha = 0;
|
|
228
|
+
for (const zone of bandSteps) {
|
|
229
|
+
const dep = Math.min(zone.depression, 89);
|
|
230
|
+
const bandAlpha = zone.alpha - prevAlpha;
|
|
231
|
+
if (bandAlpha < 2e-3) {
|
|
232
|
+
prevAlpha = zone.alpha;
|
|
233
|
+
continue;
|
|
229
234
|
}
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
const y = latToY(terminator[i].lat, H);
|
|
235
|
-
if (Math.abs(x - prevX) > W * 0.5 && i < terminator.length - 1) {
|
|
236
|
-
const edgeX = isSunrise ? prevX < W / 2 ? 0 : W : prevX > W / 2 ? W : 0;
|
|
237
|
-
ctx.lineTo(edgeX, y);
|
|
238
|
-
ctx.closePath();
|
|
239
|
-
ctx.fill();
|
|
240
|
-
ctx.beginPath();
|
|
241
|
-
ctx.moveTo(isSunrise ? 0 : W, y);
|
|
242
|
-
}
|
|
243
|
-
ctx.lineTo(x, y);
|
|
244
|
-
prevX = x;
|
|
235
|
+
const terminator = calculateTerminatorContinuous(now, dep);
|
|
236
|
+
if (terminator.sunset.length === 0) {
|
|
237
|
+
prevAlpha = zone.alpha;
|
|
238
|
+
continue;
|
|
245
239
|
}
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
const
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
240
|
+
const poly = buildNightPolygon(terminator);
|
|
241
|
+
if (poly.length < 3) {
|
|
242
|
+
prevAlpha = zone.alpha;
|
|
243
|
+
continue;
|
|
244
|
+
}
|
|
245
|
+
ctx.fillStyle = `rgba(0, 6, 24, ${bandAlpha.toFixed(4)})`;
|
|
246
|
+
for (const offset of [-360, 0, 360]) {
|
|
247
|
+
ctx.beginPath();
|
|
248
|
+
for (let i = 0; i < poly.length; i++) {
|
|
249
|
+
const lat = poly[i][0];
|
|
250
|
+
const lon = poly[i][1] + offset;
|
|
251
|
+
const x = lonToX(lon, W);
|
|
252
|
+
const y = latToY(lat, H);
|
|
253
|
+
if (i === 0) {
|
|
254
|
+
ctx.moveTo(x, y);
|
|
255
|
+
} else {
|
|
256
|
+
ctx.lineTo(x, y);
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
ctx.closePath();
|
|
260
|
+
ctx.fill();
|
|
260
261
|
}
|
|
261
|
-
|
|
262
|
+
prevAlpha = zone.alpha;
|
|
262
263
|
}
|
|
263
264
|
}
|
|
264
265
|
ctx.fillStyle = COLORS.land;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"GroundTrackMap.js","sources":["../../../src/react/charts/GroundTrackMap.tsx"],"sourcesContent":["/**\n * @zendir/ui - GroundTrackMap Component (Enhanced)\n * \n * Full-featured 2D world map showing spacecraft ground tracks, current positions,\n * ground station locations, coverage footprints, satellite icons, and day/night terminator.\n * \n * Inspired by Astro UXDS TT&C Monitor / GRM Dashboard ground map patterns:\n * - Satellite icons with status-colored indicators (not just dots)\n * - Ground station antenna icons with coverage circles\n * - Sensor/antenna footprint cone projections\n * - Future orbit tracks (dashed) vs past tracks (solid)\n * - Day/night terminator\n * - Multi-satellite support\n * - Interactive hover tooltips\n * - AOS/LOS markers on track\n * - Legend with satellite/station counts\n * \n * Astro UX Compliance:\n * - Theme-integrated via useTheme()\n * - Uses official Astro status colors from theme tokens\n * - Uses Astro monitoring icon patterns\n * - 14pt minimum for labels (AstroUXDS guideline)\n * - Consistent with Astro data visualization patterns\n */\n\nimport React, { useRef, useEffect, useCallback, useMemo, useState } from 'react';\nimport { useTheme } from '../theme';\nimport { AstroIcon } from '../core/AstroIcon';\nimport type { GroundTrackPoint, GroundStation } from '../types';\nimport type { GroundTrackMapLeafletProps } from './GroundTrackMapLeaflet';\n\n// =============================================================================\n// Types\n// =============================================================================\n\nexport interface SatelliteTrack {\n /** Unique satellite identifier */\n id: string;\n /** Satellite display name */\n name: string;\n /** Ground track points */\n groundTrack: GroundTrackPoint[];\n /** Track color */\n color?: string;\n /** Satellite status for icon coloring */\n status?: 'normal' | 'caution' | 'serious' | 'critical' | 'standby' | 'off';\n /** Access mask (true = in contact) */\n accessMask?: boolean[];\n /** Show this satellite's sensor footprint */\n showFootprint?: boolean;\n /** Footprint radius in degrees (sensor cone half-angle on ground) */\n footprintRadius?: number;\n /** Show future (predicted) track as dashed line */\n futureTrackIndex?: number;\n /** AOS/LOS markers */\n passMarkers?: Array<{ type: 'aos' | 'los'; latitude: number; longitude: number; label?: string }>;\n}\n\nexport interface GroundStationEnhanced extends GroundStation {\n /** Station status */\n status?: 'normal' | 'caution' | 'serious' | 'critical' | 'standby' | 'off';\n /** Show coverage circle */\n showCoverage?: boolean;\n /** Coverage radius in degrees (visibility horizon) */\n coverageRadius?: number;\n /** Station type (affects icon) */\n type?: 'dish' | 'phased-array' | 'omni' | 'relay';\n}\n\n/** \n * SRO-compatible team path format (space-range-operator CustomMap data shape).\n * Provide this as an alternative to `satellites` for plug-and-play compatibility.\n */\nexport interface TeamPath {\n /** Team/satellite name */\n name: string;\n /** Track color */\n color?: string;\n /** Ordered position array */\n positions: Array<{\n lat: number;\n lng: number;\n time?: number;\n altitude?: number | null;\n }>;\n}\n\n/** SRO-compatible ground station format */\nexport interface SROGroundStation {\n name: string;\n latitude: number;\n longitude: number;\n altitude?: number | null;\n}\n\n/**\n * User-placed point-of-interest pin on the map.\n * Designed for collaborative scenarios where operators share annotated locations\n * across terminals in real time (e.g. targets, waypoints, hazard markers).\n */\nexport interface MapPin {\n /** Unique identifier (used as key and for update/remove callbacks) */\n id: string;\n /** Latitude in degrees (−90 … 90) */\n latitude: number;\n /** Longitude in degrees (−180 … 180) */\n longitude: number;\n /** Short label displayed next to the pin (e.g. \"Target Alpha\") */\n label?: string;\n /** Pin color — any CSS color string. Defaults to accent primary. */\n color?: string;\n /** Optional description shown in tooltip on hover */\n description?: string;\n /** Who placed the pin (display-only; not enforced) */\n createdBy?: string;\n}\n\nexport interface GroundTrackMapProps {\n /** Single satellite ground track (legacy API) */\n groundTrack?: GroundTrackPoint[];\n /** Multiple satellite tracks */\n satellites?: SatelliteTrack[];\n /** Ground stations to display */\n groundStations?: (GroundStation | GroundStationEnhanced | SROGroundStation)[];\n /** Access mask for single-satellite mode (legacy API) */\n accessMask?: boolean[];\n\n // --- SRO-compatible alternate API (space-range-operator plug-and-play) ---\n /** Team path data (SRO CustomMap format) — auto-converts to satellites internally */\n teamPaths?: TeamPath[];\n\n // --- Display options ---\n /** Show day/night terminator */\n showTerminator?: boolean;\n /** Override the time used for the day/night terminator calculation.\n * Defaults to wall-clock `new Date()` when omitted.\n * Pass a simulation/mission UTC to keep the terminator in sync with a sim clock. */\n terminatorTime?: Date;\n /** Show grid lines */\n showGrid?: boolean;\n /** Show legend */\n showLegend?: boolean;\n /** Show equator highlight */\n showEquator?: boolean;\n /** Show recenter button (SRO compat) */\n showRecenterButton?: boolean;\n\n // --- State ---\n /** Whether data is still loading (SRO compat) */\n isLoading?: boolean;\n /** Message when there is no data (SRO compat) */\n emptyMessage?: string;\n\n // --- Sizing ---\n /** Chart width (default: 100%) */\n width?: number | string;\n /** Chart height — number for fixed px, or CSS string like '100%' to fill a flex parent */\n height?: number | string;\n /** Minimum height CSS string (SRO compat, e.g. \"400px\") */\n minHeight?: string;\n\n // --- Centering ---\n /** Default center [lat, lon] (SRO compat) */\n defaultCenter?: [number, number];\n /** Default zoom level 1-10 (SRO compat, affects initial scale) */\n defaultZoom?: number;\n\n /** Custom className */\n className?: string;\n /** Click handler for satellites */\n onSatelliteClick?: (satelliteId: string) => void;\n /** Click handler for ground stations */\n onStationClick?: (stationId: string) => void;\n\n /** Map backend: 'leaflet' (default, requires leaflet peer) or 'canvas' */\n mapProvider?: 'leaflet' | 'canvas';\n /** Tile URL for Leaflet (default: CartoDB dark). Ignored when mapProvider is 'canvas'. */\n tileUrl?: string;\n\n // --- Collaborative Pins (Points of Interest) ---\n /** Array of user-placed pins displayed on the map */\n pins?: MapPin[];\n /**\n * Allow operators to place new pins by clicking the map.\n * When true, a single-click on an empty area opens a pin creation flow.\n * Requires `onPinAdd` to be set. Default false.\n */\n pinsEditable?: boolean;\n /** Called when the user places a new pin (provides lat/lon; consumer assigns id and persists) */\n onPinAdd?: (pin: Omit<MapPin, 'id'>) => void;\n /** Called when the user edits an existing pin (label, color, description change) */\n onPinUpdate?: (pin: MapPin) => void;\n /** Called when the user removes a pin */\n onPinRemove?: (pinId: string) => void;\n}\n\n// =============================================================================\n// Constants — aligned with official Astro UXDS status colors\n// These MUST match the STATUS_COLORS in unified/theme.ts and ThemeProvider\n// =============================================================================\n\nconst STATUS_COLOR_MAP: Record<string, string> = {\n normal: '#56f000', // Green — Astro UXDS \"Normal\"\n standby: '#2dccff', // Cyan — Astro UXDS \"Standby\"\n caution: '#fce83a', // Yellow — Astro UXDS \"Caution\"\n serious: '#ffb302', // Orange — Astro UXDS \"Serious\"\n critical: '#ff3838', // Red — Astro UXDS \"Critical\"\n off: '#a4abb6', // Grey — Astro UXDS \"Off\"\n};\n\nconst DEFAULT_TRACK_COLORS = [\n '#2dccff', '#3E3CFF', '#9D70FF', '#56f000', '#fce83a', '#ff7849', '#2dd4bf', '#ff3838',\n];\n\n// Earth constants for realistic coverage calculations\nconst EARTH_RADIUS_KM = 6371;\nconst _DEG_PER_KM = 1 / 111.32; // Approximate degrees per km at equator\n\n// Higher-quality world map coastline paths (Mercator-compatible)\nconst WORLD_MAP_PATHS = [\n // North America\n 'M-168,70 L-168,60 L-145,63 L-140,58 L-130,55 L-125,49 L-122,48 L-117,33 L-115,28 L-105,22 L-97,18 L-88,20 L-82,25 L-82,30 L-80,32 L-67,47 L-60,46 L-55,52 L-56,60 L-65,70 L-85,73 L-100,72 L-130,70 L-168,70',\n // South America\n 'M-82,12 L-77,8 L-73,11 L-65,5 L-60,5 L-50,0 L-45,-5 L-43,-10 L-39,-15 L-38,-22 L-40,-25 L-43,-30 L-48,-33 L-50,-40 L-55,-48 L-57,-52 L-68,-55 L-73,-50 L-78,-42 L-80,-20 L-78,-5 L-82,12',\n // Europe + Africa\n 'M-10,72 L-5,60 L-5,48 L0,44 L3,44 L5,48 L8,44 L3,37 L-5,35 L-17,28 L-17,15 L-8,10 L-4,5 L5,5 L10,2 L15,-3 L22,-12 L28,-22 L33,-30 L30,-35 L22,-35 L18,-27 L12,-20 L8,-10 L3,0 L-5,10 L-12,15 L-15,20 L-13,32 L-10,38 L0,48 L8,55 L15,58 L20,60 L25,65 L30,70 L-10,72',\n // Asia \n 'M30,70 L40,65 L50,60 L60,55 L70,50 L80,45 L90,45 L100,40 L105,35 L110,30 L115,25 L120,30 L125,35 L130,40 L135,45 L140,50 L145,55 L150,60 L155,65 L160,68 L170,70 L180,70 L180,72 L30,72 L30,70',\n // Australia\n 'M112,-12 L118,-14 L125,-15 L132,-13 L138,-15 L143,-17 L148,-20 L152,-23 L153,-28 L150,-33 L147,-37 L140,-38 L134,-36 L130,-33 L126,-30 L120,-30 L116,-28 L114,-24 L113,-18 L112,-12',\n // Antarctica\n 'M-180,-72 L180,-72 L180,-90 L-180,-90 Z',\n];\n\n// =============================================================================\n// Helper Functions\n// =============================================================================\n\n/**\n * Calculate ground station visibility radius in degrees.\n * Uses spherical Earth geometry: for a station with minimum elevation angle `elDeg`\n * and a satellite at altitude `altKm`, the half-angle subtended at Earth center is:\n * ρ = acos(R / (R + h)) − el_rad\n * Then the ground-range in degrees ≈ ρ * (180/π).\n * If no altitude context, coverageRadius prop is used directly (in degrees).\n */\nfunction _visibilityRadiusDeg(altitudeKm: number = 500, minElevationDeg: number = 5): number {\n const elRad = (minElevationDeg * Math.PI) / 180;\n const rho = Math.acos(EARTH_RADIUS_KM / (EARTH_RADIUS_KM + altitudeKm)) - elRad;\n return Math.max(0, (rho * 180) / Math.PI);\n}\n\n/**\n * Draw a ground-circle footprint using great-circle math.\n * Given a center lat/lon and an angular radius (in degrees measured along\n * Earth's surface), compute N points around the circle on the sphere, then\n * project each to equirectangular pixel coordinates.\n *\n * This produces a circle that is geographically correct AND looks natural\n * on the map — at high latitudes the circle is wider in longitude because\n * that's how equirectangular projection works, but it's capped by reality\n * rather than a simple 1/cos blowup.\n */\nfunction drawCoverageEllipse(\n ctx: CanvasRenderingContext2D,\n centerLat: number,\n centerLon: number,\n radiusDeg: number,\n W: number,\n H: number,\n lonToX: (lon: number, w: number) => number,\n latToY: (lat: number, h: number) => number,\n numSegments: number = 72,\n): void {\n const lat0 = (centerLat * Math.PI) / 180;\n const lon0 = (centerLon * Math.PI) / 180;\n const d = (radiusDeg * Math.PI) / 180; // angular radius in radians\n\n // Compute the center pixel once — all points are placed relative to this\n // so the circle never jumps across the date-line seam.\n const cx = lonToX(centerLon, W);\n const cy = latToY(centerLat, H);\n const pxPerDegLon = W / 360;\n const pxPerDegLat = H / 180;\n\n ctx.beginPath();\n for (let i = 0; i <= numSegments; i++) {\n const bearing = (i / numSegments) * 2 * Math.PI;\n\n // Great-circle destination formula\n const sinLat = Math.sin(lat0) * Math.cos(d) + Math.cos(lat0) * Math.sin(d) * Math.cos(bearing);\n const lat = Math.asin(Math.max(-1, Math.min(1, sinLat)));\n const lon = lon0 + Math.atan2(\n Math.sin(bearing) * Math.sin(d) * Math.cos(lat0),\n Math.cos(d) - Math.sin(lat0) * sinLat,\n );\n\n const pLatDeg = (lat * 180) / Math.PI;\n const pLonDeg = (lon * 180) / Math.PI;\n\n // Pixel offset from center — no normalization, so no date-line jump\n const dLon = pLonDeg - centerLon;\n const dLat = pLatDeg - centerLat;\n const px = cx + dLon * pxPerDegLon;\n const py = cy - dLat * pxPerDegLat; // Y inverted (lat goes up, pixels go down)\n\n if (i === 0) ctx.moveTo(px, py);\n else ctx.lineTo(px, py);\n }\n ctx.closePath();\n}\n//sz ee*s \n\n/**\n * Calculate solar terminator points for a given date and solar depression angle.\n *\n * `depressionDeg` offsets the zenith angle to produce twilight boundaries:\n * - 0° = geometric sunset/sunrise\n * - 6° = civil twilight (IAU/USNO)\n * - 12° = nautical twilight\n * - 18° = astronomical twilight\n */\nfunction calculateTerminator(\n date: Date,\n depressionDeg: number = 0,\n numPoints: number = 72,\n): Array<{ lat: number; lon: number }> {\n const dayOfYear = Math.floor((date.getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 86400000);\n const declination = -23.44 * Math.cos((2 * Math.PI / 365) * (dayOfYear + 10));\n const decRad = (declination * Math.PI) / 180;\n const hourAngle = ((date.getUTCHours() + date.getUTCMinutes() / 60) / 24) * 360 - 180;\n const depRad = (depressionDeg * Math.PI) / 180;\n\n const points: Array<{ lat: number; lon: number }> = [];\n for (let i = 0; i <= numPoints; i++) {\n const latRad = ((i / numPoints) * 180 - 90) * (Math.PI / 180);\n const cosH = -(Math.sin(depRad) + Math.sin(latRad) * Math.sin(decRad))\n / (Math.cos(latRad) * Math.cos(decRad));\n let lon: number;\n if (cosH < -1) lon = hourAngle + 180;\n else if (cosH > 1) lon = hourAngle;\n else lon = hourAngle + (Math.acos(cosH) * 180) / Math.PI;\n\n while (lon > 180) lon -= 360;\n while (lon < -180) lon += 360;\n points.push({ lat: (i / numPoints) * 180 - 90, lon });\n }\n return points;\n}\n\n// =============================================================================\n// Component\n// =============================================================================\n\nexport function GroundTrackMap({\n groundTrack,\n satellites = [],\n groundStations = [],\n accessMask,\n teamPaths,\n showTerminator = true,\n terminatorTime,\n showGrid = true,\n showLegend = true,\n showEquator = true,\n showRecenterButton = false,\n isLoading = false,\n emptyMessage = 'No orbital data available',\n width = '100%',\n height = '100%',\n minHeight = '400px',\n defaultCenter,\n defaultZoom,\n className = '',\n onSatelliteClick,\n onStationClick,\n mapProvider = 'leaflet',\n tileUrl,\n pins,\n pinsEditable = false,\n onPinAdd,\n onPinUpdate,\n onPinRemove,\n}: GroundTrackMapProps): React.ReactElement {\n const { tokens } = useTheme();\n const canvasRef = useRef<HTMLCanvasElement>(null);\n const containerRef = useRef<HTMLDivElement>(null);\n const [tooltip, setTooltip] = useState<{\n x: number; y: number;\n type: 'satellite' | 'station';\n name: string;\n status: string;\n statusColor: string;\n lat: number;\n lon: number;\n alt?: number;\n // Ground station extras\n stationType?: string;\n network?: string;\n coverageRadius?: number;\n // Satellite extras\n trackColor?: string;\n hasFootprint?: boolean;\n futurePoints?: number;\n pastPoints?: number;\n } | null>(null);\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n const [_viewOffset, setViewOffset] = useState<{ x: number; y: number }>({ x: 0, y: 0 });\n\n // Normalize satellites: support SRO teamPaths, legacy single-track, and native API\n const allSatellites = useMemo<SatelliteTrack[]>(() => {\n // Priority 1: native satellites prop\n if (satellites.length > 0) return satellites;\n // Priority 2: SRO teamPaths format — auto-convert\n if (teamPaths && teamPaths.length > 0) {\n return teamPaths.map((tp, idx) => ({\n id: `team-${idx}`,\n name: tp.name,\n color: tp.color,\n status: 'normal' as const,\n groundTrack: tp.positions\n .filter(p => p.lat != null && p.lng != null)\n .map(p => ({\n latitude: p.lat,\n longitude: p.lng,\n altitude: p.altitude ?? 0,\n timestamp: p.time != null ? new Date(p.time * 1000).toISOString() : new Date().toISOString(),\n })),\n }));\n }\n // Priority 3: legacy single-track API\n if (groundTrack && groundTrack.length > 0) {\n return [{\n id: 'primary',\n name: 'Satellite',\n groundTrack,\n status: 'normal',\n accessMask,\n }];\n }\n return [];\n }, [satellites, groundTrack, accessMask, teamPaths]);\n\n // Stable defaults for Leaflet so animated updates don't get new refs and trigger map recreation\n const leafletDefaultCenter = useMemo<[number, number]>(() => defaultCenter ?? [20, 0], [defaultCenter]);\n const leafletDefaultZoom = useMemo(() => defaultZoom ?? 2, [defaultZoom]);\n\n // Optional Leaflet map: load when mapProvider is 'leaflet'\n const [LeafletMap, setLeafletMap] = useState<React.ComponentType<GroundTrackMapLeafletProps> | null>(null);\n const [leafletFailed, setLeafletFailed] = useState(false);\n\n useEffect(() => {\n if (mapProvider !== 'leaflet') return;\n import('./GroundTrackMapLeaflet')\n .then((m) => setLeafletMap(() => m.GroundTrackMapLeaflet))\n .catch((err) => {\n console.warn('[GroundTrackMap] Leaflet backend failed to load. Install \"leaflet\" and ensure it is resolvable. Falling back to static canvas.', err);\n setLeafletFailed(true);\n });\n }, [mapProvider]);\n\n // All canvas-backend hooks must run unconditionally (same hook count every render)\n // Recenter handler (canvas backend)\n const handleRecenter = useCallback(() => {\n setViewOffset({ x: 0, y: 0 });\n }, []);\n\n const COLORS = useMemo(() => ({\n background: tokens.colors.background.base,\n grid: tokens.colors.border.muted,\n text: tokens.colors.text.secondary,\n accent: tokens.colors.accent.primary,\n land: 'rgba(30, 40, 58, 0.92)',\n ocean: tokens.colors.background.base,\n night: 'rgba(0, 10, 40, 0.35)',\n equator: 'rgba(88, 166, 255, 0.25)',\n }), [tokens]);\n\n const lonToX = useCallback((lon: number, w: number) => {\n let normalizedLon = lon;\n while (normalizedLon > 180) normalizedLon -= 360;\n while (normalizedLon < -180) normalizedLon += 360;\n return ((normalizedLon + 180) / 360) * w;\n }, []);\n\n const latToY = useCallback((lat: number, h: number) => {\n return ((90 - lat) / 180) * h;\n }, []);\n\n const draw = useCallback(() => {\n const canvas = canvasRef.current;\n const container = containerRef.current;\n if (!canvas || !container) return;\n\n const ctx = canvas.getContext('2d');\n if (!ctx) return;\n\n const dpr = window.devicePixelRatio || 1;\n const rect = container.getBoundingClientRect();\n const W = rect.width;\n // When height is a CSS string ('100%'), use the measured container height\n const H = typeof height === 'number' ? height : rect.height || 400;\n\n canvas.width = W * dpr;\n canvas.height = H * dpr;\n canvas.style.width = `${W}px`;\n canvas.style.height = `${H}px`;\n ctx.scale(dpr, dpr);\n\n // === Background ===\n ctx.fillStyle = COLORS.background;\n ctx.fillRect(0, 0, W, H);\n\n // === Day/Night Terminator with realistic twilight gradient ===\n // Uses IAU/USNO twilight zones: civil (6°), nautical (12°), astronomical (18°)\n if (showTerminator) {\n const now = terminatorTime ?? new Date();\n const twilightZones = [\n { depression: 6, alpha: 0.08 }, // civil twilight\n { depression: 12, alpha: 0.10 }, // nautical twilight\n { depression: 18, alpha: 0.12 }, // astronomical twilight\n { depression: 90, alpha: 0.22 }, // full night (beyond 18°)\n ];\n\n const normLon = (l: number) => { let r = l; while (r > 180) r -= 360; while (r < -180) r += 360; return r; };\n\n // Helper: fill the night side of a single terminator line\n const fillNightSide = (terminator: Array<{ lat: number; lon: number }>, isSunrise: boolean) => {\n // For sunset: fill from terminator to right edge\n // For sunrise: fill from left edge to terminator\n ctx.beginPath();\n if (isSunrise) {\n ctx.moveTo(0, 0);\n ctx.lineTo(0, H);\n } else {\n ctx.moveTo(W, 0);\n ctx.lineTo(W, H);\n }\n let prevX = isSunrise ? 0 : W;\n for (let i = terminator.length - 1; i >= 0; i--) {\n const tLon = isSunrise ? normLon(terminator[i].lon - 180) : terminator[i].lon;\n const x = lonToX(tLon, W);\n const y = latToY(terminator[i].lat, H);\n if (Math.abs(x - prevX) > W * 0.5 && i < terminator.length - 1) {\n const edgeX = isSunrise ? (prevX < W / 2 ? 0 : W) : (prevX > W / 2 ? W : 0);\n ctx.lineTo(edgeX, y);\n ctx.closePath(); ctx.fill();\n ctx.beginPath();\n ctx.moveTo(isSunrise ? 0 : W, y);\n }\n ctx.lineTo(x, y);\n prevX = x;\n }\n ctx.closePath();\n ctx.fill();\n };\n\n // Draw each twilight zone as a progressively darker overlay\n let prevTerminator: Array<{ lat: number; lon: number }> | null = null;\n for (const zone of twilightZones) {\n const dep = Math.min(zone.depression, 89);\n const terminator = calculateTerminator(now, dep);\n ctx.fillStyle = `rgba(0, 6, 24, ${zone.alpha})`;\n\n if (prevTerminator) {\n // Band between previous boundary and this one — fill the delta\n fillNightSide(terminator, false);\n fillNightSide(terminator, true);\n } else {\n // First zone (civil): fill from geometric terminator outward\n fillNightSide(terminator, false);\n fillNightSide(terminator, true);\n }\n prevTerminator = terminator;\n }\n }\n\n // === World Map (land visible against ocean) ===\n ctx.fillStyle = COLORS.land;\n ctx.strokeStyle = 'rgba(100, 120, 160, 0.25)';\n ctx.lineWidth = 1;\n\n for (const pathStr of WORLD_MAP_PATHS) {\n const commands = pathStr.split(' ');\n ctx.beginPath();\n for (const cmd of commands) {\n if (cmd.startsWith('M') || cmd.startsWith('L')) {\n const coords = cmd.substring(1).split(',');\n const lon = parseFloat(coords[0]);\n const lat = parseFloat(coords[1]);\n const x = lonToX(lon, W);\n const y = latToY(lat, H);\n if (cmd.startsWith('M')) ctx.moveTo(x, y);\n else ctx.lineTo(x, y);\n } else if (cmd === 'Z') {\n ctx.closePath();\n }\n }\n ctx.fill();\n ctx.stroke();\n }\n\n // === Grid ===\n if (showGrid) {\n ctx.strokeStyle = COLORS.grid;\n ctx.lineWidth = 0.4;\n for (let lon = -180; lon <= 180; lon += 30) {\n const x = lonToX(lon, W);\n ctx.beginPath(); ctx.moveTo(x, 0); ctx.lineTo(x, H); ctx.stroke();\n }\n for (let lat = -90; lat <= 90; lat += 30) {\n const y = latToY(lat, H);\n ctx.beginPath(); ctx.moveTo(0, y); ctx.lineTo(W, y); ctx.stroke();\n }\n }\n\n // === Equator ===\n if (showEquator) {\n const eqY = latToY(0, H);\n ctx.strokeStyle = COLORS.equator;\n ctx.lineWidth = 1;\n ctx.setLineDash([6, 4]);\n ctx.beginPath(); ctx.moveTo(0, eqY); ctx.lineTo(W, eqY); ctx.stroke();\n ctx.setLineDash([]);\n }\n\n // === Ground Station Coverage Circles (latitude-corrected ellipses) ===\n for (const gs of groundStations) {\n const enhanced = gs as GroundStationEnhanced;\n if (enhanced.showCoverage && enhanced.coverageRadius) {\n const statusColor = STATUS_COLOR_MAP[enhanced.status || 'standby'] || STATUS_COLOR_MAP.standby;\n\n // Use the coverage radius as-is (in degrees) — caller can compute\n // from visibilityRadiusDeg() or provide a direct value\n drawCoverageEllipse(ctx, gs.latitude, gs.longitude, enhanced.coverageRadius, W, H, lonToX, latToY);\n ctx.fillStyle = `${statusColor}15`;\n ctx.fill();\n ctx.strokeStyle = `${statusColor}40`;\n ctx.lineWidth = 1;\n ctx.setLineDash([4, 3]);\n ctx.stroke();\n ctx.setLineDash([]);\n }\n }\n\n // === Ground Stations (Astro UX Icons + Status Badges) ===\n for (const gs of groundStations) {\n const enhanced = gs as GroundStationEnhanced;\n const x = lonToX(gs.longitude, W);\n const y = latToY(gs.latitude, H);\n const status = enhanced.status || 'standby';\n const statusColor = STATUS_COLOR_MAP[status] || STATUS_COLOR_MAP.standby;\n const stationType = enhanced.type || 'dish';\n const iconSize = 20;\n const halfIcon = iconSize / 2;\n\n ctx.save();\n ctx.translate(x, y);\n\n // -- Background circle (subtle glow) --\n ctx.beginPath();\n ctx.arc(0, 0, halfIcon + 2, 0, Math.PI * 2);\n ctx.fillStyle = `${statusColor}18`;\n ctx.fill();\n\n // -- Icon background pill --\n ctx.beginPath();\n ctx.arc(0, 0, halfIcon, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.85)';\n ctx.fill();\n ctx.strokeStyle = `${statusColor}66`;\n ctx.lineWidth = 1.5;\n ctx.stroke();\n\n // -- Draw Astro UX icon via SVG paths scaled to icon size --\n ctx.save();\n const iconScale = iconSize / 24;\n ctx.translate(-halfIcon, -halfIcon);\n ctx.scale(iconScale, iconScale);\n\n if (stationType === 'dish' || stationType === 'omni') {\n // Antenna icon (concentric arcs — Astro antenna pattern)\n ctx.strokeStyle = statusColor;\n ctx.fillStyle = statusColor;\n ctx.lineWidth = 2 / iconScale;\n // Outer arc\n ctx.beginPath(); ctx.arc(12, 12, 10, Math.PI * 1.25, Math.PI * 1.75); ctx.stroke();\n // Middle arc\n ctx.beginPath(); ctx.arc(12, 12, 6.5, Math.PI * 1.2, Math.PI * 1.8); ctx.stroke();\n // Inner arc\n ctx.beginPath(); ctx.arc(12, 12, 3, Math.PI * 1.15, Math.PI * 1.85); ctx.stroke();\n // Center dot (feed point)\n ctx.beginPath(); ctx.arc(12, 12, 1.5, 0, Math.PI * 2); ctx.fill();\n // Stand\n ctx.lineWidth = 1.5 / iconScale;\n ctx.beginPath(); ctx.moveTo(12, 14); ctx.lineTo(12, 20); ctx.stroke();\n ctx.beginPath(); ctx.moveTo(8, 20); ctx.lineTo(16, 20); ctx.stroke();\n } else if (stationType === 'phased-array') {\n // Phased array icon (grid pattern)\n ctx.strokeStyle = statusColor;\n ctx.fillStyle = statusColor;\n ctx.lineWidth = 1.5 / iconScale;\n ctx.strokeRect(5, 4, 14, 12);\n // Grid lines\n for (let gx = 8; gx <= 16; gx += 4) {\n ctx.beginPath(); ctx.moveTo(gx, 4); ctx.lineTo(gx, 16); ctx.stroke();\n }\n for (let gy = 8; gy <= 12; gy += 4) {\n ctx.beginPath(); ctx.moveTo(5, gy); ctx.lineTo(19, gy); ctx.stroke();\n }\n // Stand\n ctx.beginPath(); ctx.moveTo(12, 16); ctx.lineTo(12, 21); ctx.stroke();\n ctx.beginPath(); ctx.moveTo(8, 21); ctx.lineTo(16, 21); ctx.stroke();\n } else {\n // Relay: satellite dish with signal waves\n ctx.strokeStyle = statusColor;\n ctx.fillStyle = statusColor;\n ctx.lineWidth = 2 / iconScale;\n // Dish bowl\n ctx.beginPath(); ctx.arc(10, 12, 7, Math.PI * 0.8, Math.PI * 1.7); ctx.stroke();\n // Feed arm\n ctx.lineWidth = 1.5 / iconScale;\n ctx.beginPath(); ctx.moveTo(10, 12); ctx.lineTo(16, 6); ctx.stroke();\n // Signal waves\n ctx.lineWidth = 1.2 / iconScale;\n ctx.beginPath(); ctx.arc(17, 5, 2.5, Math.PI * 1.2, Math.PI * 1.7); ctx.stroke();\n ctx.beginPath(); ctx.arc(17, 5, 4.5, Math.PI * 1.15, Math.PI * 1.75); ctx.stroke();\n // Base\n ctx.lineWidth = 1.5 / iconScale;\n ctx.beginPath(); ctx.moveTo(10, 14); ctx.lineTo(10, 20); ctx.stroke();\n ctx.beginPath(); ctx.moveTo(6, 20); ctx.lineTo(14, 20); ctx.stroke();\n }\n\n ctx.restore(); // icon scaling\n\n // -- Astro UX Status Badge (top-left of icon) --\n const badgeX = -halfIcon + 1;\n const badgeY = -halfIcon + 1;\n const badgeSize = 7;\n\n // Badge background ring\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 2 + 2, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.9)';\n ctx.fill();\n\n // Status shape per Astro UX spec (StatusIndicator.tsx canonical reference):\n // off → small filled dot, standby → ring (hollow circle),\n // normal → filled circle, caution → square,\n // serious → diamond, critical → triangle pointing DOWN\n ctx.fillStyle = statusColor;\n ctx.strokeStyle = statusColor;\n ctx.lineWidth = 1;\n if (status === 'normal') {\n // Filled circle for normal\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 2, 0, Math.PI * 2);\n ctx.fill();\n } else if (status === 'standby') {\n // Ring (hollow circle) for standby\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 2 - 0.5, 0, Math.PI * 2);\n ctx.lineWidth = 2;\n ctx.stroke();\n } else if (status === 'caution') {\n // Square for caution\n const hs = badgeSize / 2;\n ctx.fillRect(badgeX - hs, badgeY - hs, badgeSize, badgeSize);\n } else if (status === 'serious') {\n // Diamond for serious\n const hs = badgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(badgeX, badgeY - hs);\n ctx.lineTo(badgeX + hs, badgeY);\n ctx.lineTo(badgeX, badgeY + hs);\n ctx.lineTo(badgeX - hs, badgeY);\n ctx.closePath();\n ctx.fill();\n } else if (status === 'critical') {\n // Triangle pointing DOWN for critical\n const hs = badgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(badgeX, badgeY + hs); // bottom point\n ctx.lineTo(badgeX + hs, badgeY - hs); // top-right\n ctx.lineTo(badgeX - hs, badgeY - hs); // top-left\n ctx.closePath();\n ctx.fill();\n } else {\n // Off: small filled dot\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 3, 0, Math.PI * 2);\n ctx.fill();\n }\n\n ctx.restore();\n\n // Label\n ctx.fillStyle = COLORS.text;\n ctx.font = '9px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(gs.name.length > 14 ? gs.name.slice(0, 14) + '…' : gs.name, x, y + halfIcon + 12);\n }\n\n // === Satellite Tracks ===\n allSatellites.forEach((sat, satIdx) => {\n const track = sat.groundTrack;\n if (!track || track.length === 0) return;\n\n const trackColor = sat.color || DEFAULT_TRACK_COLORS[satIdx % DEFAULT_TRACK_COLORS.length];\n const futureIdx = sat.futureTrackIndex ?? track.length;\n\n // Draw past track (solid)\n ctx.strokeStyle = trackColor;\n ctx.lineWidth = 2;\n ctx.lineCap = 'round';\n ctx.lineJoin = 'round';\n ctx.setLineDash([]);\n ctx.beginPath();\n\n let prevX = lonToX(track[0].longitude, W);\n let prevY = latToY(track[0].latitude, H);\n ctx.moveTo(prevX, prevY);\n\n for (let i = 1; i < Math.min(futureIdx, track.length); i++) {\n const x = lonToX(track[i].longitude, W);\n const y = latToY(track[i].latitude, H);\n const crossesDateLine = Math.abs(x - prevX) > W / 2;\n if (crossesDateLine) {\n ctx.stroke(); ctx.beginPath(); ctx.moveTo(x, y);\n } else {\n ctx.lineTo(x, y);\n }\n prevX = x; prevY = y;\n }\n ctx.stroke();\n\n // Draw future track (dashed)\n if (futureIdx < track.length) {\n ctx.strokeStyle = `${trackColor}88`;\n ctx.lineWidth = 1.5;\n ctx.setLineDash([6, 4]);\n ctx.beginPath();\n ctx.moveTo(lonToX(track[futureIdx].longitude, W), latToY(track[futureIdx].latitude, H));\n let fpx = lonToX(track[futureIdx].longitude, W);\n\n for (let i = futureIdx + 1; i < track.length; i++) {\n const x = lonToX(track[i].longitude, W);\n const y = latToY(track[i].latitude, H);\n if (Math.abs(x - fpx) > W / 2) {\n ctx.stroke(); ctx.beginPath(); ctx.moveTo(x, y);\n } else {\n ctx.lineTo(x, y);\n }\n fpx = x;\n }\n ctx.stroke();\n ctx.setLineDash([]);\n }\n\n // Draw access segments (contact passes)\n if (sat.accessMask && sat.accessMask.some(Boolean)) {\n ctx.strokeStyle = STATUS_COLOR_MAP.normal;\n ctx.lineWidth = 3;\n ctx.setLineDash([]);\n ctx.beginPath();\n let drawing = false;\n let apx = lonToX(track[0].longitude, W);\n\n for (let i = 0; i < track.length; i++) {\n const x = lonToX(track[i].longitude, W);\n const y = latToY(track[i].latitude, H);\n const active = sat.accessMask[i];\n const crossesDateLine = i > 0 && Math.abs(x - apx) > W / 2;\n\n if (active && !crossesDateLine) {\n if (!drawing) { ctx.moveTo(x, y); drawing = true; }\n else ctx.lineTo(x, y);\n } else {\n if (drawing) { ctx.stroke(); ctx.beginPath(); drawing = false; }\n }\n apx = x;\n }\n if (drawing) ctx.stroke();\n }\n\n // AOS/LOS markers\n if (sat.passMarkers) {\n for (const marker of sat.passMarkers) {\n const mx = lonToX(marker.longitude, W);\n const my = latToY(marker.latitude, H);\n const markerColor = marker.type === 'aos' ? STATUS_COLOR_MAP.normal : STATUS_COLOR_MAP.critical;\n\n ctx.save();\n ctx.translate(mx, my);\n // Triangle marker\n ctx.beginPath();\n if (marker.type === 'aos') {\n ctx.moveTo(0, -6); ctx.lineTo(4, 2); ctx.lineTo(-4, 2);\n } else {\n ctx.moveTo(0, 6); ctx.lineTo(4, -2); ctx.lineTo(-4, -2);\n }\n ctx.closePath();\n ctx.fillStyle = markerColor;\n ctx.fill();\n ctx.restore();\n\n if (marker.label) {\n ctx.fillStyle = markerColor;\n ctx.font = '8px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(marker.label, mx, marker.type === 'aos' ? my - 9 : my + 13);\n }\n }\n }\n\n // Satellite footprint (latitude-corrected ellipse)\n if (sat.showFootprint && sat.footprintRadius) {\n const lastPt = futureIdx > 0 ? track[Math.min(futureIdx - 1, track.length - 1)] : track[track.length - 1];\n\n drawCoverageEllipse(ctx, lastPt.latitude, lastPt.longitude, sat.footprintRadius, W, H, lonToX, latToY);\n ctx.fillStyle = `${trackColor}12`;\n ctx.fill();\n ctx.strokeStyle = `${trackColor}50`;\n ctx.lineWidth = 1;\n ctx.setLineDash([3, 3]);\n ctx.stroke();\n ctx.setLineDash([]);\n }\n\n // Current position satellite icon (Astro UX style)\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, track.length - 1) : track.length - 1;\n const current = track[currentIdx];\n const cx = lonToX(current.longitude, W);\n const cy = latToY(current.latitude, H);\n const satStatus = sat.status || 'normal';\n const statusColor = STATUS_COLOR_MAP[satStatus] || STATUS_COLOR_MAP.normal;\n const satIconSize = 22;\n const satHalf = satIconSize / 2;\n\n // Outer pulsing glow\n ctx.beginPath();\n ctx.arc(cx, cy, satHalf + 6, 0, Math.PI * 2);\n ctx.fillStyle = `${statusColor}12`;\n ctx.fill();\n\n ctx.beginPath();\n ctx.arc(cx, cy, satHalf + 3, 0, Math.PI * 2);\n ctx.fillStyle = `${statusColor}20`;\n ctx.fill();\n\n // Icon background\n ctx.save();\n ctx.translate(cx, cy);\n\n ctx.beginPath();\n ctx.arc(0, 0, satHalf, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.88)';\n ctx.fill();\n ctx.strokeStyle = `${statusColor}88`;\n ctx.lineWidth = 1.5;\n ctx.stroke();\n\n // Draw satellite shape (body + solar panels)\n ctx.fillStyle = statusColor;\n // Body (rounded rect approximation)\n const bw = 5, bh = 3.5;\n ctx.beginPath();\n ctx.moveTo(-bw / 2 + 0.5, -bh / 2);\n ctx.lineTo(bw / 2 - 0.5, -bh / 2);\n ctx.arcTo(bw / 2, -bh / 2, bw / 2, -bh / 2 + 0.5, 0.5);\n ctx.lineTo(bw / 2, bh / 2 - 0.5);\n ctx.arcTo(bw / 2, bh / 2, bw / 2 - 0.5, bh / 2, 0.5);\n ctx.lineTo(-bw / 2 + 0.5, bh / 2);\n ctx.arcTo(-bw / 2, bh / 2, -bw / 2, bh / 2 - 0.5, 0.5);\n ctx.lineTo(-bw / 2, -bh / 2 + 0.5);\n ctx.arcTo(-bw / 2, -bh / 2, -bw / 2 + 0.5, -bh / 2, 0.5);\n ctx.closePath();\n ctx.fill();\n\n // Solar panel left\n ctx.fillStyle = `${statusColor}BB`;\n ctx.fillRect(-satHalf + 2, -2, 5, 4);\n // Panel lines\n ctx.strokeStyle = `${statusColor}55`;\n ctx.lineWidth = 0.5;\n ctx.beginPath();\n ctx.moveTo(-satHalf + 4.5, -2); ctx.lineTo(-satHalf + 4.5, 2);\n ctx.stroke();\n\n // Solar panel right\n ctx.fillStyle = `${statusColor}BB`;\n ctx.fillRect(satHalf - 7, -2, 5, 4);\n ctx.strokeStyle = `${statusColor}55`;\n ctx.beginPath();\n ctx.moveTo(satHalf - 4.5, -2); ctx.lineTo(satHalf - 4.5, 2);\n ctx.stroke();\n\n // Panel connectors\n ctx.strokeStyle = statusColor;\n ctx.lineWidth = 1;\n ctx.beginPath();\n ctx.moveTo(-bw / 2, 0); ctx.lineTo(-satHalf + 7, 0);\n ctx.moveTo(bw / 2, 0); ctx.lineTo(satHalf - 7, 0);\n ctx.stroke();\n\n // -- Astro UX Status Badge (top-left of satellite icon) --\n // Shapes per canonical StatusIndicator.tsx:\n // off → small dot, standby → ring, normal → filled circle,\n // caution → square, serious → diamond, critical → triangle DOWN\n const satBadgeX = -satHalf + 2;\n const satBadgeY = -satHalf + 2;\n const satBadgeSize = 7;\n\n // Badge background\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 2 + 2, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.9)';\n ctx.fill();\n\n ctx.fillStyle = statusColor;\n ctx.strokeStyle = statusColor;\n ctx.lineWidth = 1;\n if (satStatus === 'normal') {\n // Filled circle\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 2, 0, Math.PI * 2);\n ctx.fill();\n } else if (satStatus === 'standby') {\n // Ring (hollow circle)\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 2 - 0.5, 0, Math.PI * 2);\n ctx.lineWidth = 2;\n ctx.stroke();\n } else if (satStatus === 'caution') {\n const hs = satBadgeSize / 2;\n ctx.fillRect(satBadgeX - hs, satBadgeY - hs, satBadgeSize, satBadgeSize);\n } else if (satStatus === 'serious') {\n // Diamond\n const hs = satBadgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(satBadgeX, satBadgeY - hs);\n ctx.lineTo(satBadgeX + hs, satBadgeY);\n ctx.lineTo(satBadgeX, satBadgeY + hs);\n ctx.lineTo(satBadgeX - hs, satBadgeY);\n ctx.closePath();\n ctx.fill();\n } else if (satStatus === 'critical') {\n // Triangle pointing DOWN\n const hs = satBadgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(satBadgeX, satBadgeY + hs); // bottom point\n ctx.lineTo(satBadgeX + hs, satBadgeY - hs); // top-right\n ctx.lineTo(satBadgeX - hs, satBadgeY - hs); // top-left\n ctx.closePath();\n ctx.fill();\n } else {\n // Off: small filled dot\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 3, 0, Math.PI * 2);\n ctx.fill();\n }\n\n ctx.restore();\n\n // Satellite label\n ctx.fillStyle = '#fff';\n ctx.font = 'bold 10px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(sat.name, cx, cy - satHalf - 6);\n });\n\n // === Legend ===\n if (showLegend && (allSatellites.length > 0 || groundStations.length > 0)) {\n const legendX = 10;\n const legendY = 10;\n const lineHeight = 14;\n const items: Array<{ color: string; label: string; dash?: boolean }> = [];\n\n allSatellites.forEach((sat, idx) => {\n const c = sat.color || DEFAULT_TRACK_COLORS[idx % DEFAULT_TRACK_COLORS.length];\n items.push({ color: c, label: sat.name });\n });\n if (groundStations.length > 0) {\n items.push({ color: STATUS_COLOR_MAP.standby, label: `${groundStations.length} Ground Station${groundStations.length > 1 ? 's' : ''}` });\n }\n\n const maxLabelWidth = Math.max(...items.map(item => ctx.measureText(item.label).width)) + 28;\n const legendH = items.length * lineHeight + 10;\n\n ctx.fillStyle = 'rgba(0, 0, 0, 0.6)';\n ctx.strokeStyle = 'rgba(100, 116, 139, 0.3)';\n ctx.lineWidth = 1;\n const r = 4;\n ctx.beginPath();\n ctx.moveTo(legendX + r, legendY);\n ctx.lineTo(legendX + maxLabelWidth - r, legendY);\n ctx.arcTo(legendX + maxLabelWidth, legendY, legendX + maxLabelWidth, legendY + r, r);\n ctx.lineTo(legendX + maxLabelWidth, legendY + legendH - r);\n ctx.arcTo(legendX + maxLabelWidth, legendY + legendH, legendX + maxLabelWidth - r, legendY + legendH, r);\n ctx.lineTo(legendX + r, legendY + legendH);\n ctx.arcTo(legendX, legendY + legendH, legendX, legendY + legendH - r, r);\n ctx.lineTo(legendX, legendY + r);\n ctx.arcTo(legendX, legendY, legendX + r, legendY, r);\n ctx.closePath();\n ctx.fill();\n ctx.stroke();\n\n items.forEach((item, i) => {\n const iy = legendY + 8 + i * lineHeight;\n // Color swatch\n ctx.fillStyle = item.color;\n ctx.beginPath();\n ctx.arc(legendX + 10, iy + 3, 4, 0, Math.PI * 2);\n ctx.fill();\n // Label\n ctx.fillStyle = 'rgba(255, 255, 255, 0.85)';\n ctx.font = '10px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'left';\n ctx.fillText(item.label, legendX + 20, iy + 7);\n });\n }\n\n // === Pins (read-only on canvas backend) ===\n if (pins && pins.length > 0) {\n pins.forEach((pin) => {\n const px = lonToX(pin.longitude, W);\n const py = latToY(pin.latitude, H);\n const pinColor = pin.color || tokens.colors.accent.primary;\n\n // Drop shadow\n ctx.save();\n ctx.shadowColor = 'rgba(0,0,0,0.4)';\n ctx.shadowBlur = 6;\n ctx.shadowOffsetY = 2;\n\n // Pin body: inverted teardrop shape\n ctx.beginPath();\n ctx.arc(px, py - 10, 6, Math.PI, 0);\n ctx.lineTo(px, py);\n ctx.closePath();\n ctx.fillStyle = pinColor;\n ctx.fill();\n ctx.restore();\n\n // Inner dot\n ctx.beginPath();\n ctx.arc(px, py - 10, 2.5, 0, Math.PI * 2);\n ctx.fillStyle = '#0d1323';\n ctx.fill();\n\n // Label\n if (pin.label) {\n ctx.fillStyle = 'rgba(255,255,255,0.9)';\n ctx.font = '10px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(pin.label, px, py - 20);\n }\n });\n }\n\n // === No Data Message ===\n if (allSatellites.length === 0 && groundStations.length === 0 && (!pins || pins.length === 0)) {\n ctx.fillStyle = COLORS.text;\n ctx.font = '12px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(emptyMessage, W / 2, H / 2);\n }\n }, [allSatellites, groundStations, height, showTerminator, terminatorTime, showGrid, showLegend, showEquator, lonToX, latToY, COLORS, emptyMessage, pins, tokens.colors.accent.primary]);\n\n // Tooltip on mouse move\n const handleMouseMove = useCallback((e: React.MouseEvent<HTMLCanvasElement>) => {\n const canvas = canvasRef.current;\n if (!canvas) return;\n const rect = canvas.getBoundingClientRect();\n const mx = e.clientX - rect.left;\n const my = e.clientY - rect.top;\n const W = rect.width;\n const H = typeof height === 'number' ? height : rect.height;\n\n // Check satellite positions\n for (const sat of allSatellites) {\n if (!sat.groundTrack.length) continue;\n const futureIdx = sat.futureTrackIndex ?? sat.groundTrack.length;\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, sat.groundTrack.length - 1) : sat.groundTrack.length - 1;\n const lastPt = sat.groundTrack[currentIdx];\n const sx = lonToX(lastPt.longitude, W);\n const sy = latToY(lastPt.latitude, H);\n const dist = Math.sqrt((mx - sx) ** 2 + (my - sy) ** 2);\n if (dist < 18) {\n const satStatus = sat.status || 'normal';\n setTooltip({\n x: e.clientX - rect.left + 15,\n y: e.clientY - rect.top - 10,\n type: 'satellite',\n name: sat.name,\n status: satStatus,\n statusColor: STATUS_COLOR_MAP[satStatus] || STATUS_COLOR_MAP.normal,\n lat: lastPt.latitude,\n lon: lastPt.longitude,\n alt: lastPt.altitude,\n trackColor: sat.color || DEFAULT_TRACK_COLORS[allSatellites.indexOf(sat) % DEFAULT_TRACK_COLORS.length],\n hasFootprint: sat.showFootprint,\n futurePoints: sat.groundTrack.length - futureIdx,\n pastPoints: futureIdx,\n });\n return;\n }\n }\n\n // Check ground stations\n for (const gs of groundStations) {\n const gx = lonToX(gs.longitude, W);\n const gy = latToY(gs.latitude, H);\n const dist = Math.sqrt((mx - gx) ** 2 + (my - gy) ** 2);\n if (dist < 16) {\n const enhanced = gs as GroundStationEnhanced;\n const gsStatus = enhanced.status || 'standby';\n setTooltip({\n x: e.clientX - rect.left + 15,\n y: e.clientY - rect.top - 10,\n type: 'station',\n name: gs.name,\n status: gsStatus,\n statusColor: STATUS_COLOR_MAP[gsStatus] || STATUS_COLOR_MAP.standby,\n lat: gs.latitude,\n lon: gs.longitude,\n stationType: enhanced.type || 'dish',\n network: ('network' in gs && gs.network) ? String(gs.network) : undefined,\n coverageRadius: enhanced.coverageRadius,\n });\n return;\n }\n }\n\n setTooltip(null);\n }, [allSatellites, groundStations, height, lonToX, latToY]);\n\n const handleClick = useCallback((e: React.MouseEvent<HTMLCanvasElement>) => {\n const canvas = canvasRef.current;\n if (!canvas) return;\n const rect = canvas.getBoundingClientRect();\n const mx = e.clientX - rect.left;\n const my = e.clientY - rect.top;\n const W = rect.width;\n const H = typeof height === 'number' ? height : rect.height;\n\n for (const sat of allSatellites) {\n if (!sat.groundTrack.length) continue;\n const futureIdx = sat.futureTrackIndex ?? sat.groundTrack.length;\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, sat.groundTrack.length - 1) : sat.groundTrack.length - 1;\n const currentPt = sat.groundTrack[currentIdx];\n const sx = lonToX(currentPt.longitude, W);\n const sy = latToY(currentPt.latitude, H);\n if (Math.sqrt((mx - sx) ** 2 + (my - sy) ** 2) < 15) {\n onSatelliteClick?.(sat.id);\n return;\n }\n }\n\n for (const gs of groundStations) {\n const gx = lonToX(gs.longitude, W);\n const gy = latToY(gs.latitude, H);\n if (Math.sqrt((mx - gx) ** 2 + (my - gy) ** 2) < 12) {\n onStationClick?.('id' in gs ? gs.id : gs.name);\n return;\n }\n }\n }, [allSatellites, groundStations, height, lonToX, latToY, onSatelliteClick, onStationClick]);\n\n useEffect(() => {\n draw();\n window.addEventListener('resize', draw);\n return () => window.removeEventListener('resize', draw);\n }, [draw]);\n\n // Leaflet backend: return after all hooks so hook count is stable\n const resolvedMinHeightForLeaflet = minHeight || (typeof height === 'number' ? `${height}px` : '400px');\n\n if (mapProvider === 'leaflet') {\n if (LeafletMap) {\n return (\n <div\n role=\"img\"\n aria-label={allSatellites.length > 0 || groundStations.length > 0 ? 'Ground track map showing satellite tracks and ground stations' : 'Ground track map'}\n style={{\n display: 'block',\n width: width ?? '100%',\n height: typeof height === 'string' ? height : undefined,\n minHeight: resolvedMinHeightForLeaflet,\n }}\n >\n <LeafletMap\n allSatellites={allSatellites}\n groundStations={groundStations}\n showTerminator={showTerminator}\n terminatorTime={terminatorTime}\n showGrid={showGrid}\n showLegend={showLegend}\n showEquator={showEquator}\n showRecenterButton={showRecenterButton}\n defaultCenter={leafletDefaultCenter}\n defaultZoom={leafletDefaultZoom}\n height={height}\n width={width}\n minHeight={minHeight}\n emptyMessage={emptyMessage}\n tileUrl={tileUrl}\n className={className}\n onSatelliteClick={onSatelliteClick}\n onStationClick={onStationClick}\n pins={pins}\n pinsEditable={pinsEditable}\n onPinAdd={onPinAdd}\n onPinUpdate={onPinUpdate}\n onPinRemove={onPinRemove}\n />\n </div>\n );\n }\n if (!leafletFailed) {\n return (\n <div\n className={`zendir-ground-track-map ${className}`}\n style={{\n width,\n height: typeof height === 'string' ? height : undefined,\n minHeight: resolvedMinHeightForLeaflet,\n backgroundColor: tokens.colors.background.base,\n borderRadius: 8,\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n }}\n >\n <span style={{ color: tokens.colors.text.secondary, fontSize: 14 }}>Loading map…</span>\n </div>\n );\n }\n }\n\n const resolvedMinHeight = minHeight || (typeof height === 'number' ? `${height}px` : '400px');\n\n // Loading state\n if (isLoading) {\n return (\n <div\n className={`zendir-ground-track-map ${className}`}\n style={{\n width,\n minHeight: resolvedMinHeight,\n backgroundColor: COLORS.background,\n borderRadius: '8px',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n }}\n >\n <span style={{ color: COLORS.text, fontSize: '14px', letterSpacing: '1px' }}>Loading…</span>\n </div>\n );\n }\n\n return (\n <div\n ref={containerRef}\n role=\"img\"\n aria-label={allSatellites.length > 0 || groundStations.length > 0 ? 'Ground track map showing satellite tracks and ground stations' : 'Ground track map'}\n className={`zendir-ground-track-map ${className}`}\n style={{ width, height: typeof height === 'string' ? height : undefined, minHeight: resolvedMinHeight, backgroundColor: COLORS.background, borderRadius: '8px', overflow: 'hidden', position: 'relative' }}\n data-map-backend=\"canvas\"\n >\n {/* Hint when Leaflet was requested but failed (e.g. leaflet not installed) */}\n {mapProvider === 'leaflet' && leafletFailed && (\n <div\n style={{\n position: 'absolute',\n bottom: 8,\n left: 8,\n zIndex: 100,\n padding: '4px 8px',\n background: 'rgba(0,0,0,0.75)',\n color: '#fce83a',\n fontSize: 11,\n borderRadius: 4,\n }}\n >\n Static view (install <code style={{ fontFamily: 'monospace' }}>leaflet</code> for zoom/pan)\n </div>\n )}\n {/* Recenter button (SRO compat) */}\n {showRecenterButton && (\n <button\n type=\"button\"\n onClick={handleRecenter}\n title=\"Recenter Map\"\n aria-label=\"Recenter map\"\n style={{\n position: 'absolute',\n top: '8px',\n right: '8px',\n zIndex: 20,\n background: 'rgba(30, 41, 59, 0.85)',\n border: '1px solid rgba(100, 116, 139, 0.4)',\n borderRadius: '4px',\n color: '#fff',\n cursor: 'pointer',\n padding: '4px 8px',\n fontSize: '12px',\n lineHeight: '1',\n display: 'flex',\n alignItems: 'center',\n gap: '4px',\n }}\n >\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth=\"2\" strokeLinecap=\"round\" strokeLinejoin=\"round\">\n <circle cx=\"12\" cy=\"12\" r=\"3\" />\n <path d=\"M12 2v4M12 18v4M2 12h4M18 12h4\" />\n </svg>\n </button>\n )}\n\n <canvas\n ref={canvasRef}\n style={{ display: 'block', cursor: tooltip ? 'pointer' : 'default' }}\n onMouseMove={handleMouseMove}\n onMouseLeave={() => setTooltip(null)}\n onClick={handleClick}\n />\n {tooltip && (\n <div\n style={{\n position: 'absolute',\n left: tooltip.x,\n top: tooltip.y,\n background: 'rgba(15, 23, 42, 0.95)',\n backdropFilter: 'blur(8px)',\n border: `1px solid ${tooltip.statusColor}40`,\n borderRadius: '8px',\n padding: 0,\n pointerEvents: 'none',\n zIndex: 10,\n color: 'rgba(255, 255, 255, 0.9)',\n fontSize: '11px',\n fontFamily: '\"Inter\", system-ui, sans-serif',\n lineHeight: '1.5',\n boxShadow: `0 4px 16px rgba(0, 0, 0, 0.5), 0 0 12px ${tooltip.statusColor}15`,\n minWidth: 180,\n overflow: 'hidden',\n }}\n >\n {/* Header with status accent */}\n <div style={{\n padding: '6px 10px',\n borderBottom: `1px solid ${tooltip.statusColor}25`,\n background: `${tooltip.statusColor}10`,\n display: 'flex', alignItems: 'center', gap: 6,\n }}>\n <span style={{\n width: 8, height: 8, borderRadius: tooltip.status === 'caution' ? 1 : '50%',\n background: tooltip.statusColor, display: 'inline-block', flexShrink: 0,\n boxShadow: `0 0 4px ${tooltip.statusColor}88`,\n }} />\n <span style={{ fontWeight: 600, fontSize: 12 }}>{tooltip.name}</span>\n <span style={{\n marginLeft: 'auto', fontSize: 9, fontWeight: 600, textTransform: 'uppercase',\n color: tooltip.statusColor, letterSpacing: '0.5px',\n }}>\n {tooltip.status}\n </span>\n </div>\n {/* Details */}\n <div style={{ padding: '6px 10px', display: 'flex', flexDirection: 'column', gap: 3 }}>\n {/* Type badge */}\n <div style={{ display: 'flex', alignItems: 'center', gap: 6, marginBottom: 2 }}>\n <span style={{\n fontSize: 9, fontWeight: 600, padding: '1px 5px',\n borderRadius: 3, textTransform: 'uppercase', letterSpacing: '0.3px',\n background: tooltip.type === 'satellite' ? 'rgba(45, 204, 255, 0.15)' : 'rgba(86, 240, 0, 0.15)',\n color: tooltip.type === 'satellite' ? '#2dccff' : '#56f000',\n }}>\n {tooltip.type === 'satellite' ? (\n <>\n <AstroIcon name=\"satellite\" size=\"extra-small\" label=\"Satellite\" style={{ marginRight: 4, verticalAlign: 'middle' }} />\n Satellite\n </>\n ) : (\n <>\n <AstroIcon name=\"antenna\" size=\"extra-small\" label={tooltip.stationType || 'Ground station'} style={{ marginRight: 4, verticalAlign: 'middle' }} />\n {tooltip.stationType || 'dish'}\n </>\n )}\n </span>\n {tooltip.network && (\n <span style={{\n fontSize: 9, padding: '1px 5px', borderRadius: 3,\n background: 'rgba(157, 112, 255, 0.15)', color: '#9D70FF',\n }}>\n {tooltip.network}\n </span>\n )}\n </div>\n {/* Coordinates */}\n <div style={{ display: 'grid', gridTemplateColumns: '52px 1fr', gap: '2px 8px', fontSize: 10 }}>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Lat</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.lat >= 0 ? '+' : ''}{tooltip.lat.toFixed(4)}°\n </span>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Lon</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.lon >= 0 ? '+' : ''}{tooltip.lon.toFixed(4)}°\n </span>\n {tooltip.alt !== undefined && (\n <>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Altitude</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.alt.toFixed(1)} km\n </span>\n </>\n )}\n {tooltip.coverageRadius !== undefined && (\n <>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Coverage</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.coverageRadius.toFixed(1)}° radius\n </span>\n </>\n )}\n </div>\n {/* Satellite extras */}\n {tooltip.type === 'satellite' && (tooltip.pastPoints !== undefined || tooltip.futurePoints !== undefined) && (\n <div style={{ marginTop: 3, paddingTop: 3, borderTop: '1px solid rgba(255,255,255,0.08)', display: 'flex', gap: 10, fontSize: 9 }}>\n {tooltip.pastPoints !== undefined && (\n <span><span style={{ color: tooltip.trackColor }}>━</span> {tooltip.pastPoints} past pts</span>\n )}\n {tooltip.futurePoints !== undefined && tooltip.futurePoints > 0 && (\n <span><span style={{ color: `${tooltip.trackColor}88` }}>╌</span> {tooltip.futurePoints} predicted</span>\n )}\n {tooltip.hasFootprint && <span style={{ color: '#9D70FF' }}>◎ Footprint</span>}\n </div>\n )}\n </div>\n </div>\n )}\n </div>\n );\n}\n\nexport default GroundTrackMap;\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"GroundTrackMap.js","sources":["../../../src/react/charts/GroundTrackMap.tsx"],"sourcesContent":["/**\n * @zendir/ui - GroundTrackMap Component (Enhanced)\n * \n * Full-featured 2D world map showing spacecraft ground tracks, current positions,\n * ground station locations, coverage footprints, satellite icons, and day/night terminator.\n * \n * Inspired by Astro UXDS TT&C Monitor / GRM Dashboard ground map patterns:\n * - Satellite icons with status-colored indicators (not just dots)\n * - Ground station antenna icons with coverage circles\n * - Sensor/antenna footprint cone projections\n * - Future orbit tracks (dashed) vs past tracks (solid)\n * - Day/night terminator\n * - Multi-satellite support\n * - Interactive hover tooltips\n * - AOS/LOS markers on track\n * - Legend with satellite/station counts\n * \n * Astro UX Compliance:\n * - Theme-integrated via useTheme()\n * - Uses official Astro status colors from theme tokens\n * - Uses Astro monitoring icon patterns\n * - 14pt minimum for labels (AstroUXDS guideline)\n * - Consistent with Astro data visualization patterns\n */\n\nimport React, { useRef, useEffect, useCallback, useMemo, useState } from 'react';\nimport { useTheme } from '../theme';\nimport { AstroIcon } from '../core/AstroIcon';\nimport type { GroundTrackPoint, GroundStation } from '../types';\nimport type { GroundTrackMapLeafletProps } from './GroundTrackMapLeaflet';\n\n// =============================================================================\n// Types\n// =============================================================================\n\nexport interface SatelliteTrack {\n /** Unique satellite identifier */\n id: string;\n /** Satellite display name */\n name: string;\n /** Ground track points */\n groundTrack: GroundTrackPoint[];\n /** Track color */\n color?: string;\n /** Satellite status for icon coloring */\n status?: 'normal' | 'caution' | 'serious' | 'critical' | 'standby' | 'off';\n /** Access mask (true = in contact) */\n accessMask?: boolean[];\n /** Show this satellite's sensor footprint */\n showFootprint?: boolean;\n /** Footprint radius in degrees (sensor cone half-angle on ground) */\n footprintRadius?: number;\n /** Show future (predicted) track as dashed line */\n futureTrackIndex?: number;\n /** AOS/LOS markers */\n passMarkers?: Array<{ type: 'aos' | 'los'; latitude: number; longitude: number; label?: string }>;\n}\n\nexport interface GroundStationEnhanced extends GroundStation {\n /** Station status */\n status?: 'normal' | 'caution' | 'serious' | 'critical' | 'standby' | 'off';\n /** Show coverage circle */\n showCoverage?: boolean;\n /** Coverage radius in degrees (visibility horizon) */\n coverageRadius?: number;\n /** Station type (affects icon) */\n type?: 'dish' | 'phased-array' | 'omni' | 'relay';\n}\n\n/** \n * SRO-compatible team path format (space-range-operator CustomMap data shape).\n * Provide this as an alternative to `satellites` for plug-and-play compatibility.\n */\nexport interface TeamPath {\n /** Team/satellite name */\n name: string;\n /** Track color */\n color?: string;\n /** Ordered position array */\n positions: Array<{\n lat: number;\n lng: number;\n time?: number;\n altitude?: number | null;\n }>;\n}\n\n/** SRO-compatible ground station format */\nexport interface SROGroundStation {\n name: string;\n latitude: number;\n longitude: number;\n altitude?: number | null;\n}\n\n/**\n * User-placed point-of-interest pin on the map.\n * Designed for collaborative scenarios where operators share annotated locations\n * across terminals in real time (e.g. targets, waypoints, hazard markers).\n */\nexport interface MapPin {\n /** Unique identifier (used as key and for update/remove callbacks) */\n id: string;\n /** Latitude in degrees (−90 … 90) */\n latitude: number;\n /** Longitude in degrees (−180 … 180) */\n longitude: number;\n /** Short label displayed next to the pin (e.g. \"Target Alpha\") */\n label?: string;\n /** Pin color — any CSS color string. Defaults to accent primary. */\n color?: string;\n /** Optional description shown in tooltip on hover */\n description?: string;\n /** Who placed the pin (display-only; not enforced) */\n createdBy?: string;\n}\n\nexport interface GroundTrackMapProps {\n /** Single satellite ground track (legacy API) */\n groundTrack?: GroundTrackPoint[];\n /** Multiple satellite tracks */\n satellites?: SatelliteTrack[];\n /** Ground stations to display */\n groundStations?: (GroundStation | GroundStationEnhanced | SROGroundStation)[];\n /** Access mask for single-satellite mode (legacy API) */\n accessMask?: boolean[];\n\n // --- SRO-compatible alternate API (space-range-operator plug-and-play) ---\n /** Team path data (SRO CustomMap format) — auto-converts to satellites internally */\n teamPaths?: TeamPath[];\n\n // --- Display options ---\n /** Show day/night terminator */\n showTerminator?: boolean;\n /** Override the time used for the day/night terminator calculation.\n * Defaults to wall-clock `new Date()` when omitted.\n * Pass a simulation/mission UTC to keep the terminator in sync with a sim clock. */\n terminatorTime?: Date;\n /** Show grid lines */\n showGrid?: boolean;\n /** Show legend */\n showLegend?: boolean;\n /** Show equator highlight */\n showEquator?: boolean;\n /** Show recenter button (SRO compat) */\n showRecenterButton?: boolean;\n\n // --- State ---\n /** Whether data is still loading (SRO compat) */\n isLoading?: boolean;\n /** Message when there is no data (SRO compat) */\n emptyMessage?: string;\n\n // --- Sizing ---\n /** Chart width (default: 100%) */\n width?: number | string;\n /** Chart height — number for fixed px, or CSS string like '100%' to fill a flex parent */\n height?: number | string;\n /** Minimum height CSS string (SRO compat, e.g. \"400px\") */\n minHeight?: string;\n\n // --- Centering ---\n /** Default center [lat, lon] (SRO compat) */\n defaultCenter?: [number, number];\n /** Default zoom level 1-10 (SRO compat, affects initial scale) */\n defaultZoom?: number;\n\n /** Custom className */\n className?: string;\n /** Click handler for satellites */\n onSatelliteClick?: (satelliteId: string) => void;\n /** Click handler for ground stations */\n onStationClick?: (stationId: string) => void;\n\n /** Map backend: 'leaflet' (default, requires leaflet peer) or 'canvas' */\n mapProvider?: 'leaflet' | 'canvas';\n /** Tile URL for Leaflet (default: CartoDB dark). Ignored when mapProvider is 'canvas'. */\n tileUrl?: string;\n\n // --- Collaborative Pins (Points of Interest) ---\n /** Array of user-placed pins displayed on the map */\n pins?: MapPin[];\n /**\n * Allow operators to place new pins by clicking the map.\n * When true, a single-click on an empty area opens a pin creation flow.\n * Requires `onPinAdd` to be set. Default false.\n */\n pinsEditable?: boolean;\n /** Called when the user places a new pin (provides lat/lon; consumer assigns id and persists) */\n onPinAdd?: (pin: Omit<MapPin, 'id'>) => void;\n /** Called when the user edits an existing pin (label, color, description change) */\n onPinUpdate?: (pin: MapPin) => void;\n /** Called when the user removes a pin */\n onPinRemove?: (pinId: string) => void;\n}\n\n// =============================================================================\n// Constants — aligned with official Astro UXDS status colors\n// These MUST match the STATUS_COLORS in unified/theme.ts and ThemeProvider\n// =============================================================================\n\nconst STATUS_COLOR_MAP: Record<string, string> = {\n normal: '#56f000', // Green — Astro UXDS \"Normal\"\n standby: '#2dccff', // Cyan — Astro UXDS \"Standby\"\n caution: '#fce83a', // Yellow — Astro UXDS \"Caution\"\n serious: '#ffb302', // Orange — Astro UXDS \"Serious\"\n critical: '#ff3838', // Red — Astro UXDS \"Critical\"\n off: '#a4abb6', // Grey — Astro UXDS \"Off\"\n};\n\nconst DEFAULT_TRACK_COLORS = [\n '#2dccff', '#3E3CFF', '#9D70FF', '#56f000', '#fce83a', '#ff7849', '#2dd4bf', '#ff3838',\n];\n\n// Earth constants for realistic coverage calculations\nconst EARTH_RADIUS_KM = 6371;\nconst _DEG_PER_KM = 1 / 111.32; // Approximate degrees per km at equator\n\n// Higher-quality world map coastline paths (Mercator-compatible)\nconst WORLD_MAP_PATHS = [\n // North America\n 'M-168,70 L-168,60 L-145,63 L-140,58 L-130,55 L-125,49 L-122,48 L-117,33 L-115,28 L-105,22 L-97,18 L-88,20 L-82,25 L-82,30 L-80,32 L-67,47 L-60,46 L-55,52 L-56,60 L-65,70 L-85,73 L-100,72 L-130,70 L-168,70',\n // South America\n 'M-82,12 L-77,8 L-73,11 L-65,5 L-60,5 L-50,0 L-45,-5 L-43,-10 L-39,-15 L-38,-22 L-40,-25 L-43,-30 L-48,-33 L-50,-40 L-55,-48 L-57,-52 L-68,-55 L-73,-50 L-78,-42 L-80,-20 L-78,-5 L-82,12',\n // Europe + Africa\n 'M-10,72 L-5,60 L-5,48 L0,44 L3,44 L5,48 L8,44 L3,37 L-5,35 L-17,28 L-17,15 L-8,10 L-4,5 L5,5 L10,2 L15,-3 L22,-12 L28,-22 L33,-30 L30,-35 L22,-35 L18,-27 L12,-20 L8,-10 L3,0 L-5,10 L-12,15 L-15,20 L-13,32 L-10,38 L0,48 L8,55 L15,58 L20,60 L25,65 L30,70 L-10,72',\n // Asia \n 'M30,70 L40,65 L50,60 L60,55 L70,50 L80,45 L90,45 L100,40 L105,35 L110,30 L115,25 L120,30 L125,35 L130,40 L135,45 L140,50 L145,55 L150,60 L155,65 L160,68 L170,70 L180,70 L180,72 L30,72 L30,70',\n // Australia\n 'M112,-12 L118,-14 L125,-15 L132,-13 L138,-15 L143,-17 L148,-20 L152,-23 L153,-28 L150,-33 L147,-37 L140,-38 L134,-36 L130,-33 L126,-30 L120,-30 L116,-28 L114,-24 L113,-18 L112,-12',\n // Antarctica\n 'M-180,-72 L180,-72 L180,-90 L-180,-90 Z',\n];\n\n// =============================================================================\n// Helper Functions\n// =============================================================================\n\n/**\n * Calculate ground station visibility radius in degrees.\n * Uses spherical Earth geometry: for a station with minimum elevation angle `elDeg`\n * and a satellite at altitude `altKm`, the half-angle subtended at Earth center is:\n * ρ = acos(R / (R + h)) − el_rad\n * Then the ground-range in degrees ≈ ρ * (180/π).\n * If no altitude context, coverageRadius prop is used directly (in degrees).\n */\nfunction _visibilityRadiusDeg(altitudeKm: number = 500, minElevationDeg: number = 5): number {\n const elRad = (minElevationDeg * Math.PI) / 180;\n const rho = Math.acos(EARTH_RADIUS_KM / (EARTH_RADIUS_KM + altitudeKm)) - elRad;\n return Math.max(0, (rho * 180) / Math.PI);\n}\n\n/**\n * Draw a ground-circle footprint using great-circle math.\n * Given a center lat/lon and an angular radius (in degrees measured along\n * Earth's surface), compute N points around the circle on the sphere, then\n * project each to equirectangular pixel coordinates.\n *\n * This produces a circle that is geographically correct AND looks natural\n * on the map — at high latitudes the circle is wider in longitude because\n * that's how equirectangular projection works, but it's capped by reality\n * rather than a simple 1/cos blowup.\n */\nfunction drawCoverageEllipse(\n ctx: CanvasRenderingContext2D,\n centerLat: number,\n centerLon: number,\n radiusDeg: number,\n W: number,\n H: number,\n lonToX: (lon: number, w: number) => number,\n latToY: (lat: number, h: number) => number,\n numSegments: number = 72,\n): void {\n const lat0 = (centerLat * Math.PI) / 180;\n const lon0 = (centerLon * Math.PI) / 180;\n const d = (radiusDeg * Math.PI) / 180; // angular radius in radians\n\n // Compute the center pixel once — all points are placed relative to this\n // so the circle never jumps across the date-line seam.\n const cx = lonToX(centerLon, W);\n const cy = latToY(centerLat, H);\n const pxPerDegLon = W / 360;\n const pxPerDegLat = H / 180;\n\n ctx.beginPath();\n for (let i = 0; i <= numSegments; i++) {\n const bearing = (i / numSegments) * 2 * Math.PI;\n\n // Great-circle destination formula\n const sinLat = Math.sin(lat0) * Math.cos(d) + Math.cos(lat0) * Math.sin(d) * Math.cos(bearing);\n const lat = Math.asin(Math.max(-1, Math.min(1, sinLat)));\n const lon = lon0 + Math.atan2(\n Math.sin(bearing) * Math.sin(d) * Math.cos(lat0),\n Math.cos(d) - Math.sin(lat0) * sinLat,\n );\n\n const pLatDeg = (lat * 180) / Math.PI;\n const pLonDeg = (lon * 180) / Math.PI;\n\n // Pixel offset from center — no normalization, so no date-line jump\n const dLon = pLonDeg - centerLon;\n const dLat = pLatDeg - centerLat;\n const px = cx + dLon * pxPerDegLon;\n const py = cy - dLat * pxPerDegLat; // Y inverted (lat goes up, pixels go down)\n\n if (i === 0) ctx.moveTo(px, py);\n else ctx.lineTo(px, py);\n }\n ctx.closePath();\n}\n//sz ee*s \n\n/**\n * Calculate solar terminator points for a given date and solar depression angle.\n *\n * `depressionDeg` offsets the zenith angle to produce twilight boundaries:\n * - 0° = geometric sunset/sunrise\n * - 6° = civil twilight (IAU/USNO)\n * - 12° = nautical twilight\n * - 18° = astronomical twilight\n */\nfunction calculateTerminatorContinuous(\n date: Date,\n depressionDeg: number = 0,\n numPoints: number = 360,\n): { sunset: Array<[number, number]>, sunrise: Array<[number, number]> } {\n const dayOfYear = Math.floor((date.getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 86400000);\n const declination = -23.44 * Math.cos((2 * Math.PI / 365) * (dayOfYear + 10));\n const decRad = (declination * Math.PI) / 180;\n const hourAngle = ((date.getUTCHours() + date.getUTCMinutes() / 60) / 24) * 360 - 180;\n const depRad = (depressionDeg * Math.PI) / 180;\n\n const sunset: Array<[number, number]> = [];\n const sunrise: Array<[number, number]> = [];\n\n for (let i = 0; i <= numPoints; i++) {\n const lat = -89.999 + (i / numPoints) * 179.998;\n const latRad = lat * (Math.PI / 180);\n const cosH = -(Math.sin(depRad) + Math.sin(latRad) * Math.sin(decRad))\n / (Math.cos(latRad) * Math.cos(decRad));\n \n if (cosH < -1) {\n // All day. No points.\n } else if (cosH > 1) {\n // All night. Covers all longitudes.\n sunset.push([lat, hourAngle]);\n sunrise.push([lat, hourAngle + 360]);\n } else {\n // Normal twilight\n const H = (Math.acos(cosH) * 180) / Math.PI;\n sunset.push([lat, hourAngle + H]);\n sunrise.push([lat, hourAngle + 360 - H]);\n }\n }\n \n return { sunset, sunrise };\n}\n\nfunction buildNightPolygon(\n terminator: { sunset: Array<[number, number]>, sunrise: Array<[number, number]> }\n): [number, number][] {\n const { sunset, sunrise } = terminator;\n if (!sunset.length || !sunrise.length) return [];\n\n const poly: [number, number][] = [];\n\n // Left edge of night (sunset): South to North\n poly.push(...sunset);\n\n // Right edge of night (sunrise): North to South\n poly.push(...[...sunrise].reverse());\n\n return poly;\n}\n\n// =============================================================================\n// Component\n// =============================================================================\n\nexport function GroundTrackMap({\n groundTrack,\n satellites = [],\n groundStations = [],\n accessMask,\n teamPaths,\n showTerminator = true,\n terminatorTime,\n showGrid = true,\n showLegend = true,\n showEquator = true,\n showRecenterButton = false,\n isLoading = false,\n emptyMessage = 'No orbital data available',\n width = '100%',\n height = '100%',\n minHeight = '400px',\n defaultCenter,\n defaultZoom,\n className = '',\n onSatelliteClick,\n onStationClick,\n mapProvider = 'leaflet',\n tileUrl,\n pins,\n pinsEditable = false,\n onPinAdd,\n onPinUpdate,\n onPinRemove,\n}: GroundTrackMapProps): React.ReactElement {\n const { tokens } = useTheme();\n const canvasRef = useRef<HTMLCanvasElement>(null);\n const containerRef = useRef<HTMLDivElement>(null);\n const [tooltip, setTooltip] = useState<{\n x: number; y: number;\n type: 'satellite' | 'station';\n name: string;\n status: string;\n statusColor: string;\n lat: number;\n lon: number;\n alt?: number;\n // Ground station extras\n stationType?: string;\n network?: string;\n coverageRadius?: number;\n // Satellite extras\n trackColor?: string;\n hasFootprint?: boolean;\n futurePoints?: number;\n pastPoints?: number;\n } | null>(null);\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n const [_viewOffset, setViewOffset] = useState<{ x: number; y: number }>({ x: 0, y: 0 });\n\n // Normalize satellites: support SRO teamPaths, legacy single-track, and native API\n const allSatellites = useMemo<SatelliteTrack[]>(() => {\n // Priority 1: native satellites prop\n if (satellites.length > 0) return satellites;\n // Priority 2: SRO teamPaths format — auto-convert\n if (teamPaths && teamPaths.length > 0) {\n return teamPaths.map((tp, idx) => ({\n id: `team-${idx}`,\n name: tp.name,\n color: tp.color,\n status: 'normal' as const,\n groundTrack: tp.positions\n .filter(p => p.lat != null && p.lng != null)\n .map(p => ({\n latitude: p.lat,\n longitude: p.lng,\n altitude: p.altitude ?? 0,\n timestamp: p.time != null ? new Date(p.time * 1000).toISOString() : new Date().toISOString(),\n })),\n }));\n }\n // Priority 3: legacy single-track API\n if (groundTrack && groundTrack.length > 0) {\n return [{\n id: 'primary',\n name: 'Satellite',\n groundTrack,\n status: 'normal',\n accessMask,\n }];\n }\n return [];\n }, [satellites, groundTrack, accessMask, teamPaths]);\n\n // Stable defaults for Leaflet so animated updates don't get new refs and trigger map recreation\n const leafletDefaultCenter = useMemo<[number, number]>(() => defaultCenter ?? [20, 0], [defaultCenter]);\n const leafletDefaultZoom = useMemo(() => defaultZoom ?? 2, [defaultZoom]);\n\n // Optional Leaflet map: load when mapProvider is 'leaflet'\n const [LeafletMap, setLeafletMap] = useState<React.ComponentType<GroundTrackMapLeafletProps> | null>(null);\n const [leafletFailed, setLeafletFailed] = useState(false);\n\n useEffect(() => {\n if (mapProvider !== 'leaflet') return;\n import('./GroundTrackMapLeaflet')\n .then((m) => setLeafletMap(() => m.GroundTrackMapLeaflet))\n .catch((err) => {\n console.warn('[GroundTrackMap] Leaflet backend failed to load. Install \"leaflet\" and ensure it is resolvable. Falling back to static canvas.', err);\n setLeafletFailed(true);\n });\n }, [mapProvider]);\n\n // All canvas-backend hooks must run unconditionally (same hook count every render)\n // Recenter handler (canvas backend)\n const handleRecenter = useCallback(() => {\n setViewOffset({ x: 0, y: 0 });\n }, []);\n\n const COLORS = useMemo(() => ({\n background: tokens.colors.background.base,\n grid: tokens.colors.border.muted,\n text: tokens.colors.text.secondary,\n accent: tokens.colors.accent.primary,\n land: 'rgba(30, 40, 58, 0.92)',\n ocean: tokens.colors.background.base,\n night: 'rgba(0, 10, 40, 0.35)',\n equator: 'rgba(88, 166, 255, 0.25)',\n }), [tokens]);\n\n const lonToX = useCallback((lon: number, w: number) => {\n let normalizedLon = lon;\n while (normalizedLon > 180) normalizedLon -= 360;\n while (normalizedLon < -180) normalizedLon += 360;\n return ((normalizedLon + 180) / 360) * w;\n }, []);\n\n const latToY = useCallback((lat: number, h: number) => {\n return ((90 - lat) / 180) * h;\n }, []);\n\n const draw = useCallback(() => {\n const canvas = canvasRef.current;\n const container = containerRef.current;\n if (!canvas || !container) return;\n\n const ctx = canvas.getContext('2d');\n if (!ctx) return;\n\n const dpr = window.devicePixelRatio || 1;\n const rect = container.getBoundingClientRect();\n const W = rect.width;\n // When height is a CSS string ('100%'), use the measured container height\n const H = typeof height === 'number' ? height : rect.height || 400;\n\n canvas.width = W * dpr;\n canvas.height = H * dpr;\n canvas.style.width = `${W}px`;\n canvas.style.height = `${H}px`;\n ctx.scale(dpr, dpr);\n\n // === Background ===\n ctx.fillStyle = COLORS.background;\n ctx.fillRect(0, 0, W, H);\n\n // === Day/Night Terminator with smooth graduated twilight ===\n // Many thin bands (every 2° of solar depression) produce a seamless\n // day→night gradient instead of hard-edged zones.\n if (showTerminator) {\n const now = terminatorTime ?? new Date();\n\n const BAND_STEP = 2;\n const MAX_DEP = 24;\n const NIGHT_ALPHA = 0.55;\n const bandSteps: Array<{ depression: number; alpha: number }> = [];\n for (let d = 0; d <= MAX_DEP; d += BAND_STEP) {\n // Smooth quadratic ramp: 0° → 0.0, 18° → ~0.45, 24° → 0.55\n const t = Math.min(d / 18, 1);\n const alpha = t * t * (NIGHT_ALPHA * 0.82) + (d > 18 ? (d - 18) / 6 * (NIGHT_ALPHA * 0.18) : 0);\n bandSteps.push({ depression: d, alpha: Math.min(alpha, NIGHT_ALPHA) });\n }\n bandSteps.push({ depression: 90, alpha: NIGHT_ALPHA });\n\n let prevAlpha = 0;\n for (const zone of bandSteps) {\n const dep = Math.min(zone.depression, 89);\n const bandAlpha = zone.alpha - prevAlpha;\n if (bandAlpha < 0.002) { prevAlpha = zone.alpha; continue; }\n\n const terminator = calculateTerminatorContinuous(now, dep);\n if (terminator.sunset.length === 0) { prevAlpha = zone.alpha; continue; }\n \n const poly = buildNightPolygon(terminator);\n if (poly.length < 3) { prevAlpha = zone.alpha; continue; }\n\n ctx.fillStyle = `rgba(0, 6, 24, ${bandAlpha.toFixed(4)})`;\n \n // Draw the polygon (wrapping around longitude boundaries by drawing 3 copies)\n for (const offset of [-360, 0, 360]) {\n ctx.beginPath();\n for (let i = 0; i < poly.length; i++) {\n const lat = poly[i][0];\n const lon = poly[i][1] + offset;\n const x = lonToX(lon, W);\n const y = latToY(lat, H);\n \n // To prevent lines jumping completely across the canvas from right to left\n // if a polygon crosses the map boundary, we should probably handle it.\n // But since we draw 3 full copies (-360, 0, +360), they overlap seamlessly.\n if (i === 0) {\n ctx.moveTo(x, y);\n } else {\n // If distance is huge (e.g. > W/2), it means it's wrapping around the map edge.\n // Move instead of line to avoid a horizontal streak.\n // (Actually, since we unwrap longitude continuously in calculateTerminatorContinuous,\n // there are NO jumps in longitude inside the polygon! It's one continuous shape.)\n ctx.lineTo(x, y);\n }\n }\n ctx.closePath();\n ctx.fill();\n }\n \n prevAlpha = zone.alpha;\n }\n }\n\n // === World Map (land visible against ocean) ===\n ctx.fillStyle = COLORS.land;\n ctx.strokeStyle = 'rgba(100, 120, 160, 0.25)';\n ctx.lineWidth = 1;\n\n for (const pathStr of WORLD_MAP_PATHS) {\n const commands = pathStr.split(' ');\n ctx.beginPath();\n for (const cmd of commands) {\n if (cmd.startsWith('M') || cmd.startsWith('L')) {\n const coords = cmd.substring(1).split(',');\n const lon = parseFloat(coords[0]);\n const lat = parseFloat(coords[1]);\n const x = lonToX(lon, W);\n const y = latToY(lat, H);\n if (cmd.startsWith('M')) ctx.moveTo(x, y);\n else ctx.lineTo(x, y);\n } else if (cmd === 'Z') {\n ctx.closePath();\n }\n }\n ctx.fill();\n ctx.stroke();\n }\n\n // === Grid ===\n if (showGrid) {\n ctx.strokeStyle = COLORS.grid;\n ctx.lineWidth = 0.4;\n for (let lon = -180; lon <= 180; lon += 30) {\n const x = lonToX(lon, W);\n ctx.beginPath(); ctx.moveTo(x, 0); ctx.lineTo(x, H); ctx.stroke();\n }\n for (let lat = -90; lat <= 90; lat += 30) {\n const y = latToY(lat, H);\n ctx.beginPath(); ctx.moveTo(0, y); ctx.lineTo(W, y); ctx.stroke();\n }\n }\n\n // === Equator ===\n if (showEquator) {\n const eqY = latToY(0, H);\n ctx.strokeStyle = COLORS.equator;\n ctx.lineWidth = 1;\n ctx.setLineDash([6, 4]);\n ctx.beginPath(); ctx.moveTo(0, eqY); ctx.lineTo(W, eqY); ctx.stroke();\n ctx.setLineDash([]);\n }\n\n // === Ground Station Coverage Circles (latitude-corrected ellipses) ===\n for (const gs of groundStations) {\n const enhanced = gs as GroundStationEnhanced;\n if (enhanced.showCoverage && enhanced.coverageRadius) {\n const statusColor = STATUS_COLOR_MAP[enhanced.status || 'standby'] || STATUS_COLOR_MAP.standby;\n\n // Use the coverage radius as-is (in degrees) — caller can compute\n // from visibilityRadiusDeg() or provide a direct value\n drawCoverageEllipse(ctx, gs.latitude, gs.longitude, enhanced.coverageRadius, W, H, lonToX, latToY);\n ctx.fillStyle = `${statusColor}15`;\n ctx.fill();\n ctx.strokeStyle = `${statusColor}40`;\n ctx.lineWidth = 1;\n ctx.setLineDash([4, 3]);\n ctx.stroke();\n ctx.setLineDash([]);\n }\n }\n\n // === Ground Stations (Astro UX Icons + Status Badges) ===\n for (const gs of groundStations) {\n const enhanced = gs as GroundStationEnhanced;\n const x = lonToX(gs.longitude, W);\n const y = latToY(gs.latitude, H);\n const status = enhanced.status || 'standby';\n const statusColor = STATUS_COLOR_MAP[status] || STATUS_COLOR_MAP.standby;\n const stationType = enhanced.type || 'dish';\n const iconSize = 20;\n const halfIcon = iconSize / 2;\n\n ctx.save();\n ctx.translate(x, y);\n\n // -- Background circle (subtle glow) --\n ctx.beginPath();\n ctx.arc(0, 0, halfIcon + 2, 0, Math.PI * 2);\n ctx.fillStyle = `${statusColor}18`;\n ctx.fill();\n\n // -- Icon background pill --\n ctx.beginPath();\n ctx.arc(0, 0, halfIcon, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.85)';\n ctx.fill();\n ctx.strokeStyle = `${statusColor}66`;\n ctx.lineWidth = 1.5;\n ctx.stroke();\n\n // -- Draw Astro UX icon via SVG paths scaled to icon size --\n ctx.save();\n const iconScale = iconSize / 24;\n ctx.translate(-halfIcon, -halfIcon);\n ctx.scale(iconScale, iconScale);\n\n if (stationType === 'dish' || stationType === 'omni') {\n // Antenna icon (concentric arcs — Astro antenna pattern)\n ctx.strokeStyle = statusColor;\n ctx.fillStyle = statusColor;\n ctx.lineWidth = 2 / iconScale;\n // Outer arc\n ctx.beginPath(); ctx.arc(12, 12, 10, Math.PI * 1.25, Math.PI * 1.75); ctx.stroke();\n // Middle arc\n ctx.beginPath(); ctx.arc(12, 12, 6.5, Math.PI * 1.2, Math.PI * 1.8); ctx.stroke();\n // Inner arc\n ctx.beginPath(); ctx.arc(12, 12, 3, Math.PI * 1.15, Math.PI * 1.85); ctx.stroke();\n // Center dot (feed point)\n ctx.beginPath(); ctx.arc(12, 12, 1.5, 0, Math.PI * 2); ctx.fill();\n // Stand\n ctx.lineWidth = 1.5 / iconScale;\n ctx.beginPath(); ctx.moveTo(12, 14); ctx.lineTo(12, 20); ctx.stroke();\n ctx.beginPath(); ctx.moveTo(8, 20); ctx.lineTo(16, 20); ctx.stroke();\n } else if (stationType === 'phased-array') {\n // Phased array icon (grid pattern)\n ctx.strokeStyle = statusColor;\n ctx.fillStyle = statusColor;\n ctx.lineWidth = 1.5 / iconScale;\n ctx.strokeRect(5, 4, 14, 12);\n // Grid lines\n for (let gx = 8; gx <= 16; gx += 4) {\n ctx.beginPath(); ctx.moveTo(gx, 4); ctx.lineTo(gx, 16); ctx.stroke();\n }\n for (let gy = 8; gy <= 12; gy += 4) {\n ctx.beginPath(); ctx.moveTo(5, gy); ctx.lineTo(19, gy); ctx.stroke();\n }\n // Stand\n ctx.beginPath(); ctx.moveTo(12, 16); ctx.lineTo(12, 21); ctx.stroke();\n ctx.beginPath(); ctx.moveTo(8, 21); ctx.lineTo(16, 21); ctx.stroke();\n } else {\n // Relay: satellite dish with signal waves\n ctx.strokeStyle = statusColor;\n ctx.fillStyle = statusColor;\n ctx.lineWidth = 2 / iconScale;\n // Dish bowl\n ctx.beginPath(); ctx.arc(10, 12, 7, Math.PI * 0.8, Math.PI * 1.7); ctx.stroke();\n // Feed arm\n ctx.lineWidth = 1.5 / iconScale;\n ctx.beginPath(); ctx.moveTo(10, 12); ctx.lineTo(16, 6); ctx.stroke();\n // Signal waves\n ctx.lineWidth = 1.2 / iconScale;\n ctx.beginPath(); ctx.arc(17, 5, 2.5, Math.PI * 1.2, Math.PI * 1.7); ctx.stroke();\n ctx.beginPath(); ctx.arc(17, 5, 4.5, Math.PI * 1.15, Math.PI * 1.75); ctx.stroke();\n // Base\n ctx.lineWidth = 1.5 / iconScale;\n ctx.beginPath(); ctx.moveTo(10, 14); ctx.lineTo(10, 20); ctx.stroke();\n ctx.beginPath(); ctx.moveTo(6, 20); ctx.lineTo(14, 20); ctx.stroke();\n }\n\n ctx.restore(); // icon scaling\n\n // -- Astro UX Status Badge (top-left of icon) --\n const badgeX = -halfIcon + 1;\n const badgeY = -halfIcon + 1;\n const badgeSize = 7;\n\n // Badge background ring\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 2 + 2, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.9)';\n ctx.fill();\n\n // Status shape per Astro UX spec (StatusIndicator.tsx canonical reference):\n // off → small filled dot, standby → ring (hollow circle),\n // normal → filled circle, caution → square,\n // serious → diamond, critical → triangle pointing DOWN\n ctx.fillStyle = statusColor;\n ctx.strokeStyle = statusColor;\n ctx.lineWidth = 1;\n if (status === 'normal') {\n // Filled circle for normal\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 2, 0, Math.PI * 2);\n ctx.fill();\n } else if (status === 'standby') {\n // Ring (hollow circle) for standby\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 2 - 0.5, 0, Math.PI * 2);\n ctx.lineWidth = 2;\n ctx.stroke();\n } else if (status === 'caution') {\n // Square for caution\n const hs = badgeSize / 2;\n ctx.fillRect(badgeX - hs, badgeY - hs, badgeSize, badgeSize);\n } else if (status === 'serious') {\n // Diamond for serious\n const hs = badgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(badgeX, badgeY - hs);\n ctx.lineTo(badgeX + hs, badgeY);\n ctx.lineTo(badgeX, badgeY + hs);\n ctx.lineTo(badgeX - hs, badgeY);\n ctx.closePath();\n ctx.fill();\n } else if (status === 'critical') {\n // Triangle pointing DOWN for critical\n const hs = badgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(badgeX, badgeY + hs); // bottom point\n ctx.lineTo(badgeX + hs, badgeY - hs); // top-right\n ctx.lineTo(badgeX - hs, badgeY - hs); // top-left\n ctx.closePath();\n ctx.fill();\n } else {\n // Off: small filled dot\n ctx.beginPath();\n ctx.arc(badgeX, badgeY, badgeSize / 3, 0, Math.PI * 2);\n ctx.fill();\n }\n\n ctx.restore();\n\n // Label\n ctx.fillStyle = COLORS.text;\n ctx.font = '9px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(gs.name.length > 14 ? gs.name.slice(0, 14) + '…' : gs.name, x, y + halfIcon + 12);\n }\n\n // === Satellite Tracks ===\n allSatellites.forEach((sat, satIdx) => {\n const track = sat.groundTrack;\n if (!track || track.length === 0) return;\n\n const trackColor = sat.color || DEFAULT_TRACK_COLORS[satIdx % DEFAULT_TRACK_COLORS.length];\n const futureIdx = sat.futureTrackIndex ?? track.length;\n\n // Draw past track (solid)\n ctx.strokeStyle = trackColor;\n ctx.lineWidth = 2;\n ctx.lineCap = 'round';\n ctx.lineJoin = 'round';\n ctx.setLineDash([]);\n ctx.beginPath();\n\n let prevX = lonToX(track[0].longitude, W);\n let prevY = latToY(track[0].latitude, H);\n ctx.moveTo(prevX, prevY);\n\n for (let i = 1; i < Math.min(futureIdx, track.length); i++) {\n const x = lonToX(track[i].longitude, W);\n const y = latToY(track[i].latitude, H);\n const crossesDateLine = Math.abs(x - prevX) > W / 2;\n if (crossesDateLine) {\n ctx.stroke(); ctx.beginPath(); ctx.moveTo(x, y);\n } else {\n ctx.lineTo(x, y);\n }\n prevX = x; prevY = y;\n }\n ctx.stroke();\n\n // Draw future track (dashed)\n if (futureIdx < track.length) {\n ctx.strokeStyle = `${trackColor}88`;\n ctx.lineWidth = 1.5;\n ctx.setLineDash([6, 4]);\n ctx.beginPath();\n ctx.moveTo(lonToX(track[futureIdx].longitude, W), latToY(track[futureIdx].latitude, H));\n let fpx = lonToX(track[futureIdx].longitude, W);\n\n for (let i = futureIdx + 1; i < track.length; i++) {\n const x = lonToX(track[i].longitude, W);\n const y = latToY(track[i].latitude, H);\n if (Math.abs(x - fpx) > W / 2) {\n ctx.stroke(); ctx.beginPath(); ctx.moveTo(x, y);\n } else {\n ctx.lineTo(x, y);\n }\n fpx = x;\n }\n ctx.stroke();\n ctx.setLineDash([]);\n }\n\n // Draw access segments (contact passes)\n if (sat.accessMask && sat.accessMask.some(Boolean)) {\n ctx.strokeStyle = STATUS_COLOR_MAP.normal;\n ctx.lineWidth = 3;\n ctx.setLineDash([]);\n ctx.beginPath();\n let drawing = false;\n let apx = lonToX(track[0].longitude, W);\n\n for (let i = 0; i < track.length; i++) {\n const x = lonToX(track[i].longitude, W);\n const y = latToY(track[i].latitude, H);\n const active = sat.accessMask[i];\n const crossesDateLine = i > 0 && Math.abs(x - apx) > W / 2;\n\n if (active && !crossesDateLine) {\n if (!drawing) { ctx.moveTo(x, y); drawing = true; }\n else ctx.lineTo(x, y);\n } else {\n if (drawing) { ctx.stroke(); ctx.beginPath(); drawing = false; }\n }\n apx = x;\n }\n if (drawing) ctx.stroke();\n }\n\n // AOS/LOS markers\n if (sat.passMarkers) {\n for (const marker of sat.passMarkers) {\n const mx = lonToX(marker.longitude, W);\n const my = latToY(marker.latitude, H);\n const markerColor = marker.type === 'aos' ? STATUS_COLOR_MAP.normal : STATUS_COLOR_MAP.critical;\n\n ctx.save();\n ctx.translate(mx, my);\n // Triangle marker\n ctx.beginPath();\n if (marker.type === 'aos') {\n ctx.moveTo(0, -6); ctx.lineTo(4, 2); ctx.lineTo(-4, 2);\n } else {\n ctx.moveTo(0, 6); ctx.lineTo(4, -2); ctx.lineTo(-4, -2);\n }\n ctx.closePath();\n ctx.fillStyle = markerColor;\n ctx.fill();\n ctx.restore();\n\n if (marker.label) {\n ctx.fillStyle = markerColor;\n ctx.font = '8px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(marker.label, mx, marker.type === 'aos' ? my - 9 : my + 13);\n }\n }\n }\n\n // Satellite footprint (latitude-corrected ellipse)\n if (sat.showFootprint && sat.footprintRadius) {\n const lastPt = futureIdx > 0 ? track[Math.min(futureIdx - 1, track.length - 1)] : track[track.length - 1];\n\n drawCoverageEllipse(ctx, lastPt.latitude, lastPt.longitude, sat.footprintRadius, W, H, lonToX, latToY);\n ctx.fillStyle = `${trackColor}12`;\n ctx.fill();\n ctx.strokeStyle = `${trackColor}50`;\n ctx.lineWidth = 1;\n ctx.setLineDash([3, 3]);\n ctx.stroke();\n ctx.setLineDash([]);\n }\n\n // Current position satellite icon (Astro UX style)\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, track.length - 1) : track.length - 1;\n const current = track[currentIdx];\n const cx = lonToX(current.longitude, W);\n const cy = latToY(current.latitude, H);\n const satStatus = sat.status || 'normal';\n const statusColor = STATUS_COLOR_MAP[satStatus] || STATUS_COLOR_MAP.normal;\n const satIconSize = 22;\n const satHalf = satIconSize / 2;\n\n // Outer pulsing glow\n ctx.beginPath();\n ctx.arc(cx, cy, satHalf + 6, 0, Math.PI * 2);\n ctx.fillStyle = `${statusColor}12`;\n ctx.fill();\n\n ctx.beginPath();\n ctx.arc(cx, cy, satHalf + 3, 0, Math.PI * 2);\n ctx.fillStyle = `${statusColor}20`;\n ctx.fill();\n\n // Icon background\n ctx.save();\n ctx.translate(cx, cy);\n\n ctx.beginPath();\n ctx.arc(0, 0, satHalf, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.88)';\n ctx.fill();\n ctx.strokeStyle = `${statusColor}88`;\n ctx.lineWidth = 1.5;\n ctx.stroke();\n\n // Draw satellite shape (body + solar panels)\n ctx.fillStyle = statusColor;\n // Body (rounded rect approximation)\n const bw = 5, bh = 3.5;\n ctx.beginPath();\n ctx.moveTo(-bw / 2 + 0.5, -bh / 2);\n ctx.lineTo(bw / 2 - 0.5, -bh / 2);\n ctx.arcTo(bw / 2, -bh / 2, bw / 2, -bh / 2 + 0.5, 0.5);\n ctx.lineTo(bw / 2, bh / 2 - 0.5);\n ctx.arcTo(bw / 2, bh / 2, bw / 2 - 0.5, bh / 2, 0.5);\n ctx.lineTo(-bw / 2 + 0.5, bh / 2);\n ctx.arcTo(-bw / 2, bh / 2, -bw / 2, bh / 2 - 0.5, 0.5);\n ctx.lineTo(-bw / 2, -bh / 2 + 0.5);\n ctx.arcTo(-bw / 2, -bh / 2, -bw / 2 + 0.5, -bh / 2, 0.5);\n ctx.closePath();\n ctx.fill();\n\n // Solar panel left\n ctx.fillStyle = `${statusColor}BB`;\n ctx.fillRect(-satHalf + 2, -2, 5, 4);\n // Panel lines\n ctx.strokeStyle = `${statusColor}55`;\n ctx.lineWidth = 0.5;\n ctx.beginPath();\n ctx.moveTo(-satHalf + 4.5, -2); ctx.lineTo(-satHalf + 4.5, 2);\n ctx.stroke();\n\n // Solar panel right\n ctx.fillStyle = `${statusColor}BB`;\n ctx.fillRect(satHalf - 7, -2, 5, 4);\n ctx.strokeStyle = `${statusColor}55`;\n ctx.beginPath();\n ctx.moveTo(satHalf - 4.5, -2); ctx.lineTo(satHalf - 4.5, 2);\n ctx.stroke();\n\n // Panel connectors\n ctx.strokeStyle = statusColor;\n ctx.lineWidth = 1;\n ctx.beginPath();\n ctx.moveTo(-bw / 2, 0); ctx.lineTo(-satHalf + 7, 0);\n ctx.moveTo(bw / 2, 0); ctx.lineTo(satHalf - 7, 0);\n ctx.stroke();\n\n // -- Astro UX Status Badge (top-left of satellite icon) --\n // Shapes per canonical StatusIndicator.tsx:\n // off → small dot, standby → ring, normal → filled circle,\n // caution → square, serious → diamond, critical → triangle DOWN\n const satBadgeX = -satHalf + 2;\n const satBadgeY = -satHalf + 2;\n const satBadgeSize = 7;\n\n // Badge background\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 2 + 2, 0, Math.PI * 2);\n ctx.fillStyle = 'rgba(15, 23, 42, 0.9)';\n ctx.fill();\n\n ctx.fillStyle = statusColor;\n ctx.strokeStyle = statusColor;\n ctx.lineWidth = 1;\n if (satStatus === 'normal') {\n // Filled circle\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 2, 0, Math.PI * 2);\n ctx.fill();\n } else if (satStatus === 'standby') {\n // Ring (hollow circle)\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 2 - 0.5, 0, Math.PI * 2);\n ctx.lineWidth = 2;\n ctx.stroke();\n } else if (satStatus === 'caution') {\n const hs = satBadgeSize / 2;\n ctx.fillRect(satBadgeX - hs, satBadgeY - hs, satBadgeSize, satBadgeSize);\n } else if (satStatus === 'serious') {\n // Diamond\n const hs = satBadgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(satBadgeX, satBadgeY - hs);\n ctx.lineTo(satBadgeX + hs, satBadgeY);\n ctx.lineTo(satBadgeX, satBadgeY + hs);\n ctx.lineTo(satBadgeX - hs, satBadgeY);\n ctx.closePath();\n ctx.fill();\n } else if (satStatus === 'critical') {\n // Triangle pointing DOWN\n const hs = satBadgeSize / 2;\n ctx.beginPath();\n ctx.moveTo(satBadgeX, satBadgeY + hs); // bottom point\n ctx.lineTo(satBadgeX + hs, satBadgeY - hs); // top-right\n ctx.lineTo(satBadgeX - hs, satBadgeY - hs); // top-left\n ctx.closePath();\n ctx.fill();\n } else {\n // Off: small filled dot\n ctx.beginPath();\n ctx.arc(satBadgeX, satBadgeY, satBadgeSize / 3, 0, Math.PI * 2);\n ctx.fill();\n }\n\n ctx.restore();\n\n // Satellite label\n ctx.fillStyle = '#fff';\n ctx.font = 'bold 10px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(sat.name, cx, cy - satHalf - 6);\n });\n\n // === Legend ===\n if (showLegend && (allSatellites.length > 0 || groundStations.length > 0)) {\n const legendX = 10;\n const legendY = 10;\n const lineHeight = 14;\n const items: Array<{ color: string; label: string; dash?: boolean }> = [];\n\n allSatellites.forEach((sat, idx) => {\n const c = sat.color || DEFAULT_TRACK_COLORS[idx % DEFAULT_TRACK_COLORS.length];\n items.push({ color: c, label: sat.name });\n });\n if (groundStations.length > 0) {\n items.push({ color: STATUS_COLOR_MAP.standby, label: `${groundStations.length} Ground Station${groundStations.length > 1 ? 's' : ''}` });\n }\n\n const maxLabelWidth = Math.max(...items.map(item => ctx.measureText(item.label).width)) + 28;\n const legendH = items.length * lineHeight + 10;\n\n ctx.fillStyle = 'rgba(0, 0, 0, 0.6)';\n ctx.strokeStyle = 'rgba(100, 116, 139, 0.3)';\n ctx.lineWidth = 1;\n const r = 4;\n ctx.beginPath();\n ctx.moveTo(legendX + r, legendY);\n ctx.lineTo(legendX + maxLabelWidth - r, legendY);\n ctx.arcTo(legendX + maxLabelWidth, legendY, legendX + maxLabelWidth, legendY + r, r);\n ctx.lineTo(legendX + maxLabelWidth, legendY + legendH - r);\n ctx.arcTo(legendX + maxLabelWidth, legendY + legendH, legendX + maxLabelWidth - r, legendY + legendH, r);\n ctx.lineTo(legendX + r, legendY + legendH);\n ctx.arcTo(legendX, legendY + legendH, legendX, legendY + legendH - r, r);\n ctx.lineTo(legendX, legendY + r);\n ctx.arcTo(legendX, legendY, legendX + r, legendY, r);\n ctx.closePath();\n ctx.fill();\n ctx.stroke();\n\n items.forEach((item, i) => {\n const iy = legendY + 8 + i * lineHeight;\n // Color swatch\n ctx.fillStyle = item.color;\n ctx.beginPath();\n ctx.arc(legendX + 10, iy + 3, 4, 0, Math.PI * 2);\n ctx.fill();\n // Label\n ctx.fillStyle = 'rgba(255, 255, 255, 0.85)';\n ctx.font = '10px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'left';\n ctx.fillText(item.label, legendX + 20, iy + 7);\n });\n }\n\n // === Pins (read-only on canvas backend) ===\n if (pins && pins.length > 0) {\n pins.forEach((pin) => {\n const px = lonToX(pin.longitude, W);\n const py = latToY(pin.latitude, H);\n const pinColor = pin.color || tokens.colors.accent.primary;\n\n // Drop shadow\n ctx.save();\n ctx.shadowColor = 'rgba(0,0,0,0.4)';\n ctx.shadowBlur = 6;\n ctx.shadowOffsetY = 2;\n\n // Pin body: inverted teardrop shape\n ctx.beginPath();\n ctx.arc(px, py - 10, 6, Math.PI, 0);\n ctx.lineTo(px, py);\n ctx.closePath();\n ctx.fillStyle = pinColor;\n ctx.fill();\n ctx.restore();\n\n // Inner dot\n ctx.beginPath();\n ctx.arc(px, py - 10, 2.5, 0, Math.PI * 2);\n ctx.fillStyle = '#0d1323';\n ctx.fill();\n\n // Label\n if (pin.label) {\n ctx.fillStyle = 'rgba(255,255,255,0.9)';\n ctx.font = '10px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(pin.label, px, py - 20);\n }\n });\n }\n\n // === No Data Message ===\n if (allSatellites.length === 0 && groundStations.length === 0 && (!pins || pins.length === 0)) {\n ctx.fillStyle = COLORS.text;\n ctx.font = '12px \"Inter\", system-ui, sans-serif';\n ctx.textAlign = 'center';\n ctx.fillText(emptyMessage, W / 2, H / 2);\n }\n }, [allSatellites, groundStations, height, showTerminator, terminatorTime, showGrid, showLegend, showEquator, lonToX, latToY, COLORS, emptyMessage, pins, tokens.colors.accent.primary]);\n\n // Tooltip on mouse move\n const handleMouseMove = useCallback((e: React.MouseEvent<HTMLCanvasElement>) => {\n const canvas = canvasRef.current;\n if (!canvas) return;\n const rect = canvas.getBoundingClientRect();\n const mx = e.clientX - rect.left;\n const my = e.clientY - rect.top;\n const W = rect.width;\n const H = typeof height === 'number' ? height : rect.height;\n\n // Check satellite positions\n for (const sat of allSatellites) {\n if (!sat.groundTrack.length) continue;\n const futureIdx = sat.futureTrackIndex ?? sat.groundTrack.length;\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, sat.groundTrack.length - 1) : sat.groundTrack.length - 1;\n const lastPt = sat.groundTrack[currentIdx];\n const sx = lonToX(lastPt.longitude, W);\n const sy = latToY(lastPt.latitude, H);\n const dist = Math.sqrt((mx - sx) ** 2 + (my - sy) ** 2);\n if (dist < 18) {\n const satStatus = sat.status || 'normal';\n setTooltip({\n x: e.clientX - rect.left + 15,\n y: e.clientY - rect.top - 10,\n type: 'satellite',\n name: sat.name,\n status: satStatus,\n statusColor: STATUS_COLOR_MAP[satStatus] || STATUS_COLOR_MAP.normal,\n lat: lastPt.latitude,\n lon: lastPt.longitude,\n alt: lastPt.altitude,\n trackColor: sat.color || DEFAULT_TRACK_COLORS[allSatellites.indexOf(sat) % DEFAULT_TRACK_COLORS.length],\n hasFootprint: sat.showFootprint,\n futurePoints: sat.groundTrack.length - futureIdx,\n pastPoints: futureIdx,\n });\n return;\n }\n }\n\n // Check ground stations\n for (const gs of groundStations) {\n const gx = lonToX(gs.longitude, W);\n const gy = latToY(gs.latitude, H);\n const dist = Math.sqrt((mx - gx) ** 2 + (my - gy) ** 2);\n if (dist < 16) {\n const enhanced = gs as GroundStationEnhanced;\n const gsStatus = enhanced.status || 'standby';\n setTooltip({\n x: e.clientX - rect.left + 15,\n y: e.clientY - rect.top - 10,\n type: 'station',\n name: gs.name,\n status: gsStatus,\n statusColor: STATUS_COLOR_MAP[gsStatus] || STATUS_COLOR_MAP.standby,\n lat: gs.latitude,\n lon: gs.longitude,\n stationType: enhanced.type || 'dish',\n network: ('network' in gs && gs.network) ? String(gs.network) : undefined,\n coverageRadius: enhanced.coverageRadius,\n });\n return;\n }\n }\n\n setTooltip(null);\n }, [allSatellites, groundStations, height, lonToX, latToY]);\n\n const handleClick = useCallback((e: React.MouseEvent<HTMLCanvasElement>) => {\n const canvas = canvasRef.current;\n if (!canvas) return;\n const rect = canvas.getBoundingClientRect();\n const mx = e.clientX - rect.left;\n const my = e.clientY - rect.top;\n const W = rect.width;\n const H = typeof height === 'number' ? height : rect.height;\n\n for (const sat of allSatellites) {\n if (!sat.groundTrack.length) continue;\n const futureIdx = sat.futureTrackIndex ?? sat.groundTrack.length;\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, sat.groundTrack.length - 1) : sat.groundTrack.length - 1;\n const currentPt = sat.groundTrack[currentIdx];\n const sx = lonToX(currentPt.longitude, W);\n const sy = latToY(currentPt.latitude, H);\n if (Math.sqrt((mx - sx) ** 2 + (my - sy) ** 2) < 15) {\n onSatelliteClick?.(sat.id);\n return;\n }\n }\n\n for (const gs of groundStations) {\n const gx = lonToX(gs.longitude, W);\n const gy = latToY(gs.latitude, H);\n if (Math.sqrt((mx - gx) ** 2 + (my - gy) ** 2) < 12) {\n onStationClick?.('id' in gs ? gs.id : gs.name);\n return;\n }\n }\n }, [allSatellites, groundStations, height, lonToX, latToY, onSatelliteClick, onStationClick]);\n\n useEffect(() => {\n draw();\n window.addEventListener('resize', draw);\n return () => window.removeEventListener('resize', draw);\n }, [draw]);\n\n // Leaflet backend: return after all hooks so hook count is stable\n const resolvedMinHeightForLeaflet = minHeight || (typeof height === 'number' ? `${height}px` : '400px');\n\n if (mapProvider === 'leaflet') {\n if (LeafletMap) {\n return (\n <div\n role=\"img\"\n aria-label={allSatellites.length > 0 || groundStations.length > 0 ? 'Ground track map showing satellite tracks and ground stations' : 'Ground track map'}\n style={{\n display: 'block',\n width: width ?? '100%',\n height: typeof height === 'string' ? height : undefined,\n minHeight: resolvedMinHeightForLeaflet,\n }}\n >\n <LeafletMap\n allSatellites={allSatellites}\n groundStations={groundStations}\n showTerminator={showTerminator}\n terminatorTime={terminatorTime}\n showGrid={showGrid}\n showLegend={showLegend}\n showEquator={showEquator}\n showRecenterButton={showRecenterButton}\n defaultCenter={leafletDefaultCenter}\n defaultZoom={leafletDefaultZoom}\n height={height}\n width={width}\n minHeight={minHeight}\n emptyMessage={emptyMessage}\n tileUrl={tileUrl}\n className={className}\n onSatelliteClick={onSatelliteClick}\n onStationClick={onStationClick}\n pins={pins}\n pinsEditable={pinsEditable}\n onPinAdd={onPinAdd}\n onPinUpdate={onPinUpdate}\n onPinRemove={onPinRemove}\n />\n </div>\n );\n }\n if (!leafletFailed) {\n return (\n <div\n className={`zendir-ground-track-map ${className}`}\n style={{\n width,\n height: typeof height === 'string' ? height : undefined,\n minHeight: resolvedMinHeightForLeaflet,\n backgroundColor: tokens.colors.background.base,\n borderRadius: 8,\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n }}\n >\n <span style={{ color: tokens.colors.text.secondary, fontSize: 14 }}>Loading map…</span>\n </div>\n );\n }\n }\n\n const resolvedMinHeight = minHeight || (typeof height === 'number' ? `${height}px` : '400px');\n\n // Loading state\n if (isLoading) {\n return (\n <div\n className={`zendir-ground-track-map ${className}`}\n style={{\n width,\n minHeight: resolvedMinHeight,\n backgroundColor: COLORS.background,\n borderRadius: '8px',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n }}\n >\n <span style={{ color: COLORS.text, fontSize: '14px', letterSpacing: '1px' }}>Loading…</span>\n </div>\n );\n }\n\n return (\n <div\n ref={containerRef}\n role=\"img\"\n aria-label={allSatellites.length > 0 || groundStations.length > 0 ? 'Ground track map showing satellite tracks and ground stations' : 'Ground track map'}\n className={`zendir-ground-track-map ${className}`}\n style={{ width, height: typeof height === 'string' ? height : undefined, minHeight: resolvedMinHeight, backgroundColor: COLORS.background, borderRadius: '8px', overflow: 'hidden', position: 'relative' }}\n data-map-backend=\"canvas\"\n >\n {/* Hint when Leaflet was requested but failed (e.g. leaflet not installed) */}\n {mapProvider === 'leaflet' && leafletFailed && (\n <div\n style={{\n position: 'absolute',\n bottom: 8,\n left: 8,\n zIndex: 100,\n padding: '4px 8px',\n background: 'rgba(0,0,0,0.75)',\n color: '#fce83a',\n fontSize: 11,\n borderRadius: 4,\n }}\n >\n Static view (install <code style={{ fontFamily: 'monospace' }}>leaflet</code> for zoom/pan)\n </div>\n )}\n {/* Recenter button (SRO compat) */}\n {showRecenterButton && (\n <button\n type=\"button\"\n onClick={handleRecenter}\n title=\"Recenter Map\"\n aria-label=\"Recenter map\"\n style={{\n position: 'absolute',\n top: '8px',\n right: '8px',\n zIndex: 20,\n background: 'rgba(30, 41, 59, 0.85)',\n border: '1px solid rgba(100, 116, 139, 0.4)',\n borderRadius: '4px',\n color: '#fff',\n cursor: 'pointer',\n padding: '4px 8px',\n fontSize: '12px',\n lineHeight: '1',\n display: 'flex',\n alignItems: 'center',\n gap: '4px',\n }}\n >\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth=\"2\" strokeLinecap=\"round\" strokeLinejoin=\"round\">\n <circle cx=\"12\" cy=\"12\" r=\"3\" />\n <path d=\"M12 2v4M12 18v4M2 12h4M18 12h4\" />\n </svg>\n </button>\n )}\n\n <canvas\n ref={canvasRef}\n style={{ display: 'block', cursor: tooltip ? 'pointer' : 'default' }}\n onMouseMove={handleMouseMove}\n onMouseLeave={() => setTooltip(null)}\n onClick={handleClick}\n />\n {tooltip && (\n <div\n style={{\n position: 'absolute',\n left: tooltip.x,\n top: tooltip.y,\n background: 'rgba(15, 23, 42, 0.95)',\n backdropFilter: 'blur(8px)',\n border: `1px solid ${tooltip.statusColor}40`,\n borderRadius: '8px',\n padding: 0,\n pointerEvents: 'none',\n zIndex: 10,\n color: 'rgba(255, 255, 255, 0.9)',\n fontSize: '11px',\n fontFamily: '\"Inter\", system-ui, sans-serif',\n lineHeight: '1.5',\n boxShadow: `0 4px 16px rgba(0, 0, 0, 0.5), 0 0 12px ${tooltip.statusColor}15`,\n minWidth: 180,\n overflow: 'hidden',\n }}\n >\n {/* Header with status accent */}\n <div style={{\n padding: '6px 10px',\n borderBottom: `1px solid ${tooltip.statusColor}25`,\n background: `${tooltip.statusColor}10`,\n display: 'flex', alignItems: 'center', gap: 6,\n }}>\n <span style={{\n width: 8, height: 8, borderRadius: tooltip.status === 'caution' ? 1 : '50%',\n background: tooltip.statusColor, display: 'inline-block', flexShrink: 0,\n boxShadow: `0 0 4px ${tooltip.statusColor}88`,\n }} />\n <span style={{ fontWeight: 600, fontSize: 12 }}>{tooltip.name}</span>\n <span style={{\n marginLeft: 'auto', fontSize: 9, fontWeight: 600, textTransform: 'uppercase',\n color: tooltip.statusColor, letterSpacing: '0.5px',\n }}>\n {tooltip.status}\n </span>\n </div>\n {/* Details */}\n <div style={{ padding: '6px 10px', display: 'flex', flexDirection: 'column', gap: 3 }}>\n {/* Type badge */}\n <div style={{ display: 'flex', alignItems: 'center', gap: 6, marginBottom: 2 }}>\n <span style={{\n fontSize: 9, fontWeight: 600, padding: '1px 5px',\n borderRadius: 3, textTransform: 'uppercase', letterSpacing: '0.3px',\n background: tooltip.type === 'satellite' ? 'rgba(45, 204, 255, 0.15)' : 'rgba(86, 240, 0, 0.15)',\n color: tooltip.type === 'satellite' ? '#2dccff' : '#56f000',\n }}>\n {tooltip.type === 'satellite' ? (\n <>\n <AstroIcon name=\"satellite\" size=\"extra-small\" label=\"Satellite\" style={{ marginRight: 4, verticalAlign: 'middle' }} />\n Satellite\n </>\n ) : (\n <>\n <AstroIcon name=\"antenna\" size=\"extra-small\" label={tooltip.stationType || 'Ground station'} style={{ marginRight: 4, verticalAlign: 'middle' }} />\n {tooltip.stationType || 'dish'}\n </>\n )}\n </span>\n {tooltip.network && (\n <span style={{\n fontSize: 9, padding: '1px 5px', borderRadius: 3,\n background: 'rgba(157, 112, 255, 0.15)', color: '#9D70FF',\n }}>\n {tooltip.network}\n </span>\n )}\n </div>\n {/* Coordinates */}\n <div style={{ display: 'grid', gridTemplateColumns: '52px 1fr', gap: '2px 8px', fontSize: 10 }}>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Lat</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.lat >= 0 ? '+' : ''}{tooltip.lat.toFixed(4)}°\n </span>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Lon</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.lon >= 0 ? '+' : ''}{tooltip.lon.toFixed(4)}°\n </span>\n {tooltip.alt !== undefined && (\n <>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Altitude</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.alt.toFixed(1)} km\n </span>\n </>\n )}\n {tooltip.coverageRadius !== undefined && (\n <>\n <span style={{ color: 'rgba(255,255,255,0.5)' }}>Coverage</span>\n <span style={{ fontFamily: '\"Roboto Mono\", monospace', fontWeight: 500 }}>\n {tooltip.coverageRadius.toFixed(1)}° radius\n </span>\n </>\n )}\n </div>\n {/* Satellite extras */}\n {tooltip.type === 'satellite' && (tooltip.pastPoints !== undefined || tooltip.futurePoints !== undefined) && (\n <div style={{ marginTop: 3, paddingTop: 3, borderTop: '1px solid rgba(255,255,255,0.08)', display: 'flex', gap: 10, fontSize: 9 }}>\n {tooltip.pastPoints !== undefined && (\n <span><span style={{ color: tooltip.trackColor }}>━</span> {tooltip.pastPoints} past pts</span>\n )}\n {tooltip.futurePoints !== undefined && tooltip.futurePoints > 0 && (\n <span><span style={{ color: `${tooltip.trackColor}88` }}>╌</span> {tooltip.futurePoints} predicted</span>\n )}\n {tooltip.hasFootprint && <span style={{ color: '#9D70FF' }}>◎ Footprint</span>}\n </div>\n )}\n </div>\n </div>\n )}\n </div>\n );\n}\n\nexport default GroundTrackMap;\n"],"names":[],"mappings":"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"}
|
|
@@ -102,42 +102,37 @@ function createStationDivIcon(statusColor, type, status) {
|
|
|
102
102
|
tooltipAnchor: [0, -14]
|
|
103
103
|
});
|
|
104
104
|
}
|
|
105
|
-
function calculateTerminatorContinuous(date, depressionDeg = 0, numPoints =
|
|
105
|
+
function calculateTerminatorContinuous(date, depressionDeg = 0, numPoints = 360) {
|
|
106
106
|
const dayOfYear = Math.floor((date.getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 864e5);
|
|
107
107
|
const declination = -23.44 * Math.cos(2 * Math.PI / 365 * (dayOfYear + 10));
|
|
108
108
|
const decRad = declination * Math.PI / 180;
|
|
109
109
|
const hourAngle = (date.getUTCHours() + date.getUTCMinutes() / 60) / 24 * 360 - 180;
|
|
110
110
|
const depRad = depressionDeg * Math.PI / 180;
|
|
111
|
-
const
|
|
112
|
-
|
|
111
|
+
const sunset = [];
|
|
112
|
+
const sunrise = [];
|
|
113
113
|
for (let i = 0; i <= numPoints; i++) {
|
|
114
|
-
const
|
|
114
|
+
const lat = -89.999 + i / numPoints * 179.998;
|
|
115
|
+
const latRad = lat * (Math.PI / 180);
|
|
115
116
|
const cosH = -(Math.sin(depRad) + Math.sin(latRad) * Math.sin(decRad)) / (Math.cos(latRad) * Math.cos(decRad));
|
|
116
|
-
|
|
117
|
-
if (cosH
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
117
|
+
if (cosH < -1) ;
|
|
118
|
+
else if (cosH > 1) {
|
|
119
|
+
sunset.push([lat, hourAngle]);
|
|
120
|
+
sunrise.push([lat, hourAngle + 360]);
|
|
121
|
+
} else {
|
|
122
|
+
const H = Math.acos(cosH) * 180 / Math.PI;
|
|
123
|
+
sunset.push([lat, hourAngle + H]);
|
|
124
|
+
sunrise.push([lat, hourAngle + 360 - H]);
|
|
123
125
|
}
|
|
124
|
-
prevLon = lon;
|
|
125
|
-
points.push([i / numPoints * 180 - 90, lon]);
|
|
126
126
|
}
|
|
127
|
-
return
|
|
128
|
-
}
|
|
129
|
-
function buildBandPolygon(outerSunset, innerSunset) {
|
|
130
|
-
if (!outerSunset.length || !innerSunset.length) return [];
|
|
131
|
-
const outerSunrise = outerSunset.map(([lat, lon]) => [lat, lon - 180]);
|
|
132
|
-
const innerSunrise = innerSunset.map(([lat, lon]) => [lat, lon - 180]);
|
|
133
|
-
const sunsetBand = [...outerSunset, ...[...innerSunset].reverse()];
|
|
134
|
-
const sunriseBand = [...innerSunrise, ...[...outerSunrise].reverse()];
|
|
135
|
-
return [...sunsetBand, ...sunriseBand];
|
|
127
|
+
return { sunset, sunrise };
|
|
136
128
|
}
|
|
137
|
-
function buildNightPolygon(
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
129
|
+
function buildNightPolygon(terminator) {
|
|
130
|
+
const { sunset, sunrise } = terminator;
|
|
131
|
+
if (!sunset.length || !sunrise.length) return [];
|
|
132
|
+
const poly = [];
|
|
133
|
+
poly.push(...sunset);
|
|
134
|
+
poly.push(...[...sunrise].reverse());
|
|
135
|
+
return poly;
|
|
141
136
|
}
|
|
142
137
|
function GroundTrackMapLeaflet({
|
|
143
138
|
allSatellites,
|
|
@@ -271,44 +266,37 @@ function GroundTrackMapLeaflet({
|
|
|
271
266
|
clearLayers();
|
|
272
267
|
if (showTerminator) {
|
|
273
268
|
const now = terminatorTime ?? /* @__PURE__ */ new Date();
|
|
274
|
-
const
|
|
275
|
-
const
|
|
276
|
-
const
|
|
277
|
-
const
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
269
|
+
const BAND_STEP = 2;
|
|
270
|
+
const MAX_DEP = 24;
|
|
271
|
+
const NIGHT_OPACITY = 0.55;
|
|
272
|
+
const bandSteps = [];
|
|
273
|
+
for (let d = 0; d <= MAX_DEP; d += BAND_STEP) {
|
|
274
|
+
const t = Math.min(d / 18, 1);
|
|
275
|
+
const opacity = t * t * (NIGHT_OPACITY * 0.82) + (d > 18 ? (d - 18) / 6 * (NIGHT_OPACITY * 0.18) : 0);
|
|
276
|
+
bandSteps.push({ depression: d, opacity: Math.min(opacity, NIGHT_OPACITY) });
|
|
277
|
+
}
|
|
278
|
+
bandSteps.push({ depression: 90, opacity: NIGHT_OPACITY });
|
|
279
|
+
let prevOpacity = 0;
|
|
280
|
+
for (const b of bandSteps) {
|
|
281
|
+
const terminator = calculateTerminatorContinuous(now, Math.min(b.depression, 89));
|
|
282
|
+
const bandOpacity = b.opacity - prevOpacity;
|
|
283
|
+
if (bandOpacity < 2e-3 || terminator.sunset.length === 0) {
|
|
284
|
+
prevOpacity = b.opacity;
|
|
285
|
+
continue;
|
|
286
|
+
}
|
|
287
|
+
const poly = buildNightPolygon(terminator);
|
|
288
|
+
if (poly.length < 3) continue;
|
|
290
289
|
[0, 360, -360].forEach((offset) => {
|
|
291
290
|
const shifted = poly.map(([lat, lon]) => [lat, lon + offset]);
|
|
292
291
|
addLayer(L.polygon(shifted, {
|
|
293
292
|
color: "transparent",
|
|
294
|
-
fillColor: "#
|
|
295
|
-
fillOpacity:
|
|
293
|
+
fillColor: "#000a20",
|
|
294
|
+
fillOpacity: bandOpacity,
|
|
296
295
|
interactive: false
|
|
297
296
|
}));
|
|
298
297
|
});
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
[geometric, sunriseGeometric].forEach((line) => {
|
|
302
|
-
[0, 360, -360].forEach((offset) => {
|
|
303
|
-
const shifted = line.map(([lat, lon]) => [lat, lon + offset]);
|
|
304
|
-
addLayer(L.polyline(shifted, {
|
|
305
|
-
color: "rgba(88, 166, 255, 0.18)",
|
|
306
|
-
weight: 1,
|
|
307
|
-
dashArray: "4 6",
|
|
308
|
-
interactive: false
|
|
309
|
-
}));
|
|
310
|
-
});
|
|
311
|
-
});
|
|
298
|
+
prevOpacity = b.opacity;
|
|
299
|
+
}
|
|
312
300
|
}
|
|
313
301
|
if (showGrid) {
|
|
314
302
|
const gridStyle = { color: "rgba(157, 112, 255, 0.12)", weight: 0.5, interactive: false };
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"GroundTrackMapLeaflet.js","sources":["../../../src/react/charts/GroundTrackMapLeaflet.tsx"],"sourcesContent":["/**\r\n * @zendir/ui - GroundTrackMap Leaflet Provider\r\n *\r\n * Leaflet-based 2D world map for ground tracks, stations, and terminator.\r\n * Uses Zendir hybrid theme; all interactions enabled (zoom, scroll, drag, etc.).\r\n * Import leaflet-zendir.css when using this component.\r\n */\r\n\r\nimport React, { useRef, useEffect, useCallback, useMemo, useState } from 'react';\r\nimport { useTheme } from '../theme';\r\nimport type { GroundStation } from '../types';\r\nimport type { SatelliteTrack, GroundStationEnhanced, SROGroundStation, MapPin } from './GroundTrackMap';\r\n\r\n// Optional Leaflet — consumer must install leaflet when using mapProvider=\"leaflet\"\r\nimport L from 'leaflet';\r\nimport 'leaflet/dist/leaflet.css';\r\nimport './leaflet-zendir.css';\r\nimport {\r\n splitPolylineAtAntimeridian,\r\n segmentsWithWorldCopies,\r\n geodesicCirclePoints,\r\n splitRingAtAntimeridian,\r\n} from './groundTrackMapLeafletUtils';\r\nimport {\r\n DEFAULT_TILE,\r\n FALLBACK_TILE,\r\n TILE_ERROR_THRESHOLD,\r\n CARTO_ATTRIBUTION,\r\n OSM_ATTRIBUTION,\r\n} from './groundTrackMapLeafletTiles';\r\n\r\nconst STATUS_COLOR_MAP: Record<string, string> = {\r\n normal: '#56f000',\r\n standby: '#2dccff',\r\n caution: '#fce83a',\r\n serious: '#ffb302',\r\n critical: '#ff3838',\r\n off: '#a4abb6',\r\n};\r\n\r\nconst DEFAULT_TRACK_COLORS = [\r\n '#2dccff', '#3E3CFF', '#9D70FF', '#56f000', '#fce83a', '#ff7849', '#2dd4bf', '#ff3838',\r\n];\r\n\r\n// =============================================================================\r\n// Icon helpers — produce L.DivIcon with inline SVG (no external assets needed)\r\n// =============================================================================\r\n\r\n/** Satellite spacecraft icon with solar panels, matching Astro UXDS style. */\r\nfunction createSatDivIcon(statusColor: string, trackColor: string): L.DivIcon {\r\n // Status badge shape (Astro UXDS spec)\r\n const badge = statusColor === '#56f000'\r\n ? `<circle cx=\"7\" cy=\"7\" r=\"3\" fill=\"${statusColor}\"/>` // normal: filled dot\r\n : statusColor === '#2dccff'\r\n ? `<circle cx=\"7\" cy=\"7\" r=\"2.5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\"/>` // standby: ring\r\n : statusColor === '#fce83a'\r\n ? `<rect x=\"4.5\" y=\"4.5\" width=\"5\" height=\"5\" fill=\"${statusColor}\"/>` // caution: square\r\n : statusColor === '#ffb302'\r\n ? `<polygon points=\"7,3.5 10.5,7 7,10.5 3.5,7\" fill=\"${statusColor}\"/>` // serious: diamond (Astro UXDS)\r\n : statusColor === '#ff3838'\r\n ? `<polygon points=\"7,10 4,4 10,4\" fill=\"${statusColor}\"/>` // critical: triangle down\r\n : `<circle cx=\"7\" cy=\"7\" r=\"2\" fill=\"${statusColor}\"/>`; // off: small dot\r\n\r\n const svg = `<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"34\" height=\"34\" viewBox=\"0 0 34 34\">\r\n <!-- Glow ring -->\r\n <circle cx=\"17\" cy=\"17\" r=\"15\" fill=\"${statusColor}\" fill-opacity=\"0.12\"/>\r\n <!-- Main body circle -->\r\n <circle cx=\"17\" cy=\"17\" r=\"11\" fill=\"#0d1323\" stroke=\"${trackColor}\" stroke-width=\"1.5\"/>\r\n <!-- Left solar panel -->\r\n <rect x=\"2\" y=\"15\" width=\"9\" height=\"4\" rx=\"0.5\" fill=\"${trackColor}\" fill-opacity=\"0.75\"/>\r\n <line x1=\"5.5\" y1=\"15\" x2=\"5.5\" y2=\"19\" stroke=\"${trackColor}\" stroke-width=\"0.5\" stroke-opacity=\"0.45\"/>\r\n <!-- Connector left -->\r\n <line x1=\"11\" y1=\"17\" x2=\"13\" y2=\"17\" stroke=\"${trackColor}\" stroke-width=\"1.2\"/>\r\n <!-- Satellite bus body -->\r\n <rect x=\"13\" y=\"14.5\" width=\"8\" height=\"5\" rx=\"1\" fill=\"${statusColor}\"/>\r\n <!-- Connector right -->\r\n <line x1=\"21\" y1=\"17\" x2=\"23\" y2=\"17\" stroke=\"${trackColor}\" stroke-width=\"1.2\"/>\r\n <!-- Right solar panel -->\r\n <rect x=\"23\" y=\"15\" width=\"9\" height=\"4\" rx=\"0.5\" fill=\"${trackColor}\" fill-opacity=\"0.75\"/>\r\n <line x1=\"26.5\" y1=\"15\" x2=\"26.5\" y2=\"19\" stroke=\"${trackColor}\" stroke-width=\"0.5\" stroke-opacity=\"0.45\"/>\r\n <!-- Status badge (top-left) -->\r\n <circle cx=\"7\" cy=\"7\" r=\"5.5\" fill=\"#0d1323\"/>\r\n ${badge}\r\n </svg>`;\r\n\r\n return L.divIcon({\r\n html: svg,\r\n className: 'zendir-sat-icon',\r\n iconSize: [34, 34] as unknown as L.PointExpression,\r\n iconAnchor: [17, 17] as unknown as L.PointExpression,\r\n tooltipAnchor: [0, -18] as unknown as L.PointExpression,\r\n });\r\n}\r\n\r\n/** Ground station icon — dish/antenna shape, matching Astro UXDS style. */\r\nfunction createStationDivIcon(statusColor: string, type: string, status: string): L.DivIcon {\r\n // Status badge\r\n const badge = status === 'normal'\r\n ? `<circle cx=\"5\" cy=\"5\" r=\"2.5\" fill=\"${statusColor}\"/>`\r\n : status === 'standby'\r\n ? `<circle cx=\"5\" cy=\"5\" r=\"2\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>`\r\n : status === 'caution'\r\n ? `<rect x=\"3\" y=\"3\" width=\"4\" height=\"4\" fill=\"${statusColor}\"/>`\r\n : status === 'serious'\r\n ? `<polygon points=\"5,2.5 7.5,5 5,7.5 2.5,5\" fill=\"${statusColor}\"/>`\r\n : status === 'critical'\r\n ? `<polygon points=\"5,7.5 2.5,2.5 7.5,2.5\" fill=\"${statusColor}\"/>`\r\n : `<circle cx=\"5\" cy=\"5\" r=\"1.5\" fill=\"${statusColor}\"/>`;\r\n\r\n let iconInner = '';\r\n if (type === 'phased-array') {\r\n // Grid pattern (phased array)\r\n iconInner = `\r\n <rect x=\"9\" y=\"7\" width=\"10\" height=\"8\" rx=\"0.5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"12\" y1=\"7\" x2=\"12\" y2=\"15\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"15\" y1=\"7\" x2=\"15\" y2=\"15\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"9\" y1=\"10\" x2=\"19\" y2=\"10\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"9\" y1=\"13\" x2=\"19\" y2=\"13\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"14\" y1=\"15\" x2=\"14\" y2=\"20\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"11\" y1=\"20\" x2=\"17\" y2=\"20\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>`;\r\n } else if (type === 'relay') {\r\n // Relay dish with signal arcs\r\n iconInner = `\r\n <path d=\"M 8 16 a 7 7 0 0 1 7 -7\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <path d=\"M 8 16 m 2 -2 a 5 5 0 0 1 5 -5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\" stroke-linecap=\"round\"/>\r\n <line x1=\"14\" y1=\"16\" x2=\"14\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"11\" y1=\"21\" x2=\"17\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <path d=\"M 16 8 a 3 3 0 0 1 0 4\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\" stroke-linecap=\"round\"/>\r\n <path d=\"M 17.5 6.5 a 5 5 0 0 1 0 7\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1\" stroke-linecap=\"round\"/>`;\r\n } else {\r\n // Default: dish antenna with concentric arcs (Astro UXDS antenna pattern)\r\n iconInner = `\r\n <path d=\"M 14 14 m -7 0 a 7 7 0 0 1 7 -7\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <path d=\"M 14 14 m -5 0 a 5 5 0 0 1 5 -5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <path d=\"M 14 14 m -3 0 a 3 3 0 0 1 3 -3\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <circle cx=\"14\" cy=\"14\" r=\"1.5\" fill=\"${statusColor}\"/>\r\n <line x1=\"14\" y1=\"15\" x2=\"14\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"11\" y1=\"21\" x2=\"17\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>`;\r\n }\r\n\r\n const svg = `<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"28\" height=\"28\" viewBox=\"0 0 28 28\">\r\n <!-- Glow ring -->\r\n <circle cx=\"14\" cy=\"14\" r=\"13\" fill=\"${statusColor}\" fill-opacity=\"0.1\"/>\r\n <!-- Background -->\r\n <circle cx=\"14\" cy=\"14\" r=\"11\" fill=\"#0d1323\" stroke=\"${statusColor}\" stroke-width=\"1.2\" stroke-opacity=\"0.7\"/>\r\n ${iconInner}\r\n <!-- Status badge (top-left) -->\r\n <circle cx=\"5\" cy=\"5\" r=\"5\" fill=\"#0d1323\"/>\r\n ${badge}\r\n </svg>`;\r\n\r\n return L.divIcon({\r\n html: svg,\r\n className: 'zendir-station-icon',\r\n iconSize: [28, 28] as unknown as L.PointExpression,\r\n iconAnchor: [14, 14] as unknown as L.PointExpression,\r\n tooltipAnchor: [0, -14] as unknown as L.PointExpression,\r\n });\r\n}\r\n\r\n/**\r\n * Compute a terminator line with CONTINUOUS (unwrapped) longitudes.\r\n *\r\n * `depressionDeg` offsets the solar zenith angle to produce twilight\r\n * boundaries instead of the geometric sunset line:\r\n * - 0° = geometric sunset/sunrise (day/night boundary)\r\n * - 6° = civil twilight (horizon still visible, outdoor activities possible)\r\n * - 12° = nautical twilight (horizon barely visible, stars appearing)\r\n * - 18° = astronomical twilight (sky fully dark for observation)\r\n *\r\n * These thresholds follow IAU/USNO standard definitions used in celestial\r\n * navigation, Astro UX space ops dashboards, and STK/GMAT mission tools.\r\n */\r\nfunction calculateTerminatorContinuous(\r\n date: Date,\r\n depressionDeg: number = 0,\r\n numPoints: number = 72,\r\n): Array<[number, number]> {\r\n const dayOfYear = Math.floor((date.getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 86400000);\r\n const declination = -23.44 * Math.cos((2 * Math.PI / 365) * (dayOfYear + 10));\r\n const decRad = (declination * Math.PI) / 180;\r\n const hourAngle = ((date.getUTCHours() + date.getUTCMinutes() / 60) / 24) * 360 - 180;\r\n const depRad = (depressionDeg * Math.PI) / 180;\r\n\r\n const points: Array<[number, number]> = [];\r\n let prevLon: number | null = null;\r\n for (let i = 0; i <= numPoints; i++) {\r\n const latRad = ((i / numPoints) * 180 - 90) * (Math.PI / 180);\r\n const cosH = -(Math.sin(depRad) + Math.sin(latRad) * Math.sin(decRad))\r\n / (Math.cos(latRad) * Math.cos(decRad));\r\n let lon: number;\r\n if (cosH < -1) lon = hourAngle + 180;\r\n else if (cosH > 1) lon = hourAngle;\r\n else lon = hourAngle + (Math.acos(cosH) * 180) / Math.PI;\r\n\r\n if (prevLon != null) {\r\n while (lon - prevLon > 180) lon -= 360;\r\n while (lon - prevLon < -180) lon += 360;\r\n }\r\n prevLon = lon;\r\n points.push([(i / numPoints) * 180 - 90, lon]);\r\n }\r\n return points;\r\n}\r\n\r\n/**\r\n * Build a night-side polygon between two terminator lines.\r\n * The outer line is the \"darker\" boundary, the inner is the \"lighter\" one.\r\n * Result traces: outer (lat −90→+90) → inner reversed (lat +90→−90).\r\n */\r\nfunction buildBandPolygon(\r\n outerSunset: Array<[number, number]>,\r\n innerSunset: Array<[number, number]>,\r\n): [number, number][] {\r\n if (!outerSunset.length || !innerSunset.length) return [];\r\n const outerSunrise = outerSunset.map(([lat, lon]) => [lat, lon - 180] as [number, number]);\r\n const innerSunrise = innerSunset.map(([lat, lon]) => [lat, lon - 180] as [number, number]);\r\n // Sunset side: outer → inner reversed\r\n const sunsetBand = [...outerSunset, ...[...innerSunset].reverse()];\r\n // Sunrise side: inner → outer reversed\r\n const sunriseBand = [...innerSunrise, ...[...outerSunrise].reverse()];\r\n return [...sunsetBand, ...sunriseBand];\r\n}\r\n\r\n/**\r\n * Build the core night polygon (inside the astronomical twilight boundary).\r\n */\r\nfunction buildNightPolygon(sunsetTerminator: Array<[number, number]>): [number, number][] {\r\n if (!sunsetTerminator.length) return [];\r\n const sunrise = sunsetTerminator.map(([lat, lon]) => [lat, lon - 180] as [number, number]);\r\n return [...sunsetTerminator, ...[...sunrise].reverse()];\r\n}\r\n\r\nexport type GroundStationMapItem = GroundStation | GroundStationEnhanced | SROGroundStation;\r\n\r\nexport interface GroundTrackMapLeafletProps {\r\n allSatellites: SatelliteTrack[];\r\n groundStations: GroundStationMapItem[];\r\n showTerminator?: boolean;\r\n /** Override the time used for the terminator calculation (defaults to wall-clock now). */\r\n terminatorTime?: Date;\r\n showGrid?: boolean;\r\n showLegend?: boolean;\r\n showEquator?: boolean;\r\n showRecenterButton?: boolean;\r\n defaultCenter?: [number, number];\r\n defaultZoom?: number;\r\n height?: number | string;\r\n width?: number | string;\r\n minHeight?: string;\r\n emptyMessage?: string;\r\n tileUrl?: string;\r\n className?: string;\r\n onSatelliteClick?: (id: string) => void;\r\n onStationClick?: (id: string) => void;\r\n /** Points-of-interest pins */\r\n pins?: MapPin[];\r\n /** Allow adding/editing pins via map clicks */\r\n pinsEditable?: boolean;\r\n onPinAdd?: (pin: Omit<MapPin, 'id'>) => void;\r\n onPinUpdate?: (pin: MapPin) => void;\r\n onPinRemove?: (pinId: string) => void;\r\n}\r\n\r\nexport function GroundTrackMapLeaflet({\r\n allSatellites,\r\n groundStations,\r\n showTerminator = true,\r\n terminatorTime,\r\n showGrid = false,\r\n showLegend = true,\r\n showEquator = false,\r\n showRecenterButton = true,\r\n defaultCenter = [20, 0],\r\n defaultZoom = 2,\r\n height = '100%',\r\n width = '100%',\r\n minHeight = '400px',\r\n emptyMessage = 'No orbital data available',\r\n tileUrl = DEFAULT_TILE,\r\n className = '',\r\n onSatelliteClick,\r\n onStationClick,\r\n pins,\r\n pinsEditable = false,\r\n onPinAdd,\r\n onPinUpdate,\r\n onPinRemove,\r\n}: GroundTrackMapLeafletProps): React.ReactElement {\r\n const { tokens } = useTheme();\r\n const containerRef = useRef<HTMLDivElement>(null);\r\n const mapRef = useRef<L.Map | null>(null);\r\n const tileLayerRef = useRef<L.TileLayer | null>(null);\r\n const overlayGroupRef = useRef<L.LayerGroup | null>(null);\r\n const controlsRef = useRef<L.Control[]>([]);\r\n /** Separate layer group so pin updates don't clear satellite/station layers */\r\n const pinsGroupRef = useRef<L.LayerGroup | null>(null);\r\n const [ready, setReady] = useState(false);\r\n\r\n const clearLayers = useCallback(() => {\r\n const map = mapRef.current;\r\n if (!map) return;\r\n overlayGroupRef.current?.clearLayers();\r\n controlsRef.current.forEach((ctrl) => map.removeControl(ctrl));\r\n controlsRef.current = [];\r\n }, []);\r\n\r\n const addLayer = useCallback((layer: L.Layer) => {\r\n overlayGroupRef.current?.addLayer(layer);\r\n }, []);\r\n\r\n // Create map once on mount (or when tileUrl changes). Intentionally not depending on\r\n // defaultCenter/defaultZoom so animated updates do not recreate the map (no flicker/zoom reset).\r\n useEffect(() => {\r\n // eslint-disable-next-line react-hooks/exhaustive-deps -- defaultCenter/defaultZoom used only for initial view\r\n const el = containerRef.current;\r\n if (!el) return;\r\n\r\n const center: L.LatLngExpression = defaultCenter ?? [20, 0];\r\n const zoom = defaultZoom ?? 2;\r\n\r\n const map = L.map(el, {\r\n center,\r\n zoom,\r\n zoomControl: false,\r\n scrollWheelZoom: true,\r\n doubleClickZoom: true,\r\n touchZoom: true,\r\n boxZoom: true,\r\n keyboard: true,\r\n dragging: true,\r\n attributionControl: true,\r\n maxBounds: [[-90, -540], [90, 540]],\r\n maxBoundsViscosity: 1.0,\r\n });\r\n\r\n L.control.zoom({ position: 'topright' }).addTo(map);\r\n\r\n map.attributionControl.setPrefix('');\r\n\r\n const isCartoTiles = tileUrl.includes('cartocdn');\r\n if (isCartoTiles) {\r\n map.attributionControl.addAttribution(CARTO_ATTRIBUTION);\r\n }\r\n\r\n const tileOptions: L.TileLayerOptions = {\r\n maxZoom: 19,\r\n subdomains: isCartoTiles ? 'abcd' : 'abc',\r\n // crossOrigin avoids tainted-canvas errors when Leaflet tries to read tile pixels\r\n crossOrigin: true,\r\n };\r\n\r\n const tile = L.tileLayer(tileUrl, tileOptions);\r\n tile.addTo(map);\r\n tileLayerRef.current = tile;\r\n\r\n // When primary tiles fail (network, CORS, rate limit), switch to fallback immediately.\r\n // TILE_ERROR_THRESHOLD = 1 so the first error triggers the switch — no visible black period.\r\n let hasSwitchedToFallback = false;\r\n const onTileError = () => {\r\n if (hasSwitchedToFallback) return;\r\n hasSwitchedToFallback = true;\r\n tile.off('tileerror', onTileError);\r\n tile.remove();\r\n if (isCartoTiles) {\r\n map.attributionControl.removeAttribution(CARTO_ATTRIBUTION);\r\n map.attributionControl.addAttribution(OSM_ATTRIBUTION);\r\n }\r\n const fallback = L.tileLayer(FALLBACK_TILE, {\r\n maxZoom: 19,\r\n subdomains: 'abc',\r\n crossOrigin: true,\r\n });\r\n fallback.addTo(map);\r\n tileLayerRef.current = fallback;\r\n };\r\n tile.on('tileerror', onTileError);\r\n\r\n const overlayGroup = L.layerGroup();\r\n overlayGroup.addTo(map);\r\n overlayGroupRef.current = overlayGroup;\r\n\r\n const pinsGroup = L.layerGroup();\r\n pinsGroup.addTo(map);\r\n pinsGroupRef.current = pinsGroup;\r\n\r\n mapRef.current = map;\r\n setReady(true);\r\n\r\n const onSize = () => {\r\n map.invalidateSize({ animate: false });\r\n };\r\n // Immediate RAF + two deferred calls cover: first paint, CSS transitions, flex layout settle\r\n const raf = requestAnimationFrame(onSize);\r\n const t1 = setTimeout(onSize, 150);\r\n const t2 = setTimeout(onSize, 500);\r\n\r\n return () => {\r\n clearTimeout(t1);\r\n clearTimeout(t2);\r\n cancelAnimationFrame(raf);\r\n clearLayers();\r\n pinsGroupRef.current?.clearLayers();\r\n pinsGroupRef.current = null;\r\n tileLayerRef.current = null;\r\n map.remove();\r\n mapRef.current = null;\r\n overlayGroupRef.current = null;\r\n };\r\n }, [tileUrl, clearLayers]);\r\n\r\n useEffect(() => {\r\n if (!ready || !mapRef.current) return;\r\n const map = mapRef.current;\r\n clearLayers();\r\n\r\n if (showTerminator) {\r\n const now = terminatorTime ?? new Date();\r\n // Compute terminator lines at each twilight boundary (IAU/USNO definitions)\r\n const geometric = calculateTerminatorContinuous(now, 0); // day/night boundary\r\n const civil = calculateTerminatorContinuous(now, 6); // civil twilight (6°)\r\n const nautical = calculateTerminatorContinuous(now, 12); // nautical twilight (12°)\r\n const astro = calculateTerminatorContinuous(now, 18); // astronomical twilight (18°)\r\n\r\n // Graduated twilight bands: each zone gets progressively darker\r\n const twilightBands: Array<{ poly: [number, number][]; opacity: number }> = [\r\n // Civil twilight zone (geometric → 6°): faintest tint\r\n { poly: buildBandPolygon(civil, geometric), opacity: 0.08 },\r\n // Nautical twilight zone (6° → 12°): moderate\r\n { poly: buildBandPolygon(nautical, civil), opacity: 0.10 },\r\n // Astronomical twilight zone (12° → 18°): darker\r\n { poly: buildBandPolygon(astro, nautical), opacity: 0.12 },\r\n // Full night (inside 18° boundary): deepest\r\n { poly: buildNightPolygon(astro), opacity: 0.22 },\r\n ];\r\n\r\n twilightBands.forEach(({ poly, opacity }) => {\r\n if (poly.length < 3) return;\r\n [0, 360, -360].forEach((offset) => {\r\n const shifted = poly.map(([lat, lon]) => [lat, lon + offset] as [number, number]);\r\n addLayer(L.polygon(shifted, {\r\n color: 'transparent',\r\n fillColor: '#000a28',\r\n fillOpacity: opacity,\r\n interactive: false,\r\n }));\r\n });\r\n });\r\n\r\n // Subtle boundary line at the geometric terminator only (day/night edge)\r\n const sunriseGeometric = geometric.map(([lat, lon]) => [lat, lon - 180] as [number, number]);\r\n [geometric, sunriseGeometric].forEach((line) => {\r\n [0, 360, -360].forEach((offset) => {\r\n const shifted = line.map(([lat, lon]) => [lat, lon + offset] as [number, number]);\r\n addLayer(L.polyline(shifted, {\r\n color: 'rgba(88, 166, 255, 0.18)',\r\n weight: 1,\r\n dashArray: '4 6',\r\n interactive: false,\r\n }));\r\n });\r\n });\r\n }\r\n\r\n if (showGrid) {\r\n const gridStyle = { color: 'rgba(157, 112, 255, 0.12)', weight: 0.5, interactive: false };\r\n [-360, 0, 360].forEach((offset) => {\r\n for (let lon = -180; lon <= 180; lon += 30) {\r\n const l = lon + offset;\r\n addLayer(L.polyline([[90, l], [-90, l]], gridStyle));\r\n }\r\n for (let lat = -90; lat <= 90; lat += 30) {\r\n addLayer(L.polyline([[lat, -180 + offset], [lat, 180 + offset]], gridStyle));\r\n }\r\n });\r\n }\r\n\r\n if (showEquator) {\r\n [-360, 0, 360].forEach((offset) => {\r\n addLayer(L.polyline([[0, -180 + offset], [0, 180 + offset]], {\r\n color: 'rgba(88, 166, 255, 0.25)',\r\n weight: 1,\r\n dashArray: '6, 4',\r\n interactive: false,\r\n }));\r\n });\r\n }\r\n\r\n groundStations.forEach((gs) => {\r\n const status = ('status' in gs ? gs.status : undefined) ?? 'standby';\r\n const statusColor = STATUS_COLOR_MAP[status] || STATUS_COLOR_MAP.standby;\r\n const stationType = ('type' in gs ? (gs as { type?: string }).type : undefined) ?? 'dish';\r\n const radius = 'coverageRadius' in gs ? gs.coverageRadius : undefined;\r\n const showCoverage = 'showCoverage' in gs ? gs.showCoverage : false;\r\n if (radius != null && radius > 0 && showCoverage) {\r\n const ring = geodesicCirclePoints(gs.latitude, gs.longitude, radius, 64);\r\n const rings = splitRingAtAntimeridian(ring);\r\n [0, 360, -360].forEach((offset) => {\r\n rings.forEach((r) => {\r\n const shifted = r.map(([lat, lon]) => [lat, lon + offset] as [number, number]);\r\n addLayer(L.polygon(shifted, {\r\n color: `${statusColor}55`,\r\n fillColor: statusColor,\r\n fillOpacity: 0.1,\r\n weight: 1,\r\n dashArray: '4, 3',\r\n interactive: false,\r\n }));\r\n });\r\n });\r\n }\r\n\r\n const gsIcon = createStationDivIcon(statusColor, stationType, status);\r\n const marker = L.marker([gs.latitude, gs.longitude], { icon: gsIcon });\r\n addLayer(marker);\r\n const statusLabel = status !== 'standby' ? ` · ${status}` : '';\r\n marker.bindTooltip(\r\n `<strong>${gs.name}</strong>${statusLabel}${'network' in gs && gs.network ? `<br/><span style=\"opacity:0.7\">${gs.network}</span>` : ''}`,\r\n { permanent: false, direction: 'top', className: 'zendir-leaflet-tooltip' }\r\n );\r\n marker.on('click', () => {\r\n onStationClick?.('id' in gs ? String(gs.id) : gs.name);\r\n });\r\n });\r\n\r\n allSatellites.forEach((sat, satIdx) => {\r\n const track = sat.groundTrack;\r\n if (!track || track.length === 0) return;\r\n\r\n const color = sat.color || DEFAULT_TRACK_COLORS[satIdx % DEFAULT_TRACK_COLORS.length];\r\n const futureIdx = sat.futureTrackIndex ?? track.length;\r\n\r\n const pastLatLngs = track\r\n .slice(0, futureIdx)\r\n .map((p) => [p.latitude, p.longitude] as [number, number]);\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(pastLatLngs)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color, weight: 2.5, opacity: 1 }));\r\n }\r\n });\r\n\r\n if (futureIdx < track.length) {\r\n const futureLatLngs = track\r\n .slice(futureIdx)\r\n .map((p) => [p.latitude, p.longitude] as [number, number]);\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(futureLatLngs)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color, weight: 1.5, opacity: 0.5, dashArray: '6, 4' }));\r\n }\r\n });\r\n }\r\n\r\n // Access mask segments (contact passes) — solid green overlay on track\r\n if (sat.accessMask && sat.accessMask.some(Boolean)) {\r\n let segment: [number, number][] = [];\r\n for (let i = 0; i < track.length; i++) {\r\n if (sat.accessMask[i]) {\r\n segment.push([track[i].latitude, track[i].longitude]);\r\n } else if (segment.length >= 2) {\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(segment)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color: STATUS_COLOR_MAP.normal, weight: 4, opacity: 0.9 }));\r\n }\r\n });\r\n segment = [];\r\n } else {\r\n segment = [];\r\n }\r\n }\r\n if (segment.length >= 2) {\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(segment)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color: STATUS_COLOR_MAP.normal, weight: 4, opacity: 0.9 }));\r\n }\r\n });\r\n }\r\n }\r\n\r\n if (sat.passMarkers && sat.passMarkers.length > 0) {\r\n sat.passMarkers.forEach((pm) => {\r\n const isAos = pm.type === 'aos';\r\n const pmColor = isAos ? '#56f000' : '#ff3838';\r\n const pmMarker = L.circleMarker([pm.latitude, pm.longitude], {\r\n radius: 5,\r\n fillColor: pmColor,\r\n color: `${pmColor}cc`,\r\n weight: 1.5,\r\n fillOpacity: 0.9,\r\n });\r\n addLayer(pmMarker);\r\n pmMarker.bindTooltip(`${pm.type.toUpperCase()}${pm.label ? ` – ${pm.label}` : ''}`, {\r\n permanent: false,\r\n direction: 'top',\r\n className: 'zendir-leaflet-tooltip',\r\n });\r\n });\r\n }\r\n\r\n if (sat.showFootprint && sat.footprintRadius) {\r\n const lastIdx = futureIdx > 0 ? Math.min(futureIdx - 1, track.length - 1) : track.length - 1;\r\n const last = track[lastIdx];\r\n const ring = geodesicCirclePoints(last.latitude, last.longitude, sat.footprintRadius, 64);\r\n const rings = splitRingAtAntimeridian(ring);\r\n [0, 360, -360].forEach((offset) => {\r\n rings.forEach((r) => {\r\n const shifted = r.map(([lat, lon]) => [lat, lon + offset] as [number, number]);\r\n addLayer(L.polygon(shifted, {\r\n color: `${color}70`,\r\n fillColor: color,\r\n fillOpacity: 0.1,\r\n weight: 1,\r\n dashArray: '3, 3',\r\n interactive: false,\r\n }));\r\n });\r\n });\r\n }\r\n\r\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, track.length - 1) : track.length - 1;\r\n const current = track[currentIdx];\r\n const statusColor = STATUS_COLOR_MAP[sat.status || 'normal'] || STATUS_COLOR_MAP.normal;\r\n\r\n const satIcon = createSatDivIcon(statusColor, color);\r\n const satMarker = L.marker([current.latitude, current.longitude], { icon: satIcon });\r\n addLayer(satMarker);\r\n satMarker.bindTooltip(\r\n `<strong style=\"color:${color}\">${sat.name}</strong><br/>` +\r\n `Lat ${current.latitude.toFixed(3)}° Lon ${current.longitude.toFixed(3)}°` +\r\n (current.altitude != null ? `<br/>Alt ${current.altitude.toFixed(1)} km` : '') +\r\n (sat.status ? `<br/><span style=\"color:${statusColor}\">${sat.status.toUpperCase()}</span>` : ''),\r\n { permanent: false, direction: 'top', className: 'zendir-leaflet-tooltip' }\r\n );\r\n satMarker.on('click', () => onSatelliteClick?.(sat.id));\r\n });\r\n\r\n if (showLegend && (allSatellites.length > 0 || groundStations.length > 0)) {\r\n const Legend = L.Control.extend({\r\n onAdd() {\r\n const div = L.DomUtil.create('div', 'leaflet-legend-zendir');\r\n div.style.cssText = `\r\n padding: 8px 12px; background: rgba(15,21,32,0.9); border: 1px solid rgba(157,112,255,0.2);\r\n border-radius: 8px; color: #e4e0f0; font-size: 11px; font-family: Roboto, sans-serif;\r\n `;\r\n allSatellites.forEach((s, i) => {\r\n const c = s.color || DEFAULT_TRACK_COLORS[i % DEFAULT_TRACK_COLORS.length];\r\n const row = document.createElement('div');\r\n row.style.cssText = 'display: flex; align-items: center; gap: 6px; margin: 2px 0;';\r\n const swatch = document.createElement('span');\r\n swatch.style.cssText = `width:8px;height:8px;border-radius:50%;background:${c};`;\r\n const label = document.createElement('span');\r\n label.textContent = s.name;\r\n row.appendChild(swatch);\r\n row.appendChild(label);\r\n div.appendChild(row);\r\n });\r\n if (groundStations.length > 0) {\r\n const row = document.createElement('div');\r\n row.style.cssText = 'display: flex; align-items: center; gap: 6px; margin: 2px 0; margin-top: 6px;';\r\n const swatch = document.createElement('span');\r\n swatch.style.cssText = 'width:8px;height:8px;border-radius:50%;background:#2dccff;';\r\n const label = document.createElement('span');\r\n label.textContent = `${groundStations.length} Ground Station${groundStations.length > 1 ? 's' : ''}`;\r\n row.appendChild(swatch);\r\n row.appendChild(label);\r\n div.appendChild(row);\r\n }\r\n return div;\r\n },\r\n });\r\n const legend = new Legend({ position: 'topleft' });\r\n legend.addTo(map);\r\n controlsRef.current.push(legend);\r\n }\r\n\r\n }, [\r\n ready,\r\n allSatellites,\r\n groundStations,\r\n showTerminator,\r\n terminatorTime,\r\n showGrid,\r\n showEquator,\r\n showLegend,\r\n tokens.colors.text.secondary,\r\n addLayer,\r\n clearLayers,\r\n onSatelliteClick,\r\n onStationClick,\r\n ]);\r\n\r\n // ==========================================================================\r\n // Pins layer — separate from overlayGroup so pin updates don't rebuild tracks\r\n // ==========================================================================\r\n\r\n useEffect(() => {\r\n if (!ready || !mapRef.current) return;\r\n const group = pinsGroupRef.current;\r\n if (!group) return;\r\n group.clearLayers();\r\n\r\n if (!pins || pins.length === 0) return;\r\n const accentColor = tokens.colors.accent.primary;\r\n\r\n pins.forEach((pin) => {\r\n const pinColor = pin.color || accentColor;\r\n\r\n const svg = `<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"36\" viewBox=\"0 0 24 36\">\r\n <defs>\r\n <filter id=\"pin-shadow-${pin.id}\" x=\"-30%\" y=\"-10%\" width=\"160%\" height=\"140%\">\r\n <feDropShadow dx=\"0\" dy=\"2\" stdDeviation=\"2\" flood-color=\"#000\" flood-opacity=\"0.35\"/>\r\n </filter>\r\n </defs>\r\n <g filter=\"url(#pin-shadow-${pin.id})\">\r\n <path d=\"M12 2 C6.5 2 2 6.5 2 12 C2 19 12 34 12 34 C12 34 22 19 22 12 C22 6.5 17.5 2 12 2Z\"\r\n fill=\"${pinColor}\" stroke=\"rgba(255,255,255,0.5)\" stroke-width=\"1\"/>\r\n <circle cx=\"12\" cy=\"12\" r=\"4\" fill=\"#0d1323\" stroke=\"rgba(255,255,255,0.7)\" stroke-width=\"0.8\"/>\r\n </g>\r\n </svg>`;\r\n\r\n const icon = L.divIcon({\r\n html: svg,\r\n className: 'zendir-pin-icon',\r\n iconSize: [24, 36] as unknown as L.PointExpression,\r\n iconAnchor: [12, 36] as unknown as L.PointExpression,\r\n tooltipAnchor: [0, -36] as unknown as L.PointExpression,\r\n });\r\n\r\n const marker = L.marker([pin.latitude, pin.longitude], {\r\n icon,\r\n draggable: pinsEditable,\r\n title: pin.label ?? '',\r\n });\r\n\r\n // Rich tooltip\r\n const tooltipLines = [\r\n pin.label ? `<strong>${pin.label}</strong>` : '',\r\n pin.description ? `<div style=\"opacity:0.7;font-size:11px\">${pin.description}</div>` : '',\r\n `<div style=\"font-size:10px;opacity:0.5;font-family:monospace\">${pin.latitude.toFixed(4)}°, ${pin.longitude.toFixed(4)}°</div>`,\r\n pin.createdBy ? `<div style=\"font-size:10px;opacity:0.4;margin-top:2px\">by ${pin.createdBy}</div>` : '',\r\n ].filter(Boolean).join('');\r\n marker.bindTooltip(tooltipLines, {\r\n className: 'zendir-pin-tooltip',\r\n direction: 'top',\r\n offset: [0, -4],\r\n });\r\n\r\n if (pinsEditable) {\r\n // Drag to reposition\r\n marker.on('dragend', () => {\r\n const latlng = marker.getLatLng();\r\n onPinUpdate?.({ ...pin, latitude: latlng.lat, longitude: latlng.lng });\r\n });\r\n\r\n // Right-click to delete\r\n marker.on('contextmenu', (e: L.LeafletEvent) => {\r\n L.DomEvent.stopPropagation(e as unknown as Event);\r\n onPinRemove?.(pin.id);\r\n });\r\n }\r\n\r\n group.addLayer(marker);\r\n });\r\n }, [ready, pins, pinsEditable, onPinUpdate, onPinRemove, tokens.colors.accent.primary]);\r\n\r\n // Click-to-add: listen for clicks on the map background when pinsEditable is true\r\n useEffect(() => {\r\n if (!ready || !mapRef.current || !pinsEditable || !onPinAdd) return;\r\n const map = mapRef.current;\r\n\r\n const handleClick = (e: L.LeafletMouseEvent) => {\r\n // Only fire on map background clicks — ignore clicks that bubbled from markers\r\n if ((e.originalEvent?.target as HTMLElement)?.closest?.('.leaflet-marker-icon')) return;\r\n onPinAdd({\r\n latitude: e.latlng.lat,\r\n longitude: e.latlng.lng,\r\n label: '',\r\n color: tokens.colors.accent.primary,\r\n });\r\n };\r\n\r\n map.on('click', handleClick);\r\n return () => { map.off('click', handleClick); };\r\n }, [ready, pinsEditable, onPinAdd, tokens.colors.accent.primary]);\r\n\r\n const handleRecenter = useCallback(() => {\r\n const map = mapRef.current;\r\n if (!map) return;\r\n const points: L.LatLngExpression[] = [];\r\n allSatellites.forEach((s) => {\r\n s.groundTrack.forEach((p) => points.push([p.latitude, p.longitude]));\r\n });\r\n groundStations.forEach((gs) => points.push([gs.latitude, gs.longitude]));\r\n if (points.length > 0) {\r\n const bounds = L.latLngBounds(points);\r\n map.fitBounds(bounds, { padding: [40, 40], maxZoom: 8 });\r\n } else {\r\n map.setView(defaultCenter, defaultZoom);\r\n }\r\n }, [allSatellites, groundStations, defaultCenter, defaultZoom]);\r\n\r\n const resolvedMinHeight = minHeight || (typeof height === 'number' ? `${height}px` : '400px');\r\n\r\n const isEmpty = allSatellites.length === 0 && groundStations.length === 0 && (!pins || pins.length === 0);\r\n\r\n return (\r\n <div\r\n className={`zendir-ground-track-map zendir-ground-track-map-leaflet${pinsEditable ? ' zendir-pins-editable' : ''} ${className}`}\r\n data-map-backend=\"leaflet\"\r\n style={{\r\n width,\r\n height,\r\n minHeight: resolvedMinHeight,\r\n backgroundColor: tokens.colors.background.base,\r\n borderRadius: 8,\r\n overflow: 'hidden',\r\n position: 'relative',\r\n }}\r\n >\r\n <div\r\n ref={containerRef}\r\n style={{ width: '100%', height: '100%', minHeight: resolvedMinHeight }}\r\n />\r\n {isEmpty && (\r\n <div\r\n style={{\r\n position: 'absolute',\r\n left: '50%',\r\n top: '50%',\r\n transform: 'translate(-50%, -50%)',\r\n color: tokens.colors.text.secondary,\r\n fontSize: 14,\r\n pointerEvents: 'none',\r\n }}\r\n >\r\n {emptyMessage}\r\n </div>\r\n )}\r\n {showRecenterButton && (\r\n <button\r\n type=\"button\"\r\n onClick={handleRecenter}\r\n title=\"Recenter map\"\r\n aria-label=\"Recenter map\"\r\n style={{\r\n position: 'absolute',\r\n top: 8,\r\n right: 50,\r\n zIndex: 1000,\r\n background: 'rgba(24, 29, 46, 0.9)',\r\n border: '1px solid rgba(157, 112, 255, 0.25)',\r\n borderRadius: 6,\r\n color: '#e4e0f0',\r\n cursor: 'pointer',\r\n padding: '6px 10px',\r\n fontSize: 12,\r\n display: 'flex',\r\n alignItems: 'center',\r\n gap: 4,\r\n }}\r\n >\r\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth=\"2\">\r\n <circle cx=\"12\" cy=\"12\" r=\"3\" />\r\n <path d=\"M12 2v4M12 18v4M2 12h4M18 12h4\" />\r\n </svg>\r\n Recenter\r\n </button>\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default GroundTrackMapLeaflet;\r\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"GroundTrackMapLeaflet.js","sources":["../../../src/react/charts/GroundTrackMapLeaflet.tsx"],"sourcesContent":["/**\r\n * @zendir/ui - GroundTrackMap Leaflet Provider\r\n *\r\n * Leaflet-based 2D world map for ground tracks, stations, and terminator.\r\n * Uses Zendir hybrid theme; all interactions enabled (zoom, scroll, drag, etc.).\r\n * Import leaflet-zendir.css when using this component.\r\n */\r\n\r\nimport React, { useRef, useEffect, useCallback, useMemo, useState } from 'react';\r\nimport { useTheme } from '../theme';\r\nimport type { GroundStation } from '../types';\r\nimport type { SatelliteTrack, GroundStationEnhanced, SROGroundStation, MapPin } from './GroundTrackMap';\r\n\r\n// Optional Leaflet — consumer must install leaflet when using mapProvider=\"leaflet\"\r\nimport L from 'leaflet';\r\nimport 'leaflet/dist/leaflet.css';\r\nimport './leaflet-zendir.css';\r\nimport {\r\n splitPolylineAtAntimeridian,\r\n segmentsWithWorldCopies,\r\n geodesicCirclePoints,\r\n splitRingAtAntimeridian,\r\n} from './groundTrackMapLeafletUtils';\r\nimport {\r\n DEFAULT_TILE,\r\n FALLBACK_TILE,\r\n TILE_ERROR_THRESHOLD,\r\n CARTO_ATTRIBUTION,\r\n OSM_ATTRIBUTION,\r\n} from './groundTrackMapLeafletTiles';\r\n\r\nconst STATUS_COLOR_MAP: Record<string, string> = {\r\n normal: '#56f000',\r\n standby: '#2dccff',\r\n caution: '#fce83a',\r\n serious: '#ffb302',\r\n critical: '#ff3838',\r\n off: '#a4abb6',\r\n};\r\n\r\nconst DEFAULT_TRACK_COLORS = [\r\n '#2dccff', '#3E3CFF', '#9D70FF', '#56f000', '#fce83a', '#ff7849', '#2dd4bf', '#ff3838',\r\n];\r\n\r\n// =============================================================================\r\n// Icon helpers — produce L.DivIcon with inline SVG (no external assets needed)\r\n// =============================================================================\r\n\r\n/** Satellite spacecraft icon with solar panels, matching Astro UXDS style. */\r\nfunction createSatDivIcon(statusColor: string, trackColor: string): L.DivIcon {\r\n // Status badge shape (Astro UXDS spec)\r\n const badge = statusColor === '#56f000'\r\n ? `<circle cx=\"7\" cy=\"7\" r=\"3\" fill=\"${statusColor}\"/>` // normal: filled dot\r\n : statusColor === '#2dccff'\r\n ? `<circle cx=\"7\" cy=\"7\" r=\"2.5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\"/>` // standby: ring\r\n : statusColor === '#fce83a'\r\n ? `<rect x=\"4.5\" y=\"4.5\" width=\"5\" height=\"5\" fill=\"${statusColor}\"/>` // caution: square\r\n : statusColor === '#ffb302'\r\n ? `<polygon points=\"7,3.5 10.5,7 7,10.5 3.5,7\" fill=\"${statusColor}\"/>` // serious: diamond (Astro UXDS)\r\n : statusColor === '#ff3838'\r\n ? `<polygon points=\"7,10 4,4 10,4\" fill=\"${statusColor}\"/>` // critical: triangle down\r\n : `<circle cx=\"7\" cy=\"7\" r=\"2\" fill=\"${statusColor}\"/>`; // off: small dot\r\n\r\n const svg = `<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"34\" height=\"34\" viewBox=\"0 0 34 34\">\r\n <!-- Glow ring -->\r\n <circle cx=\"17\" cy=\"17\" r=\"15\" fill=\"${statusColor}\" fill-opacity=\"0.12\"/>\r\n <!-- Main body circle -->\r\n <circle cx=\"17\" cy=\"17\" r=\"11\" fill=\"#0d1323\" stroke=\"${trackColor}\" stroke-width=\"1.5\"/>\r\n <!-- Left solar panel -->\r\n <rect x=\"2\" y=\"15\" width=\"9\" height=\"4\" rx=\"0.5\" fill=\"${trackColor}\" fill-opacity=\"0.75\"/>\r\n <line x1=\"5.5\" y1=\"15\" x2=\"5.5\" y2=\"19\" stroke=\"${trackColor}\" stroke-width=\"0.5\" stroke-opacity=\"0.45\"/>\r\n <!-- Connector left -->\r\n <line x1=\"11\" y1=\"17\" x2=\"13\" y2=\"17\" stroke=\"${trackColor}\" stroke-width=\"1.2\"/>\r\n <!-- Satellite bus body -->\r\n <rect x=\"13\" y=\"14.5\" width=\"8\" height=\"5\" rx=\"1\" fill=\"${statusColor}\"/>\r\n <!-- Connector right -->\r\n <line x1=\"21\" y1=\"17\" x2=\"23\" y2=\"17\" stroke=\"${trackColor}\" stroke-width=\"1.2\"/>\r\n <!-- Right solar panel -->\r\n <rect x=\"23\" y=\"15\" width=\"9\" height=\"4\" rx=\"0.5\" fill=\"${trackColor}\" fill-opacity=\"0.75\"/>\r\n <line x1=\"26.5\" y1=\"15\" x2=\"26.5\" y2=\"19\" stroke=\"${trackColor}\" stroke-width=\"0.5\" stroke-opacity=\"0.45\"/>\r\n <!-- Status badge (top-left) -->\r\n <circle cx=\"7\" cy=\"7\" r=\"5.5\" fill=\"#0d1323\"/>\r\n ${badge}\r\n </svg>`;\r\n\r\n return L.divIcon({\r\n html: svg,\r\n className: 'zendir-sat-icon',\r\n iconSize: [34, 34] as unknown as L.PointExpression,\r\n iconAnchor: [17, 17] as unknown as L.PointExpression,\r\n tooltipAnchor: [0, -18] as unknown as L.PointExpression,\r\n });\r\n}\r\n\r\n/** Ground station icon — dish/antenna shape, matching Astro UXDS style. */\r\nfunction createStationDivIcon(statusColor: string, type: string, status: string): L.DivIcon {\r\n // Status badge\r\n const badge = status === 'normal'\r\n ? `<circle cx=\"5\" cy=\"5\" r=\"2.5\" fill=\"${statusColor}\"/>`\r\n : status === 'standby'\r\n ? `<circle cx=\"5\" cy=\"5\" r=\"2\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>`\r\n : status === 'caution'\r\n ? `<rect x=\"3\" y=\"3\" width=\"4\" height=\"4\" fill=\"${statusColor}\"/>`\r\n : status === 'serious'\r\n ? `<polygon points=\"5,2.5 7.5,5 5,7.5 2.5,5\" fill=\"${statusColor}\"/>`\r\n : status === 'critical'\r\n ? `<polygon points=\"5,7.5 2.5,2.5 7.5,2.5\" fill=\"${statusColor}\"/>`\r\n : `<circle cx=\"5\" cy=\"5\" r=\"1.5\" fill=\"${statusColor}\"/>`;\r\n\r\n let iconInner = '';\r\n if (type === 'phased-array') {\r\n // Grid pattern (phased array)\r\n iconInner = `\r\n <rect x=\"9\" y=\"7\" width=\"10\" height=\"8\" rx=\"0.5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"12\" y1=\"7\" x2=\"12\" y2=\"15\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"15\" y1=\"7\" x2=\"15\" y2=\"15\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"9\" y1=\"10\" x2=\"19\" y2=\"10\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"9\" y1=\"13\" x2=\"19\" y2=\"13\" stroke=\"${statusColor}\" stroke-width=\"0.7\"/>\r\n <line x1=\"14\" y1=\"15\" x2=\"14\" y2=\"20\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"11\" y1=\"20\" x2=\"17\" y2=\"20\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>`;\r\n } else if (type === 'relay') {\r\n // Relay dish with signal arcs\r\n iconInner = `\r\n <path d=\"M 8 16 a 7 7 0 0 1 7 -7\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <path d=\"M 8 16 m 2 -2 a 5 5 0 0 1 5 -5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\" stroke-linecap=\"round\"/>\r\n <line x1=\"14\" y1=\"16\" x2=\"14\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"11\" y1=\"21\" x2=\"17\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <path d=\"M 16 8 a 3 3 0 0 1 0 4\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.2\" stroke-linecap=\"round\"/>\r\n <path d=\"M 17.5 6.5 a 5 5 0 0 1 0 7\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1\" stroke-linecap=\"round\"/>`;\r\n } else {\r\n // Default: dish antenna with concentric arcs (Astro UXDS antenna pattern)\r\n iconInner = `\r\n <path d=\"M 14 14 m -7 0 a 7 7 0 0 1 7 -7\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <path d=\"M 14 14 m -5 0 a 5 5 0 0 1 5 -5\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <path d=\"M 14 14 m -3 0 a 3 3 0 0 1 3 -3\" fill=\"none\" stroke=\"${statusColor}\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\r\n <circle cx=\"14\" cy=\"14\" r=\"1.5\" fill=\"${statusColor}\"/>\r\n <line x1=\"14\" y1=\"15\" x2=\"14\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>\r\n <line x1=\"11\" y1=\"21\" x2=\"17\" y2=\"21\" stroke=\"${statusColor}\" stroke-width=\"1.2\"/>`;\r\n }\r\n\r\n const svg = `<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"28\" height=\"28\" viewBox=\"0 0 28 28\">\r\n <!-- Glow ring -->\r\n <circle cx=\"14\" cy=\"14\" r=\"13\" fill=\"${statusColor}\" fill-opacity=\"0.1\"/>\r\n <!-- Background -->\r\n <circle cx=\"14\" cy=\"14\" r=\"11\" fill=\"#0d1323\" stroke=\"${statusColor}\" stroke-width=\"1.2\" stroke-opacity=\"0.7\"/>\r\n ${iconInner}\r\n <!-- Status badge (top-left) -->\r\n <circle cx=\"5\" cy=\"5\" r=\"5\" fill=\"#0d1323\"/>\r\n ${badge}\r\n </svg>`;\r\n\r\n return L.divIcon({\r\n html: svg,\r\n className: 'zendir-station-icon',\r\n iconSize: [28, 28] as unknown as L.PointExpression,\r\n iconAnchor: [14, 14] as unknown as L.PointExpression,\r\n tooltipAnchor: [0, -14] as unknown as L.PointExpression,\r\n });\r\n}\r\n\r\n/**\r\n * Compute a terminator line with CONTINUOUS (unwrapped) longitudes.\r\n *\r\n * `depressionDeg` offsets the solar zenith angle to produce twilight\r\n * boundaries instead of the geometric sunset line:\r\n * - 0° = geometric sunset/sunrise (day/night boundary)\r\n * - 6° = civil twilight (horizon still visible, outdoor activities possible)\r\n * - 12° = nautical twilight (horizon barely visible, stars appearing)\r\n * - 18° = astronomical twilight (sky fully dark for observation)\r\n *\r\n * These thresholds follow IAU/USNO standard definitions used in celestial\r\n * navigation, Astro UX space ops dashboards, and STK/GMAT mission tools.\r\n */\r\nfunction calculateTerminatorContinuous(\r\n date: Date,\r\n depressionDeg: number = 0,\r\n numPoints: number = 360,\r\n): { sunset: Array<[number, number]>, sunrise: Array<[number, number]> } {\r\n const dayOfYear = Math.floor((date.getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 86400000);\r\n const declination = -23.44 * Math.cos((2 * Math.PI / 365) * (dayOfYear + 10));\r\n const decRad = (declination * Math.PI) / 180;\r\n const hourAngle = ((date.getUTCHours() + date.getUTCMinutes() / 60) / 24) * 360 - 180;\r\n const depRad = (depressionDeg * Math.PI) / 180;\r\n\r\n const sunset: Array<[number, number]> = [];\r\n const sunrise: Array<[number, number]> = [];\r\n\r\n for (let i = 0; i <= numPoints; i++) {\r\n // Avoid exact +/- 90 to prevent longitude singularities at the poles\r\n const lat = -89.999 + (i / numPoints) * 179.998;\r\n const latRad = lat * (Math.PI / 180);\r\n const cosH = -(Math.sin(depRad) + Math.sin(latRad) * Math.sin(decRad))\r\n / (Math.cos(latRad) * Math.cos(decRad));\r\n \r\n if (cosH < -1) {\r\n // All day. No points.\r\n } else if (cosH > 1) {\r\n // All night. Covers all longitudes.\r\n // We map the night as the 360-degree span starting at hourAngle.\r\n sunset.push([lat, hourAngle]);\r\n sunrise.push([lat, hourAngle + 360]);\r\n } else {\r\n // Normal twilight\r\n const H = (Math.acos(cosH) * 180) / Math.PI;\r\n sunset.push([lat, hourAngle + H]);\r\n // Sunrise is 360 - H degrees East of the sun (which is equivalent to West of the sun,\r\n // but keeping it > sunset ensures the night polygon is continuous Eastward).\r\n sunrise.push([lat, hourAngle + 360 - H]);\r\n }\r\n }\r\n \r\n return { sunset, sunrise };\r\n}\r\n\r\n/**\r\n * Build the full night polygon, properly closing the loops around the poles.\r\n * Ensures longitudes are strictly continuous so the map projection doesn't\r\n * draw antimeridian lines across the day side.\r\n */\r\nfunction buildNightPolygon(\r\n terminator: { sunset: Array<[number, number]>, sunrise: Array<[number, number]> }\r\n): [number, number][] {\r\n const { sunset, sunrise } = terminator;\r\n if (!sunset.length || !sunrise.length) return [];\r\n\r\n const poly: [number, number][] = [];\r\n\r\n // Left edge of night (sunset): South to North\r\n poly.push(...sunset);\r\n\r\n // Right edge of night (sunrise): North to South\r\n poly.push(...[...sunrise].reverse());\r\n\r\n // Leaflet will automatically close the polygon by connecting the last point \r\n // (South end of sunrise) back to the first point (South end of sunset).\r\n // Because longitudes are continuous and bounded (sunrise.lon - sunset.lon <= 360),\r\n // this closing line will simply span the bottom or top of the map, perfectly\r\n // enclosing the night area without crossing the day side!\r\n\r\n return poly;\r\n}\r\n\r\nexport type GroundStationMapItem = GroundStation | GroundStationEnhanced | SROGroundStation;\r\n\r\nexport interface GroundTrackMapLeafletProps {\r\n allSatellites: SatelliteTrack[];\r\n groundStations: GroundStationMapItem[];\r\n showTerminator?: boolean;\r\n /** Override the time used for the terminator calculation (defaults to wall-clock now). */\r\n terminatorTime?: Date;\r\n showGrid?: boolean;\r\n showLegend?: boolean;\r\n showEquator?: boolean;\r\n showRecenterButton?: boolean;\r\n defaultCenter?: [number, number];\r\n defaultZoom?: number;\r\n height?: number | string;\r\n width?: number | string;\r\n minHeight?: string;\r\n emptyMessage?: string;\r\n tileUrl?: string;\r\n className?: string;\r\n onSatelliteClick?: (id: string) => void;\r\n onStationClick?: (id: string) => void;\r\n /** Points-of-interest pins */\r\n pins?: MapPin[];\r\n /** Allow adding/editing pins via map clicks */\r\n pinsEditable?: boolean;\r\n onPinAdd?: (pin: Omit<MapPin, 'id'>) => void;\r\n onPinUpdate?: (pin: MapPin) => void;\r\n onPinRemove?: (pinId: string) => void;\r\n}\r\n\r\nexport function GroundTrackMapLeaflet({\r\n allSatellites,\r\n groundStations,\r\n showTerminator = true,\r\n terminatorTime,\r\n showGrid = false,\r\n showLegend = true,\r\n showEquator = false,\r\n showRecenterButton = true,\r\n defaultCenter = [20, 0],\r\n defaultZoom = 2,\r\n height = '100%',\r\n width = '100%',\r\n minHeight = '400px',\r\n emptyMessage = 'No orbital data available',\r\n tileUrl = DEFAULT_TILE,\r\n className = '',\r\n onSatelliteClick,\r\n onStationClick,\r\n pins,\r\n pinsEditable = false,\r\n onPinAdd,\r\n onPinUpdate,\r\n onPinRemove,\r\n}: GroundTrackMapLeafletProps): React.ReactElement {\r\n const { tokens } = useTheme();\r\n const containerRef = useRef<HTMLDivElement>(null);\r\n const mapRef = useRef<L.Map | null>(null);\r\n const tileLayerRef = useRef<L.TileLayer | null>(null);\r\n const overlayGroupRef = useRef<L.LayerGroup | null>(null);\r\n const controlsRef = useRef<L.Control[]>([]);\r\n /** Separate layer group so pin updates don't clear satellite/station layers */\r\n const pinsGroupRef = useRef<L.LayerGroup | null>(null);\r\n const [ready, setReady] = useState(false);\r\n\r\n const clearLayers = useCallback(() => {\r\n const map = mapRef.current;\r\n if (!map) return;\r\n overlayGroupRef.current?.clearLayers();\r\n controlsRef.current.forEach((ctrl) => map.removeControl(ctrl));\r\n controlsRef.current = [];\r\n }, []);\r\n\r\n const addLayer = useCallback((layer: L.Layer) => {\r\n overlayGroupRef.current?.addLayer(layer);\r\n }, []);\r\n\r\n // Create map once on mount (or when tileUrl changes). Intentionally not depending on\r\n // defaultCenter/defaultZoom so animated updates do not recreate the map (no flicker/zoom reset).\r\n useEffect(() => {\r\n // eslint-disable-next-line react-hooks/exhaustive-deps -- defaultCenter/defaultZoom used only for initial view\r\n const el = containerRef.current;\r\n if (!el) return;\r\n\r\n const center: L.LatLngExpression = defaultCenter ?? [20, 0];\r\n const zoom = defaultZoom ?? 2;\r\n\r\n const map = L.map(el, {\r\n center,\r\n zoom,\r\n zoomControl: false,\r\n scrollWheelZoom: true,\r\n doubleClickZoom: true,\r\n touchZoom: true,\r\n boxZoom: true,\r\n keyboard: true,\r\n dragging: true,\r\n attributionControl: true,\r\n maxBounds: [[-90, -540], [90, 540]],\r\n maxBoundsViscosity: 1.0,\r\n });\r\n\r\n L.control.zoom({ position: 'topright' }).addTo(map);\r\n\r\n map.attributionControl.setPrefix('');\r\n\r\n const isCartoTiles = tileUrl.includes('cartocdn');\r\n if (isCartoTiles) {\r\n map.attributionControl.addAttribution(CARTO_ATTRIBUTION);\r\n }\r\n\r\n const tileOptions: L.TileLayerOptions = {\r\n maxZoom: 19,\r\n subdomains: isCartoTiles ? 'abcd' : 'abc',\r\n // crossOrigin avoids tainted-canvas errors when Leaflet tries to read tile pixels\r\n crossOrigin: true,\r\n };\r\n\r\n const tile = L.tileLayer(tileUrl, tileOptions);\r\n tile.addTo(map);\r\n tileLayerRef.current = tile;\r\n\r\n // When primary tiles fail (network, CORS, rate limit), switch to fallback immediately.\r\n // TILE_ERROR_THRESHOLD = 1 so the first error triggers the switch — no visible black period.\r\n let hasSwitchedToFallback = false;\r\n const onTileError = () => {\r\n if (hasSwitchedToFallback) return;\r\n hasSwitchedToFallback = true;\r\n tile.off('tileerror', onTileError);\r\n tile.remove();\r\n if (isCartoTiles) {\r\n map.attributionControl.removeAttribution(CARTO_ATTRIBUTION);\r\n map.attributionControl.addAttribution(OSM_ATTRIBUTION);\r\n }\r\n const fallback = L.tileLayer(FALLBACK_TILE, {\r\n maxZoom: 19,\r\n subdomains: 'abc',\r\n crossOrigin: true,\r\n });\r\n fallback.addTo(map);\r\n tileLayerRef.current = fallback;\r\n };\r\n tile.on('tileerror', onTileError);\r\n\r\n const overlayGroup = L.layerGroup();\r\n overlayGroup.addTo(map);\r\n overlayGroupRef.current = overlayGroup;\r\n\r\n const pinsGroup = L.layerGroup();\r\n pinsGroup.addTo(map);\r\n pinsGroupRef.current = pinsGroup;\r\n\r\n mapRef.current = map;\r\n setReady(true);\r\n\r\n const onSize = () => {\r\n map.invalidateSize({ animate: false });\r\n };\r\n // Immediate RAF + two deferred calls cover: first paint, CSS transitions, flex layout settle\r\n const raf = requestAnimationFrame(onSize);\r\n const t1 = setTimeout(onSize, 150);\r\n const t2 = setTimeout(onSize, 500);\r\n\r\n return () => {\r\n clearTimeout(t1);\r\n clearTimeout(t2);\r\n cancelAnimationFrame(raf);\r\n clearLayers();\r\n pinsGroupRef.current?.clearLayers();\r\n pinsGroupRef.current = null;\r\n tileLayerRef.current = null;\r\n map.remove();\r\n mapRef.current = null;\r\n overlayGroupRef.current = null;\r\n };\r\n }, [tileUrl, clearLayers]);\r\n\r\n useEffect(() => {\r\n if (!ready || !mapRef.current) return;\r\n const map = mapRef.current;\r\n clearLayers();\r\n\r\n if (showTerminator) {\r\n const now = terminatorTime ?? new Date();\r\n\r\n // Build a smooth graduated terminator using many thin bands (every 2°)\r\n // instead of hard-edged zones, creating a seamless day→night fade.\r\n // Depression angles: 0° (geometric sunset) through 24° (deep night).\r\n const BAND_STEP = 2;\r\n const MAX_DEP = 24;\r\n const NIGHT_OPACITY = 0.55;\r\n const bandSteps: Array<{ depression: number; opacity: number }> = [];\r\n for (let d = 0; d <= MAX_DEP; d += BAND_STEP) {\r\n // Smooth quadratic ramp: 0° → 0.0, 18° → ~0.45, 24° → 0.55\r\n const t = Math.min(d / 18, 1);\r\n const opacity = t * t * (NIGHT_OPACITY * 0.82) + (d > 18 ? (d - 18) / 6 * (NIGHT_OPACITY * 0.18) : 0);\r\n bandSteps.push({ depression: d, opacity: Math.min(opacity, NIGHT_OPACITY) });\r\n }\r\n // Final deep-night fill for everything beyond the last band\r\n bandSteps.push({ depression: 90, opacity: NIGHT_OPACITY });\r\n\r\n let prevOpacity = 0;\r\n\r\n // Draw each band as a cumulative overlay — each adds incremental darkening\r\n // By stacking full night polygons with small opacities, we avoid drawing\r\n // complex band-difference polygons, which is much more robust at the poles.\r\n for (const b of bandSteps) {\r\n const terminator = calculateTerminatorContinuous(now, Math.min(b.depression, 89));\r\n const bandOpacity = b.opacity - prevOpacity;\r\n if (bandOpacity < 0.002 || terminator.sunset.length === 0) {\r\n prevOpacity = b.opacity;\r\n continue;\r\n }\r\n\r\n const poly = buildNightPolygon(terminator);\r\n if (poly.length < 3) continue;\r\n\r\n [0, 360, -360].forEach((offset) => {\r\n const shifted = poly.map(([lat, lon]) => [lat, lon + offset] as [number, number]);\r\n addLayer(L.polygon(shifted, {\r\n color: 'transparent',\r\n fillColor: '#000a20',\r\n fillOpacity: bandOpacity,\r\n interactive: false,\r\n }));\r\n });\r\n \r\n prevOpacity = b.opacity;\r\n }\r\n }\r\n\r\n if (showGrid) {\r\n const gridStyle = { color: 'rgba(157, 112, 255, 0.12)', weight: 0.5, interactive: false };\r\n [-360, 0, 360].forEach((offset) => {\r\n for (let lon = -180; lon <= 180; lon += 30) {\r\n const l = lon + offset;\r\n addLayer(L.polyline([[90, l], [-90, l]], gridStyle));\r\n }\r\n for (let lat = -90; lat <= 90; lat += 30) {\r\n addLayer(L.polyline([[lat, -180 + offset], [lat, 180 + offset]], gridStyle));\r\n }\r\n });\r\n }\r\n\r\n if (showEquator) {\r\n [-360, 0, 360].forEach((offset) => {\r\n addLayer(L.polyline([[0, -180 + offset], [0, 180 + offset]], {\r\n color: 'rgba(88, 166, 255, 0.25)',\r\n weight: 1,\r\n dashArray: '6, 4',\r\n interactive: false,\r\n }));\r\n });\r\n }\r\n\r\n groundStations.forEach((gs) => {\r\n const status = ('status' in gs ? gs.status : undefined) ?? 'standby';\r\n const statusColor = STATUS_COLOR_MAP[status] || STATUS_COLOR_MAP.standby;\r\n const stationType = ('type' in gs ? (gs as { type?: string }).type : undefined) ?? 'dish';\r\n const radius = 'coverageRadius' in gs ? gs.coverageRadius : undefined;\r\n const showCoverage = 'showCoverage' in gs ? gs.showCoverage : false;\r\n if (radius != null && radius > 0 && showCoverage) {\r\n const ring = geodesicCirclePoints(gs.latitude, gs.longitude, radius, 64);\r\n const rings = splitRingAtAntimeridian(ring);\r\n [0, 360, -360].forEach((offset) => {\r\n rings.forEach((r) => {\r\n const shifted = r.map(([lat, lon]) => [lat, lon + offset] as [number, number]);\r\n addLayer(L.polygon(shifted, {\r\n color: `${statusColor}55`,\r\n fillColor: statusColor,\r\n fillOpacity: 0.1,\r\n weight: 1,\r\n dashArray: '4, 3',\r\n interactive: false,\r\n }));\r\n });\r\n });\r\n }\r\n\r\n const gsIcon = createStationDivIcon(statusColor, stationType, status);\r\n const marker = L.marker([gs.latitude, gs.longitude], { icon: gsIcon });\r\n addLayer(marker);\r\n const statusLabel = status !== 'standby' ? ` · ${status}` : '';\r\n marker.bindTooltip(\r\n `<strong>${gs.name}</strong>${statusLabel}${'network' in gs && gs.network ? `<br/><span style=\"opacity:0.7\">${gs.network}</span>` : ''}`,\r\n { permanent: false, direction: 'top', className: 'zendir-leaflet-tooltip' }\r\n );\r\n marker.on('click', () => {\r\n onStationClick?.('id' in gs ? String(gs.id) : gs.name);\r\n });\r\n });\r\n\r\n allSatellites.forEach((sat, satIdx) => {\r\n const track = sat.groundTrack;\r\n if (!track || track.length === 0) return;\r\n\r\n const color = sat.color || DEFAULT_TRACK_COLORS[satIdx % DEFAULT_TRACK_COLORS.length];\r\n const futureIdx = sat.futureTrackIndex ?? track.length;\r\n\r\n const pastLatLngs = track\r\n .slice(0, futureIdx)\r\n .map((p) => [p.latitude, p.longitude] as [number, number]);\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(pastLatLngs)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color, weight: 2.5, opacity: 1 }));\r\n }\r\n });\r\n\r\n if (futureIdx < track.length) {\r\n const futureLatLngs = track\r\n .slice(futureIdx)\r\n .map((p) => [p.latitude, p.longitude] as [number, number]);\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(futureLatLngs)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color, weight: 1.5, opacity: 0.5, dashArray: '6, 4' }));\r\n }\r\n });\r\n }\r\n\r\n // Access mask segments (contact passes) — solid green overlay on track\r\n if (sat.accessMask && sat.accessMask.some(Boolean)) {\r\n let segment: [number, number][] = [];\r\n for (let i = 0; i < track.length; i++) {\r\n if (sat.accessMask[i]) {\r\n segment.push([track[i].latitude, track[i].longitude]);\r\n } else if (segment.length >= 2) {\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(segment)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color: STATUS_COLOR_MAP.normal, weight: 4, opacity: 0.9 }));\r\n }\r\n });\r\n segment = [];\r\n } else {\r\n segment = [];\r\n }\r\n }\r\n if (segment.length >= 2) {\r\n segmentsWithWorldCopies(splitPolylineAtAntimeridian(segment)).forEach((seg) => {\r\n if (seg.length >= 2) {\r\n addLayer(L.polyline(seg, { color: STATUS_COLOR_MAP.normal, weight: 4, opacity: 0.9 }));\r\n }\r\n });\r\n }\r\n }\r\n\r\n if (sat.passMarkers && sat.passMarkers.length > 0) {\r\n sat.passMarkers.forEach((pm) => {\r\n const isAos = pm.type === 'aos';\r\n const pmColor = isAos ? '#56f000' : '#ff3838';\r\n const pmMarker = L.circleMarker([pm.latitude, pm.longitude], {\r\n radius: 5,\r\n fillColor: pmColor,\r\n color: `${pmColor}cc`,\r\n weight: 1.5,\r\n fillOpacity: 0.9,\r\n });\r\n addLayer(pmMarker);\r\n pmMarker.bindTooltip(`${pm.type.toUpperCase()}${pm.label ? ` – ${pm.label}` : ''}`, {\r\n permanent: false,\r\n direction: 'top',\r\n className: 'zendir-leaflet-tooltip',\r\n });\r\n });\r\n }\r\n\r\n if (sat.showFootprint && sat.footprintRadius) {\r\n const lastIdx = futureIdx > 0 ? Math.min(futureIdx - 1, track.length - 1) : track.length - 1;\r\n const last = track[lastIdx];\r\n const ring = geodesicCirclePoints(last.latitude, last.longitude, sat.footprintRadius, 64);\r\n const rings = splitRingAtAntimeridian(ring);\r\n [0, 360, -360].forEach((offset) => {\r\n rings.forEach((r) => {\r\n const shifted = r.map(([lat, lon]) => [lat, lon + offset] as [number, number]);\r\n addLayer(L.polygon(shifted, {\r\n color: `${color}70`,\r\n fillColor: color,\r\n fillOpacity: 0.1,\r\n weight: 1,\r\n dashArray: '3, 3',\r\n interactive: false,\r\n }));\r\n });\r\n });\r\n }\r\n\r\n const currentIdx = futureIdx > 0 ? Math.min(futureIdx - 1, track.length - 1) : track.length - 1;\r\n const current = track[currentIdx];\r\n const statusColor = STATUS_COLOR_MAP[sat.status || 'normal'] || STATUS_COLOR_MAP.normal;\r\n\r\n const satIcon = createSatDivIcon(statusColor, color);\r\n const satMarker = L.marker([current.latitude, current.longitude], { icon: satIcon });\r\n addLayer(satMarker);\r\n satMarker.bindTooltip(\r\n `<strong style=\"color:${color}\">${sat.name}</strong><br/>` +\r\n `Lat ${current.latitude.toFixed(3)}° Lon ${current.longitude.toFixed(3)}°` +\r\n (current.altitude != null ? `<br/>Alt ${current.altitude.toFixed(1)} km` : '') +\r\n (sat.status ? `<br/><span style=\"color:${statusColor}\">${sat.status.toUpperCase()}</span>` : ''),\r\n { permanent: false, direction: 'top', className: 'zendir-leaflet-tooltip' }\r\n );\r\n satMarker.on('click', () => onSatelliteClick?.(sat.id));\r\n });\r\n\r\n if (showLegend && (allSatellites.length > 0 || groundStations.length > 0)) {\r\n const Legend = L.Control.extend({\r\n onAdd() {\r\n const div = L.DomUtil.create('div', 'leaflet-legend-zendir');\r\n div.style.cssText = `\r\n padding: 8px 12px; background: rgba(15,21,32,0.9); border: 1px solid rgba(157,112,255,0.2);\r\n border-radius: 8px; color: #e4e0f0; font-size: 11px; font-family: Roboto, sans-serif;\r\n `;\r\n allSatellites.forEach((s, i) => {\r\n const c = s.color || DEFAULT_TRACK_COLORS[i % DEFAULT_TRACK_COLORS.length];\r\n const row = document.createElement('div');\r\n row.style.cssText = 'display: flex; align-items: center; gap: 6px; margin: 2px 0;';\r\n const swatch = document.createElement('span');\r\n swatch.style.cssText = `width:8px;height:8px;border-radius:50%;background:${c};`;\r\n const label = document.createElement('span');\r\n label.textContent = s.name;\r\n row.appendChild(swatch);\r\n row.appendChild(label);\r\n div.appendChild(row);\r\n });\r\n if (groundStations.length > 0) {\r\n const row = document.createElement('div');\r\n row.style.cssText = 'display: flex; align-items: center; gap: 6px; margin: 2px 0; margin-top: 6px;';\r\n const swatch = document.createElement('span');\r\n swatch.style.cssText = 'width:8px;height:8px;border-radius:50%;background:#2dccff;';\r\n const label = document.createElement('span');\r\n label.textContent = `${groundStations.length} Ground Station${groundStations.length > 1 ? 's' : ''}`;\r\n row.appendChild(swatch);\r\n row.appendChild(label);\r\n div.appendChild(row);\r\n }\r\n return div;\r\n },\r\n });\r\n const legend = new Legend({ position: 'topleft' });\r\n legend.addTo(map);\r\n controlsRef.current.push(legend);\r\n }\r\n\r\n }, [\r\n ready,\r\n allSatellites,\r\n groundStations,\r\n showTerminator,\r\n terminatorTime,\r\n showGrid,\r\n showEquator,\r\n showLegend,\r\n tokens.colors.text.secondary,\r\n addLayer,\r\n clearLayers,\r\n onSatelliteClick,\r\n onStationClick,\r\n ]);\r\n\r\n // ==========================================================================\r\n // Pins layer — separate from overlayGroup so pin updates don't rebuild tracks\r\n // ==========================================================================\r\n\r\n useEffect(() => {\r\n if (!ready || !mapRef.current) return;\r\n const group = pinsGroupRef.current;\r\n if (!group) return;\r\n group.clearLayers();\r\n\r\n if (!pins || pins.length === 0) return;\r\n const accentColor = tokens.colors.accent.primary;\r\n\r\n pins.forEach((pin) => {\r\n const pinColor = pin.color || accentColor;\r\n\r\n const svg = `<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"36\" viewBox=\"0 0 24 36\">\r\n <defs>\r\n <filter id=\"pin-shadow-${pin.id}\" x=\"-30%\" y=\"-10%\" width=\"160%\" height=\"140%\">\r\n <feDropShadow dx=\"0\" dy=\"2\" stdDeviation=\"2\" flood-color=\"#000\" flood-opacity=\"0.35\"/>\r\n </filter>\r\n </defs>\r\n <g filter=\"url(#pin-shadow-${pin.id})\">\r\n <path d=\"M12 2 C6.5 2 2 6.5 2 12 C2 19 12 34 12 34 C12 34 22 19 22 12 C22 6.5 17.5 2 12 2Z\"\r\n fill=\"${pinColor}\" stroke=\"rgba(255,255,255,0.5)\" stroke-width=\"1\"/>\r\n <circle cx=\"12\" cy=\"12\" r=\"4\" fill=\"#0d1323\" stroke=\"rgba(255,255,255,0.7)\" stroke-width=\"0.8\"/>\r\n </g>\r\n </svg>`;\r\n\r\n const icon = L.divIcon({\r\n html: svg,\r\n className: 'zendir-pin-icon',\r\n iconSize: [24, 36] as unknown as L.PointExpression,\r\n iconAnchor: [12, 36] as unknown as L.PointExpression,\r\n tooltipAnchor: [0, -36] as unknown as L.PointExpression,\r\n });\r\n\r\n const marker = L.marker([pin.latitude, pin.longitude], {\r\n icon,\r\n draggable: pinsEditable,\r\n title: pin.label ?? '',\r\n });\r\n\r\n // Rich tooltip\r\n const tooltipLines = [\r\n pin.label ? `<strong>${pin.label}</strong>` : '',\r\n pin.description ? `<div style=\"opacity:0.7;font-size:11px\">${pin.description}</div>` : '',\r\n `<div style=\"font-size:10px;opacity:0.5;font-family:monospace\">${pin.latitude.toFixed(4)}°, ${pin.longitude.toFixed(4)}°</div>`,\r\n pin.createdBy ? `<div style=\"font-size:10px;opacity:0.4;margin-top:2px\">by ${pin.createdBy}</div>` : '',\r\n ].filter(Boolean).join('');\r\n marker.bindTooltip(tooltipLines, {\r\n className: 'zendir-pin-tooltip',\r\n direction: 'top',\r\n offset: [0, -4],\r\n });\r\n\r\n if (pinsEditable) {\r\n // Drag to reposition\r\n marker.on('dragend', () => {\r\n const latlng = marker.getLatLng();\r\n onPinUpdate?.({ ...pin, latitude: latlng.lat, longitude: latlng.lng });\r\n });\r\n\r\n // Right-click to delete\r\n marker.on('contextmenu', (e: L.LeafletEvent) => {\r\n L.DomEvent.stopPropagation(e as unknown as Event);\r\n onPinRemove?.(pin.id);\r\n });\r\n }\r\n\r\n group.addLayer(marker);\r\n });\r\n }, [ready, pins, pinsEditable, onPinUpdate, onPinRemove, tokens.colors.accent.primary]);\r\n\r\n // Click-to-add: listen for clicks on the map background when pinsEditable is true\r\n useEffect(() => {\r\n if (!ready || !mapRef.current || !pinsEditable || !onPinAdd) return;\r\n const map = mapRef.current;\r\n\r\n const handleClick = (e: L.LeafletMouseEvent) => {\r\n // Only fire on map background clicks — ignore clicks that bubbled from markers\r\n if ((e.originalEvent?.target as HTMLElement)?.closest?.('.leaflet-marker-icon')) return;\r\n onPinAdd({\r\n latitude: e.latlng.lat,\r\n longitude: e.latlng.lng,\r\n label: '',\r\n color: tokens.colors.accent.primary,\r\n });\r\n };\r\n\r\n map.on('click', handleClick);\r\n return () => { map.off('click', handleClick); };\r\n }, [ready, pinsEditable, onPinAdd, tokens.colors.accent.primary]);\r\n\r\n const handleRecenter = useCallback(() => {\r\n const map = mapRef.current;\r\n if (!map) return;\r\n const points: L.LatLngExpression[] = [];\r\n allSatellites.forEach((s) => {\r\n s.groundTrack.forEach((p) => points.push([p.latitude, p.longitude]));\r\n });\r\n groundStations.forEach((gs) => points.push([gs.latitude, gs.longitude]));\r\n if (points.length > 0) {\r\n const bounds = L.latLngBounds(points);\r\n map.fitBounds(bounds, { padding: [40, 40], maxZoom: 8 });\r\n } else {\r\n map.setView(defaultCenter, defaultZoom);\r\n }\r\n }, [allSatellites, groundStations, defaultCenter, defaultZoom]);\r\n\r\n const resolvedMinHeight = minHeight || (typeof height === 'number' ? `${height}px` : '400px');\r\n\r\n const isEmpty = allSatellites.length === 0 && groundStations.length === 0 && (!pins || pins.length === 0);\r\n\r\n return (\r\n <div\r\n className={`zendir-ground-track-map zendir-ground-track-map-leaflet${pinsEditable ? ' zendir-pins-editable' : ''} ${className}`}\r\n data-map-backend=\"leaflet\"\r\n style={{\r\n width,\r\n height,\r\n minHeight: resolvedMinHeight,\r\n backgroundColor: tokens.colors.background.base,\r\n borderRadius: 8,\r\n overflow: 'hidden',\r\n position: 'relative',\r\n }}\r\n >\r\n <div\r\n ref={containerRef}\r\n style={{ width: '100%', height: '100%', minHeight: resolvedMinHeight }}\r\n />\r\n {isEmpty && (\r\n <div\r\n style={{\r\n position: 'absolute',\r\n left: '50%',\r\n top: '50%',\r\n transform: 'translate(-50%, -50%)',\r\n color: tokens.colors.text.secondary,\r\n fontSize: 14,\r\n pointerEvents: 'none',\r\n }}\r\n >\r\n {emptyMessage}\r\n </div>\r\n )}\r\n {showRecenterButton && (\r\n <button\r\n type=\"button\"\r\n onClick={handleRecenter}\r\n title=\"Recenter map\"\r\n aria-label=\"Recenter map\"\r\n style={{\r\n position: 'absolute',\r\n top: 8,\r\n right: 50,\r\n zIndex: 1000,\r\n background: 'rgba(24, 29, 46, 0.9)',\r\n border: '1px solid rgba(157, 112, 255, 0.25)',\r\n borderRadius: 6,\r\n color: '#e4e0f0',\r\n cursor: 'pointer',\r\n padding: '6px 10px',\r\n fontSize: 12,\r\n display: 'flex',\r\n alignItems: 'center',\r\n gap: 4,\r\n }}\r\n >\r\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth=\"2\">\r\n <circle cx=\"12\" cy=\"12\" r=\"3\" />\r\n <path d=\"M12 2v4M12 18v4M2 12h4M18 12h4\" />\r\n </svg>\r\n Recenter\r\n </button>\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default GroundTrackMapLeaflet;\r\n"],"names":[],"mappings":"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"}
|