Files
ui/src/features/flights/FlightsPage.tsx
T
Oleksandr Hutsul 274800e508 feat(flights): integrate mission-planner into Flights page
- Port mission-planner flight planning to main app (Tailwind, react-leaflet v5, react-i18next)
- Add FlightMap with click-to-add waypoints, draggable markers, polyline with arrows
- Add FlightParamsPanel with action modes, waypoint list (drag-reorder), altitude chart, wind, JSON import/export
- Add FlightListSidebar with create/delete and telemetry date
- Add collapsible left panel with quick action mode shortcuts
- Add work area / no-go zone drawing via manual mouse events (L.rectangle)
- Add AltitudeDialog and JsonEditorDialog (Tailwind modals)
- Add battery/time/distance calculations per waypoint segment
- Add satellite/classic map toggle and mini-map on point drag
- Add Camera FOV and Communication Addr fields
- Add current position display under location search
- Merge mission-planner translations under flights.planner.*
- Gitignore .superpowers session data
2026-04-17 00:31:24 +03:00

354 lines
17 KiB
TypeScript

import { useState, useEffect, useCallback } from 'react'
import { useTranslation } from 'react-i18next'
import L from 'leaflet'
import { useFlight } from '../../components/FlightContext'
import { api } from '../../api/client'
import { createSSE } from '../../api/sse'
import ConfirmDialog from '../../components/ConfirmDialog'
import FlightListSidebar from './FlightListSidebar'
import FlightParamsPanel from './FlightParamsPanel'
import FlightMap from './FlightMap'
import AltitudeDialog from './AltitudeDialog'
import JsonEditorDialog from './JsonEditorDialog'
import { newGuid, calculateDistance, calculateAllPoints, parseCoordinates, getMockAircraftParams } from './flightPlanUtils'
import { PURPOSES } from './types'
import type { Aircraft, Waypoint } from '../../types'
import type { FlightPoint, CalculatedPointInfo, MapRectangle, ActionMode, WindParams, AircraftParams } from './types'
export default function FlightsPage() {
const { t } = useTranslation()
const { flights, selectedFlight, selectFlight, refreshFlights } = useFlight()
const [mode, setMode] = useState<'params' | 'gps'>('params')
const [deleteId, setDeleteId] = useState<string | null>(null)
const [collapsed, setCollapsed] = useState(false)
const [aircrafts, setAircrafts] = useState<Aircraft[]>([])
const [liveGps, setLiveGps] = useState<{ lat: number; lon: number; satellites: number; status: string } | null>(null)
const [aircraft, setAircraft] = useState<AircraftParams | null>(null)
const [points, setPoints] = useState<FlightPoint[]>([])
const [calculatedPointInfo, setCalculatedPointInfo] = useState<CalculatedPointInfo[]>([])
const [rectangles, setRectangles] = useState<MapRectangle[]>([])
const [actionMode, setActionMode] = useState<ActionMode>('points')
const [wind, setWind] = useState<WindParams>({ direction: 0, speed: 0 })
const [initialAltitude, setInitialAltitude] = useState(1000)
const [currentPosition, setCurrentPosition] = useState<{ lat: number; lng: number }>({ lat: 47.242, lng: 35.024 })
const [locationInput, setLocationInput] = useState('')
const [altDialog, setAltDialog] = useState<{ open: boolean; point: FlightPoint | null; isEdit: boolean }>({ open: false, point: null, isEdit: false })
const [jsonDialog, setJsonDialog] = useState({ open: false, text: '' })
useEffect(() => {
api.get<Aircraft[]>('/api/flights/aircrafts').then(setAircrafts).catch(() => {})
setAircraft(getMockAircraftParams())
navigator.geolocation.getCurrentPosition(
(pos) => setCurrentPosition({ lat: pos.coords.latitude, lng: pos.coords.longitude }),
() => {},
)
}, [])
useEffect(() => {
if (!selectedFlight) { setPoints([]); return }
api.get<Waypoint[]>(`/api/flights/${selectedFlight.id}/waypoints`)
.then(wps => {
setPoints(wps.sort((a, b) => a.order - b.order).map(wp => ({
id: wp.id,
position: { lat: wp.latitude, lng: wp.longitude },
altitude: 300,
meta: [PURPOSES[0].value, PURPOSES[1].value],
})))
})
.catch(() => {})
}, [selectedFlight])
useEffect(() => {
if (!selectedFlight || mode !== 'gps') return
return createSSE<{ lat: number; lon: number; satellites: number; status: string }>(`/api/flights/${selectedFlight.id}/live-gps`, (data) => setLiveGps(data))
}, [selectedFlight, mode])
useEffect(() => {
if (!aircraft || points.length < 2) { setCalculatedPointInfo([{ bat: 100, time: 0 }]); return }
calculateAllPoints(points, aircraft, initialAltitude).then(setCalculatedPointInfo)
}, [points, aircraft, initialAltitude])
const handleCreateFlight = async (name: string) => {
await api.post('/api/flights', { name })
refreshFlights()
}
const handleDeleteFlight = async () => {
if (!deleteId) return
await api.delete(`/api/flights/${deleteId}`)
if (selectedFlight?.id === deleteId) selectFlight(null)
setDeleteId(null)
refreshFlights()
}
const addPoint = useCallback((lat: number, lng: number) => {
const pt: FlightPoint = { id: newGuid(), position: { lat, lng }, altitude: initialAltitude, meta: [PURPOSES[0].value, PURPOSES[1].value] }
setPoints(prev => [...prev, pt])
}, [initialAltitude])
const updatePointPosition = useCallback((index: number, pos: { lat: number; lng: number }) => {
setPoints(prev => prev.map((p, i) => i === index ? { ...p, position: pos } : p))
}, [])
const removePoint = useCallback((id: string) => {
setPoints(prev => prev.filter(p => p.id !== id))
}, [])
const handlePolylineClick = useCallback((e: L.LeafletMouseEvent) => {
const click = e.latlng
let closestIdx = -1, minDist = Infinity
points.forEach((p, i) => {
if (i < points.length - 1) {
const dist = L.LineUtil.pointToSegmentDistance(
L.point(click.lng, click.lat),
L.point(p.position.lng, p.position.lat),
L.point(points[i + 1].position.lng, points[i + 1].position.lat),
)
if (dist < minDist) { minDist = dist; closestIdx = i }
}
})
if (closestIdx !== -1) {
const alt = (points[closestIdx].altitude + points[closestIdx + 1].altitude) / 2
const pt: FlightPoint = { id: newGuid(), position: { lat: click.lat, lng: click.lng }, altitude: alt, meta: [PURPOSES[0].value, PURPOSES[1].value] }
setPoints(prev => { const u = [...prev]; u.splice(closestIdx + 1, 0, pt); return u })
}
}, [points])
const openEditDialog = (point: FlightPoint) => {
setAltDialog({ open: true, point: { ...point }, isEdit: true })
}
const handleAltSubmit = () => {
if (!altDialog.point) return
if (altDialog.isEdit) {
setPoints(prev => prev.map(p => p.id === altDialog.point!.id ? altDialog.point! : p))
} else {
setPoints(prev => [...prev, altDialog.point!])
}
setAltDialog({ open: false, point: null, isEdit: false })
}
const handleEditJson = () => {
const data = {
operational_height: { currentAltitude: initialAltitude },
geofences: { polygons: rectangles.map(r => {
const sw = r.bounds.getSouthWest(), ne = r.bounds.getNorthEast()
return { northWest: { lat: ne.lat, lon: sw.lng }, southEast: { lat: sw.lat, lon: ne.lng }, fence_type: r.color === 'red' ? 'EXCLUSION' : 'INCLUSION' }
})},
action_points: points.map(p => ({ point: { lat: p.position.lat, lon: p.position.lng }, height: p.altitude, action: 'search', action_specific: { targets: p.meta } })),
}
setJsonDialog({ open: true, text: JSON.stringify(data, null, 2) })
}
const handleExport = () => {
const data = {
operational_height: { currentAltitude: initialAltitude },
geofences: { polygons: rectangles.map(r => {
const sw = r.bounds.getSouthWest(), ne = r.bounds.getNorthEast()
return { northWest: { lat: ne.lat, lon: sw.lng }, southEast: { lat: sw.lat, lon: ne.lng }, fence_type: r.color === 'red' ? 'EXCLUSION' : 'INCLUSION' }
})},
action_points: points.map(p => ({ point: { lat: p.position.lat, lon: p.position.lng }, height: p.altitude, action: 'search', action_specific: { targets: p.meta } })),
}
const blob = new Blob([JSON.stringify(data, null, 2)], { type: 'application/json' })
const url = URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = url
a.download = `${selectedFlight?.name ?? 'flight-plan'}.json`
a.click()
URL.revokeObjectURL(url)
}
const handleImport = () => {
const input = document.createElement('input')
input.type = 'file'
input.accept = 'application/json'
input.onchange = async (e) => {
const file = (e.target as HTMLInputElement).files?.[0]
if (!file) return
const text = await file.text()
handleJsonSave(text)
}
input.click()
}
const handleJsonSave = (json: string) => {
try {
const data = JSON.parse(json)
if (data.action_points) {
setPoints(data.action_points.map((ap: Record<string, unknown>) => ({
id: newGuid(),
position: {
lat: (ap.point as Record<string, number>)?.lat ?? (ap.lat as number),
lng: (ap.point as Record<string, number>)?.lon ?? (ap.lon as number),
},
altitude: (ap.height as number) || 300,
meta: (ap.action_specific as Record<string, string[]>)?.targets || [PURPOSES[0].value, PURPOSES[1].value],
})))
}
if (data.geofences?.polygons) {
setRectangles(data.geofences.polygons.map((p: { southEast: { lat: number; lon: number }; northWest: { lat: number; lon: number }; fence_type?: string }) => ({
id: newGuid(),
bounds: L.latLngBounds([p.southEast.lat, p.northWest.lon], [p.northWest.lat, p.southEast.lon]),
color: p.fence_type === 'EXCLUSION' ? 'red' as const : 'green' as const,
})))
}
if (data.operational_height?.currentAltitude) setInitialAltitude(data.operational_height.currentAltitude)
setJsonDialog({ open: false, text: '' })
} catch { alert(t('flights.planner.invalidJson')) }
}
const handleSave = async () => {
if (!selectedFlight) return
const existing = await api.get<Waypoint[]>(`/api/flights/${selectedFlight.id}/waypoints`).catch(() => [] as Waypoint[])
for (const wp of existing) {
await api.delete(`/api/flights/${selectedFlight.id}/waypoints/${wp.id}`).catch(() => {})
}
for (let i = 0; i < points.length; i++) {
await api.post(`/api/flights/${selectedFlight.id}/waypoints`, {
name: `Point ${i + 1}`,
latitude: points[i].position.lat,
longitude: points[i].position.lng,
order: i,
}).catch(() => {})
}
}
const handleLocationSearch = () => {
const coords = parseCoordinates(locationInput)
if (coords) setCurrentPosition(coords)
}
const lastInfo = calculatedPointInfo[calculatedPointInfo.length - 1]
const batteryStatus = !lastInfo ? { label: '', color: '#6c757d' }
: lastInfo.bat > 12 ? { label: t('flights.planner.statusGood'), color: '#40c057' }
: lastInfo.bat > 5 ? { label: t('flights.planner.statusCaution'), color: '#FFFF00' }
: { label: t('flights.planner.statusLow'), color: '#fa5252' }
const totalDist = aircraft && points.length > 1
? points.reduce((acc, p, i) => i === 0 ? 0 : acc + calculateDistance(points[i - 1], p, aircraft.type, initialAltitude, aircraft.downang, aircraft.upang), 0).toFixed(1) + t('flights.planner.km')
: ''
const totalTimeStr = lastInfo && lastInfo.time > 0
? `${Math.floor(lastInfo.time) >= 1 ? Math.floor(lastInfo.time) + t('flights.planner.hour') : ''}${Math.floor((lastInfo.time - Math.floor(lastInfo.time)) * 60)}${t('flights.planner.minutes')}`
: ''
return (
<div className="flex h-full">
<FlightListSidebar
flights={flights}
selectedFlight={selectedFlight}
onSelect={selectFlight}
onCreate={handleCreateFlight}
onDelete={(id) => setDeleteId(id)}
/>
{collapsed ? (
<div className="w-10 bg-az-panel border-r border-az-border flex flex-col items-center py-2 gap-2 shrink-0">
<button onClick={() => setCollapsed(false)} title="Expand"
className="w-8 h-8 rounded border border-az-border text-az-text hover:border-az-orange hover:text-az-orange text-sm">&#187;</button>
<button onClick={() => setActionMode('points')} title={t('flights.planner.addPoints')}
className={`w-8 h-8 rounded border text-sm ${actionMode === 'points' ? 'border-az-orange text-az-orange bg-az-orange/20' : 'border-az-border text-az-text hover:border-az-orange'}`}>&#9679;</button>
<button onClick={() => setActionMode('workArea')} title={t('flights.planner.workArea')}
className={`w-8 h-8 rounded border text-az-green text-sm ${actionMode === 'workArea' ? 'border-az-green bg-az-green/20' : 'border-az-border hover:border-az-green'}`}>&#9635;</button>
<button onClick={() => setActionMode('prohibitedArea')} title={t('flights.planner.prohibitedArea')}
className={`w-8 h-8 rounded border text-az-red text-sm ${actionMode === 'prohibitedArea' ? 'border-az-red bg-az-red/20' : 'border-az-border hover:border-az-red'}`}>&#9635;</button>
</div>
) : (
<div className="w-80 bg-az-panel border-r border-az-border flex flex-col shrink-0">
<div className="flex border-b border-az-border items-stretch">
<button onClick={() => setMode('params')}
className={`flex-1 py-1.5 text-[10px] ${mode === 'params' ? 'bg-az-bg text-white' : 'text-az-muted'}`}>
{t('flights.params')}
</button>
<button onClick={() => setMode('gps')}
className={`flex-1 py-1.5 text-[10px] ${mode === 'gps' ? 'bg-az-bg text-white' : 'text-az-muted'}`}>
{t('flights.gpsDenied')}
</button>
<button onClick={() => setCollapsed(true)} title="Collapse"
className="px-2 text-az-muted hover:text-az-orange text-sm border-l border-az-border">&#171;</button>
</div>
{mode === 'params' && (
<FlightParamsPanel
points={points} calculatedPointInfo={calculatedPointInfo} aircrafts={aircrafts}
initialAltitude={initialAltitude} actionMode={actionMode}
wind={wind} locationInput={locationInput} currentPosition={currentPosition}
totalDistance={totalDist} totalTime={totalTimeStr} batteryStatus={batteryStatus}
onInitialAltitudeChange={setInitialAltitude} onActionModeChange={setActionMode}
onWindChange={setWind} onLocationInputChange={setLocationInput}
onLocationSearch={handleLocationSearch}
onReorderPoints={setPoints} onEditPoint={openEditDialog} onRemovePoint={removePoint}
onSave={handleSave} onUpload={handleImport} onEditJson={handleEditJson} onExport={handleExport}
/>
)}
{mode === 'gps' && (
<div className="p-2 space-y-2 text-xs">
<div>
<label className="text-az-muted block mb-1">{t('flights.liveGps')}</label>
{liveGps ? (
<div className="bg-az-bg rounded p-1.5 space-y-0.5">
<div className="text-az-text">Status: <span className="text-az-green">{liveGps.status}</span></div>
<div className="text-az-text">Lat: {liveGps.lat.toFixed(6)}</div>
<div className="text-az-text">Lon: {liveGps.lon.toFixed(6)}</div>
<div className="text-az-text">Sats: {liveGps.satellites}</div>
</div>
) : (
<div className="text-az-muted">Waiting for GPS signal...</div>
)}
</div>
<button onClick={() => setMode('params')} className="text-az-orange text-xs">
{t('flights.back')}
</button>
</div>
)}
</div>
)}
<FlightMap
points={points} calculatedPointInfo={calculatedPointInfo}
currentPosition={currentPosition} rectangles={rectangles} setRectangles={setRectangles}
rectangleColor={actionMode === 'workArea' ? 'green' : 'red'}
actionMode={actionMode}
onAddPoint={addPoint} onUpdatePoint={updatePointPosition} onRemovePoint={removePoint}
onAltitudeChange={(i, alt) => setPoints(prev => prev.map((p, idx) => idx === i ? { ...p, altitude: alt } : p))}
onMetaChange={(i, meta) => setPoints(prev => prev.map((p, idx) => idx === i ? { ...p, meta } : p))}
onPolylineClick={handlePolylineClick}
onPositionChange={setCurrentPosition}
onMapMove={() => {}}
/>
<AltitudeDialog
open={altDialog.open}
isEditMode={altDialog.isEdit}
latitude={altDialog.point?.position.lat ?? 0}
longitude={altDialog.point?.position.lng ?? 0}
altitude={altDialog.point?.altitude ?? 300}
meta={altDialog.point?.meta ?? []}
onLatitudeChange={v => setAltDialog(prev => ({ ...prev, point: prev.point ? { ...prev.point, position: { ...prev.point.position, lat: v } } : null }))}
onLongitudeChange={v => setAltDialog(prev => ({ ...prev, point: prev.point ? { ...prev.point, position: { ...prev.point.position, lng: v } } : null }))}
onAltitudeChange={v => setAltDialog(prev => ({ ...prev, point: prev.point ? { ...prev.point, altitude: v } : null }))}
onMetaChange={v => setAltDialog(prev => ({ ...prev, point: prev.point ? { ...prev.point, meta: v } : null }))}
onSubmit={handleAltSubmit}
onClose={() => setAltDialog({ open: false, point: null, isEdit: false })}
/>
<JsonEditorDialog
open={jsonDialog.open}
jsonText={jsonDialog.text}
onClose={() => setJsonDialog({ open: false, text: '' })}
onSave={handleJsonSave}
/>
<ConfirmDialog
open={!!deleteId}
title={t('common.delete')}
message="Delete this flight and all its data?"
onConfirm={handleDeleteFlight}
onCancel={() => setDeleteId(null)}
/>
</div>
)
}