wing-ops/frontend/src/tabs/prediction/components/RightPanel.tsx
jeonghyo.k 88eb6b121a feat(prediction): OpenDrift 유류 확산 시뮬레이션 통합 + CCTV/관리자 고도화
[예측]
- OpenDrift Python API 서버 및 스크립트 추가 (prediction/opendrift/)
- 시뮬레이션 상태 폴링 훅(useSimulationStatus), 로딩 오버레이 추가
- HydrParticleOverlay: deck.gl 기반 입자 궤적 시각화 레이어
- OilSpillView/LeftPanel/RightPanel: 시뮬레이션 실행·결과 표시 UI 개편
- predictionService/predictionRouter: 시뮬레이션 CRUD 및 상태 관리 API
- simulation.ts: OpenDrift 연동 엔드포인트 확장
- docs/PREDICTION-GUIDE.md: 예측 기능 개발 가이드 추가

[CCTV/항공방제]
- CCTV 오일 감지 GPU 추론 연동 (OilDetectionOverlay, useOilDetection)
- CCTV 안전관리 감지 기능 추가 (선박 출입, 침입 감지)
- oil_inference_server.py: Python GPU 추론 서버

[관리자]
- 관리자 화면 고도화 (사용자/권한/게시판/선박신호 패널)
- AdminSidebar, BoardMgmtPanel, VesselSignalPanel 신규 컴포넌트

[기타]
- DB: 시뮬레이션 결과, 선박보험 시드(1391건), 역할 정리 마이그레이션
- 팀 워크플로우 v1.6.1 동기화

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-09 14:55:46 +09:00

428 lines
15 KiB
TypeScript
Executable File

import { useState } from 'react'
import type { PredictionDetail, SimulationSummary } from '../services/predictionApi'
export function RightPanel({ onOpenBacktrack, onOpenRecalc, onOpenReport, detail, summary }: { onOpenBacktrack?: () => void; onOpenRecalc?: () => void; onOpenReport?: () => void; detail?: PredictionDetail | null; summary?: SimulationSummary | null }) {
const vessel = detail?.vessels?.[0]
const vessel2 = detail?.vessels?.[1]
const spill = detail?.spill
const insurance = vessel?.insuranceData as Array<{ type: string; insurer: string; value: string; currency: string }> | null
const [shipExpanded, setShipExpanded] = useState(false)
const [insuranceExpanded, setInsuranceExpanded] = useState(false)
return (
<div className="w-[300px] min-w-[300px] bg-bg-1 border-l border-border flex flex-col">
{/* Tab Header */}
<div className="flex border-b border-border">
<button className="flex-1 py-3 text-center text-xs font-semibold text-primary-cyan border-b-2 border-primary-cyan transition-all font-korean">
</button>
</div>
{/* Scrollable Content */}
<div className="flex-1 overflow-y-auto p-4 scrollbar-thin scrollbar-thumb-border-light scrollbar-track-transparent">
{/* 표시 정보 제어 */}
<Section title="표시 정보 제어">
<div className="grid grid-cols-2 gap-x-2.5 gap-y-1">
<CheckboxLabel checked>/</CheckboxLabel>
<CheckboxLabel checked>/</CheckboxLabel>
<CheckboxLabel></CheckboxLabel>
<CheckboxLabel></CheckboxLabel>
<CheckboxLabel> </CheckboxLabel>
<CheckboxLabel></CheckboxLabel>
</div>
</Section>
{/* 오염분석 */}
<Section title="오염분석">
<button className="w-full py-2 px-3 bg-gradient-to-r from-purple-500 to-primary-cyan text-white rounded text-[10px] font-bold font-korean">
📐
</button>
</Section>
{/* 오염 종합 상황 */}
<Section title="오염 종합 상황" badge="위험" badgeColor="red">
<div className="grid grid-cols-2 gap-0.5 text-[9px]">
<StatBox label="유출량" value={spill?.volume != null ? spill.volume.toFixed(2) : '—'} unit={spill?.unit || 'kl'} color="var(--t1)" />
<StatBox label="풍화량" value={summary ? summary.weatheredVolume.toFixed(2) : '—'} unit="m³" color="var(--orange)" />
<StatBox label="해상잔존" value={summary ? summary.remainingVolume.toFixed(2) : '—'} unit="m³" color="var(--blue)" />
<StatBox label="연안부착" value={summary ? summary.beachedVolume.toFixed(2) : '—'} unit="m³" color="var(--red)" />
<div className="col-span-2">
<StatBox label="오염해역면적" value={summary ? summary.pollutionArea.toFixed(2) : '—'} unit="km²" color="var(--cyan)" />
</div>
</div>
</Section>
{/* 확산 예측 요약 */}
<Section title="확산 예측 요약 (+18h)" badge="위험" badgeColor="red">
<div className="grid grid-cols-2 gap-0.5">
<PredictionCard value="4.7 km²" label="영향 면적" color="var(--red)" />
<PredictionCard value="6.2 km" label="최대 확산 거리" color="var(--orange)" />
<PredictionCard value="NE 42°" label="주 확산 방향" color="var(--cyan)" />
<PredictionCard value="0.35 m/s" label="확산 속도" color="var(--t1)" />
</div>
</Section>
{/* 유출유 풍화 상태 */}
<Section title="유출유 풍화 상태">
<div className="flex flex-col gap-[3px] text-[8px]">
<ProgressBar label="수면잔류" value={58} color="var(--blue)" />
<ProgressBar label="증발" value={22} color="var(--cyan)" />
<ProgressBar label="분산" value={12} color="var(--green)" />
<ProgressBar label="펜스차단" value={5} color="var(--boom)" />
<ProgressBar label="해안도달" value={3} color="var(--red)" />
</div>
</Section>
{/* 사고 선박 제원 */}
<CollapsibleSection
title="🚢 사고 선박 제원"
expanded={shipExpanded}
onToggle={() => setShipExpanded(!shipExpanded)}
>
<div className="space-y-2">
{/* 선박 카드 */}
<div className="flex items-center gap-2 p-2 border border-[rgba(6,182,212,0.15)] rounded-md" style={{
background: 'linear-gradient(135deg, rgba(6,182,212,0.06), rgba(168,85,247,0.04))',
}}>
<div className="w-[30px] h-[30px] rounded-md flex items-center justify-center text-[15px]" style={{
background: 'rgba(6,182,212,0.1)',
border: '1px solid rgba(6,182,212,0.2)',
}}>🚢</div>
<div className="flex-1">
<div className="text-[11px] font-bold text-text-1 font-korean">{vessel?.vesselNm || '—'}</div>
<div className="text-[8px] text-text-3 font-mono">IMO {vessel?.imoNo || '—'} · {vessel?.vesselTp || '—'}</div>
</div>
<span className="text-[7px] px-2 py-0.5 rounded bg-[rgba(239,68,68,0.12)] text-status-red font-bold"></span>
</div>
{/* 제원 */}
<div className="grid grid-cols-3 gap-1">
<SpecCard value={vessel?.loaM?.toFixed(1) || '—'} label="전장 LOA(m)" color="var(--purple)" />
<SpecCard value={vessel?.breadthM?.toFixed(1) || '—'} label="형폭 B(m)" color="var(--cyan)" />
<SpecCard value={vessel?.draftM?.toFixed(1) || '—'} label="흘수 d(m)" color="var(--green)" />
</div>
<div className="space-y-0.5 text-[9px] font-korean">
<InfoRow label="총톤수(GT)" value={vessel?.gt ? `${vessel.gt.toLocaleString()}` : '—'} />
<InfoRow label="재화중량(DWT)" value={vessel?.dwt ? `${vessel.dwt.toLocaleString()}` : '—'} />
<InfoRow label="건조" value={vessel?.builtYr ? `${vessel.builtYr}` : '—'} />
<InfoRow label="주기관" value={vessel?.engineDc || '—'} mono />
<InfoRow label="선적" value={vessel?.flagCd || '—'} />
<InfoRow label="호출부호" value={vessel?.callsign || '—'} mono />
</div>
{/* 충돌 상대 */}
{vessel2 && (
<div className="p-1.5 bg-[rgba(249,115,22,0.04)] border border-[rgba(249,115,22,0.12)] rounded">
<div className="text-[8px] font-bold text-status-orange font-korean mb-1"> : {vessel2.vesselNm}</div>
<div className="text-[8px] text-text-3 font-korean leading-relaxed">
{vessel2.flagCd} {vessel2.vesselTp} {vessel2.gt ? `${vessel2.gt.toLocaleString()}GT` : ''}
</div>
</div>
)}
</div>
</CollapsibleSection>
{/* 선주 / 보험 */}
<CollapsibleSection
title="🏢 선주 / 보험"
expanded={insuranceExpanded}
onToggle={() => setInsuranceExpanded(!insuranceExpanded)}
>
<div className="space-y-2">
{insurance && insurance.length > 0 ? (
<>
{insurance.filter(ins => ins.type === 'P&I').map((ins, i) => (
<InsuranceCard key={`pi-${i}`} title="🚢 P&I" color="cyan" items={[
{ label: '보험사', value: ins.insurer },
{ label: '한도', value: `${ins.currency} ${ins.value}`, mono: true },
]} />
))}
{insurance.filter(ins => ins.type === 'H&M').map((ins, i) => (
<InsuranceCard key={`hm-${i}`} title="🚢 선체보험 (H&M)" color="cyan" items={[
{ label: '보험사', value: ins.insurer },
{ label: '보험가액', value: `${ins.currency} ${ins.value}`, mono: true },
]} />
))}
{insurance.filter(ins => ins.type === 'CLC').map((ins, i) => (
<InsuranceCard key={`clc-${i}`} title="🛢 유류오염배상 (CLC)" color="red" items={[
{ label: '발급기관', value: ins.insurer },
{ label: 'CLC 한도', value: `${ins.currency} ${ins.value}`, mono: true },
]} />
))}
</>
) : (
<div className="text-[9px] text-text-3 font-korean text-center py-4"> .</div>
)}
</div>
</CollapsibleSection>
</div>
{/* Bottom Action Buttons */}
<div className="flex gap-1.5 p-3 border-t border-border">
<button className="flex-1 py-2 px-1 rounded text-[11px] font-semibold bg-gradient-to-r from-boom to-[#d97706] text-black font-korean">
💾
</button>
<button
onClick={onOpenRecalc}
className="flex-1 py-2 px-1 rounded text-[11px] font-semibold bg-[rgba(249,115,22,0.1)] border border-[rgba(249,115,22,0.3)] text-status-orange font-korean"
>
🔄
</button>
<button
onClick={onOpenReport}
className="flex-1 py-2 px-1 rounded text-[11px] font-semibold bg-gradient-to-r from-primary-cyan to-primary-blue text-white font-korean"
>
📄
</button>
<button
onClick={onOpenBacktrack}
className="flex-1 py-2 px-1 rounded text-[11px] font-semibold bg-[rgba(168,85,247,0.1)] border border-[rgba(168,85,247,0.3)] text-purple-500 font-korean"
>
🔍
</button>
</div>
</div>
)
}
// Helper Components
function Section({
title,
badge,
badgeColor,
children
}: {
title: string
badge?: string
badgeColor?: 'red' | 'green'
children: React.ReactNode
}) {
return (
<div className="bg-bg-3 border border-border rounded-md p-3.5 mb-2.5">
<div className="flex items-center justify-between mb-2.5">
<h4 className="text-xs font-semibold text-text-2 font-korean">{title}</h4>
{badge && (
<span
className={`text-[10px] font-semibold px-2 py-1 rounded-full ${
badgeColor === 'red'
? 'bg-[rgba(239,68,68,0.15)] text-status-red'
: 'bg-[rgba(34,197,94,0.15)] text-status-green'
}`}
>
{badge}
</span>
)}
</div>
{children}
</div>
)
}
function CheckboxLabel({ checked, children }: { checked?: boolean; children: string }) {
return (
<label className="flex items-center gap-1.5 text-[10px] text-text-2 font-korean cursor-pointer">
<input
type="checkbox"
defaultChecked={checked}
className="w-[13px] h-[13px]"
className="accent-[var(--cyan)]"
/>
{children}
</label>
)
}
function StatBox({
label,
value,
unit,
color
}: {
label: string
value: string
unit: string
color: string
}) {
return (
<div className="flex justify-between px-2 py-1 bg-bg-0 border border-border rounded-[3px]">
<span className="text-text-3 font-korean">
{label}
</span>
<span style={{ fontWeight: 700, color, fontFamily: 'var(--fM)' }}>
{value} <small className="font-normal text-text-3">{unit}</small>
</span>
</div>
)
}
function PredictionCard({ value, label, color }: { value: string; label: string; color: string }) {
return (
<div className="text-center py-[5px] px-1 bg-bg-0 border border-border rounded-[3px]">
<div
style={{ color }}
className="text-xs font-extrabold font-mono"
>
{value}
</div>
<div className="text-[7px] text-text-3 font-korean">
{label}
</div>
</div>
)
}
function ProgressBar({ label, value, color }: { label: string; value: number; color: string }) {
return (
<div className="flex items-center gap-1">
<span className="text-text-3 font-korean" style={{ minWidth: '38px' }}>
{label}
</span>
<div
className="flex-1 h-[5px] overflow-hidden rounded-[3px]"
style={{ background: 'rgba(255,255,255,0.05)' }}
>
<div
style={{ height: '100%', width: `${value}%`, background: color, borderRadius: '3px' }}
/>
</div>
<span
style={{ color, minWidth: '28px' }}
className="font-semibold text-right font-mono"
>
{value}%
</span>
</div>
)
}
function CollapsibleSection({
title,
expanded,
onToggle,
children
}: {
title: string
expanded: boolean
onToggle: () => void
children: React.ReactNode
}) {
return (
<div className="bg-bg-3 border border-border rounded-md p-3.5 mb-2.5">
<div
className="flex items-center justify-between cursor-pointer mb-2"
onClick={onToggle}
>
<h4 className="text-xs font-semibold text-text-2 font-korean">{title}</h4>
<span className="text-[10px] text-text-3">{expanded ? '▾' : '▸'}</span>
</div>
{expanded && children}
</div>
)
}
function SpecCard({ value, label, color }: { value: string; label: string; color: string }) {
return (
<div className="text-center py-[6px] px-0.5 bg-bg-0 border border-border rounded-md">
<div
style={{ color }}
className="text-xs font-extrabold font-mono"
>
{value}
</div>
<div className="text-[7px] text-text-3 font-korean">
{label}
</div>
</div>
)
}
function InfoRow({
label,
value,
mono,
valueColor
}: {
label: string
value: string
mono?: boolean
valueColor?: string
}) {
return (
<div className="flex justify-between py-[3px] px-[6px] bg-bg-0 rounded-[3px]">
<span className="text-text-3">{label}</span>
<span
style={{ color: valueColor || 'var(--t1)' }}
className={`font-semibold${mono ? ' font-mono' : ''}`}
>
{value}
</span>
</div>
)
}
function InsuranceCard({
title,
color,
items
}: {
title: string
color: 'cyan' | 'purple' | 'red'
items: Array<{ label: string; value: string; mono?: boolean; valueColor?: string }>
}) {
const colorMap = {
cyan: {
border: 'rgba(6,182,212,0.15)',
bg: 'rgba(6,182,212,0.02)',
text: 'var(--cyan)'
},
purple: {
border: 'rgba(168,85,247,0.15)',
bg: 'rgba(168,85,247,0.02)',
text: 'var(--purple)'
},
red: {
border: 'rgba(239,68,68,0.15)',
bg: 'rgba(239,68,68,0.02)',
text: 'var(--red)'
}
}
const colors = colorMap[color]
return (
<div
className="rounded-md"
style={{
padding: '6px 8px',
border: `1px solid ${colors.border}`,
background: colors.bg
}}
>
<div
style={{ color: colors.text }}
className="text-[8px] font-bold font-korean mb-1"
>
{title}
</div>
<div className="space-y-0.5 text-[8px] font-korean">
{items.map((item, i) => (
<div
key={i}
className="flex justify-between py-0.5 px-1"
>
<span className="text-text-3">{item.label}</span>
<span
style={{ color: item.valueColor || 'var(--t1)' }}
className={`font-semibold${item.mono ? ' font-mono' : ''}`}
>
{item.value}
</span>
</div>
))}
</div>
</div>
)
}