import { useState, useEffect } from 'react' import { useSubMenu } from '@common/hooks/useSubMenu' import { RescueTheoryView } from './RescueTheoryView' import { RescueScenarioView } from './RescueScenarioView' /* ─── Types ─── */ type AccidentType = 'collision' | 'grounding' | 'turning' | 'capsizing' | 'sharpTurn' | 'flooding' | 'sinking' type AnalysisTab = 'rescue' | 'damageStability' | 'longitudinalStrength' /* ─── 사고 유형 데이터 ─── */ const accidentTypes: { id: AccidentType; label: string; eng: string; icon: string; desc: string }[] = [ { id: 'collision', label: '충돌', eng: 'Collision', icon: '💥', desc: '외력에 의한 선체 손상' }, { id: 'grounding', label: '좌초', eng: 'Grounding', icon: '🪨', desc: '해저 접촉/암초 좌초' }, { id: 'turning', label: '선회', eng: 'Turning', icon: '🔄', desc: '급격한 방향전환 사고' }, { id: 'capsizing', label: '전복', eng: 'Capsizing', icon: '🔃', desc: '복원력 상실에 의한 전복' }, { id: 'sharpTurn', label: '급선회', eng: 'Hard Turn', icon: '↩️', desc: '고속 급선회에 의한 경사' }, { id: 'flooding', label: '침수', eng: 'Flooding', icon: '🌊', desc: '해수 유입에 의한 침수' }, { id: 'sinking', label: '침몰', eng: 'Sinking', icon: '⬇️', desc: '부력 상실에 의한 침몰' }, ] const analysisTabs: { id: AnalysisTab; label: string; icon: string }[] = [ { id: 'rescue', label: '구난분석', icon: '🚨' }, { id: 'damageStability', label: '손상복원성', icon: '⚖' }, { id: 'longitudinalStrength', label: '종강도', icon: '📏' }, ] /* ─── 사고 유형별 파라미터 ─── */ const rscTypeData: Record = { collision: { zone: 'PREDICTED\nDAMAGE ZONE', gm: '0.8', list: '15.0', trim: '2.5', buoy: 30, incident: 'M/V SEA GUARDIAN 충돌 사고', survivors: 15, total: 20, missing: 5, oilRate: '100L/min' }, grounding: { zone: 'GROUNDING\nIMPACT AREA', gm: '1.2', list: '8.0', trim: '3.8', buoy: 45, incident: 'M/V OCEAN STAR 좌초 사고', survivors: 22, total: 25, missing: 3, oilRate: '50L/min' }, turning: { zone: 'PREDICTED\nDRIFT PATH', gm: '1.5', list: '12.0', trim: '0.8', buoy: 65, incident: 'M/V PACIFIC WAVE 선회 사고', survivors: 18, total: 18, missing: 0, oilRate: '0L/min' }, capsizing: { zone: 'CAPSIZING\nRISK ZONE', gm: '0.2', list: '45.0', trim: '1.2', buoy: 10, incident: 'M/V GRAND FORTUNE 전복 사고', survivors: 8, total: 15, missing: 7, oilRate: '200L/min' }, sharpTurn: { zone: 'VESSEL\nTURNING CIRCLE', gm: '0.6', list: '25.0', trim: '0.5', buoy: 50, incident: 'M/V BLUE HORIZON 급선회 사고', survivors: 20, total: 20, missing: 0, oilRate: '0L/min' }, flooding: { zone: 'FLOODING\nSPREAD AREA', gm: '0.5', list: '18.0', trim: '4.2', buoy: 20, incident: 'M/V EASTERN GLORY 침수 사고', survivors: 12, total: 16, missing: 4, oilRate: '80L/min' }, sinking: { zone: 'PREDICTED\nSINKING AREA', gm: '0.1', list: '35.0', trim: '6.0', buoy: 5, incident: 'M/V HARMONY 침몰 사고', survivors: 10, total: 22, missing: 12, oilRate: '350L/min' }, } /* ─── 색상 헬퍼 ─── */ function gmColor(v: string) { const n = parseFloat(v); return n < 1.0 ? 'var(--red)' : n < 1.5 ? 'var(--yellow)' : 'var(--green)' } function listColor(v: string) { const n = parseFloat(v); return n > 20 ? 'var(--red)' : n > 10 ? 'var(--yellow)' : 'var(--green)' } function trimColor(v: string) { const n = parseFloat(v); return n > 3 ? 'var(--red)' : n > 1.5 ? 'var(--yellow)' : 'var(--green)' } function buoyColor(v: number) { return v < 30 ? 'var(--red)' : v < 50 ? 'var(--yellow)' : 'var(--green)' } /* ─── 상단 사고 정보바 ─── */ function TopInfoBar({ activeType }: { activeType: AccidentType }) { const d = rscTypeData[activeType] const [clock, setClock] = useState('') useEffect(() => { const tick = () => { const now = new Date() setClock(`${String(now.getHours()).padStart(2, '0')}:${String(now.getMinutes()).padStart(2, '0')}:${String(now.getSeconds()).padStart(2, '0')} KST`) } tick() const iv = setInterval(tick, 1000) return () => clearInterval(iv) }, []) return (
긴급구난지원
사고: {d.incident}
생존자: {d.survivors}/{d.total} 실종: {d.missing} GM: {d.gm}m 횡경사: {d.list}° 유출량: {d.oilRate}
{clock}
👤 Cmdr. KIM
) } /* ─── 왼쪽 패널: 사고유형 + 긴급경고 + CCTV ─── */ function LeftPanel({ activeType, onTypeChange, }: { activeType: AccidentType onTypeChange: (t: AccidentType) => void }) { return (
{/* 사고유형 제목 */}
사고 유형 (INCIDENT TYPE)
{/* 사고유형 버튼 */} {accidentTypes.map(t => ( ))} {/* 긴급 경고 */}
긴급 경고 (CRITICAL ALERTS)
GM 위험 수준 — 전복 위험
승선자 5명 미확인
유류 유출 감지 - 방제 필요
종강도 한계치 88% 접근
{/* CCTV 피드 */}
CCTV FEED #1
● REC
) } /* ─── 중앙 지도 영역 ─── */ function CenterMap({ activeType }: { activeType: AccidentType }) { const d = rscTypeData[activeType] const at = accidentTypes.find(t => t.id === activeType)! return (
{/* 해양 배경 그라데이션 */}
{/* 격자 */}
{/* 해안선 힌트 */}
{/* 사고 해역 정보 */}
사고 해역 정보
위치37°28'N, 126°15'E 수심45m 조류2.5 knots NE
{/* 선박 모형 */}
M/V SEA GUARDIAN
{/* 예측 구역 원 */}
{/* 구역 라벨 */}
{d.zone.replace('\\n', '\n')}
{/* SAR 자산 */}
ETA 5 MIN ─
ETA 15 MIN ─
🚁
6M
🚢
{/* 환경 민감 구역 */}
ENVIRONMENTALLY SENSITIVE
AREA: AQUACULTURE FARM
{/* 지도 컨트롤 */}
{['🗺', '🔍', '📐'].map((ico, i) => ( ))}
{/* 스케일 바 */}
5 km · Zoom: 100%
{/* 사고 유형 표시 */}
현재 사고 유형
{at.icon} {at.label} ({at.eng})
) } /* ─── 오른쪽 분석 패널 ─── */ function RightPanel({ activeAnalysis, onAnalysisChange, activeType, }: { activeAnalysis: AnalysisTab onAnalysisChange: (t: AnalysisTab) => void activeType: AccidentType }) { return (
{/* 분석 탭 */}
{analysisTabs.map(tab => ( ))}
{/* 패널 콘텐츠 */}
{activeAnalysis === 'rescue' && } {activeAnalysis === 'damageStability' && } {activeAnalysis === 'longitudinalStrength' && }
{/* Bottom Action Buttons */}
) } /* ─── 구난 분석 패널 ─── */ function RescuePanel({ activeType }: { activeType: AccidentType }) { const d = rscTypeData[activeType] const at = accidentTypes.find(t => t.id === activeType)! return (
구난 분석 (RESCUE ANALYSIS)
📌 현재 사고유형: {at.label} ({at.eng})
{/* 선박 단면도 SVG */}
VESSEL STATUS
FLOODED IMPACT WL LIST: {d.list}° TRIM: {d.trim}m
{/* 핵심 지표 2×2 */}
20 ? '위험' : parseFloat(d.list) > 10 ? '주의' : '정상'} (기준: 10° 이내)`} subColor={listColor(d.list)} />
잔존 부력 (Reserve Buoyancy)
{d.buoy}%
{/* 긴급 조치 버튼 */}
긴급 조치 (EMERGENCY ACTIONS)
{[ { en: 'BALLAST INJECT', ko: '밸러스트 주입', color: 'var(--red)' }, { en: 'BALLAST DISCHARGE', ko: '밸러스트 배출', color: 'var(--red)' }, { en: 'ENGINE STOP', ko: '기관 정지', color: 'var(--orange)' }, { en: 'ANCHOR DROP', ko: '묘 투하', color: 'var(--orange)' }, ].map((btn, i) => ( ))}
{/* 구난 의사결정 프로세스 */}
구난 의사결정 프로세스 (KRISO Decision Support)
{[ { label: '① 상태평가', color: 'var(--cyan)' }, { label: '② 사례분석', color: 'var(--blue)' }, { label: '③ 장비선정', color: 'var(--t2)' }, ].map((s, i) => ( <>{i > 0 && }
{s.label}
))}
{[ { label: '④ 예인력', color: 'var(--yellow)' }, { label: '⑤ 이초/인양', color: 'var(--orange)' }, { label: '⑥ 유출량', color: 'var(--red)' }, ].map((s, i) => ( <>{i > 0 && }
{s.label}
))}
{/* 유체 정역학 */}
유체 정역학 (Hydrostatics)
{[ { label: '배수량(Δ)', value: '12,450 ton' }, { label: '흘수(Draft)', value: '7.2 m' }, { label: 'KG', value: '8.5 m' }, { label: 'KM (횡)', value: '9.3 m' }, { label: 'TPC', value: '22.8 t/cm' }, { label: 'MTC', value: '185 t·m' }, ].map((r, i) => (
{r.label}
{r.value}
))}
{/* 예인력/이초력 */}
예인력 / 이초력 (Towing & Refloating)
{[ { label: '필요 예인력', value: '285 kN', color: 'var(--yellow)' }, { label: '비상 예인력', value: '420 kN', color: 'var(--red)' }, { label: '이초 반력', value: '1,850 kN', color: 'var(--orange)' }, { label: '인양 안전성', value: 'FAIL', color: 'var(--red)' }, ].map((r, i) => (
{r.label}
{r.value}
))}
※ IMO Salvage Manual / Resistance Increase Ratio 기반 산출
{/* 유출량 추정 */}
유출량 추정 (Oil Outflow Estimation)
{[ { label: '현재 유출률', value: d.oilRate, color: 'var(--orange)' }, { label: '누적 유출량', value: '6.8 kL', color: 'var(--red)' }, { label: '24h 예측', value: '145 kL', color: 'var(--red)' }, ].map((r, i) => (
{r.label}
{r.value}
))}
잔여 연료유: 210 kL | 탱크 잔량: 68% 유출
{/* CBR 사례기반 추론 */}
CBR 유사 사고 사례 (Case-Based Reasoning)
{[ { pct: '94%', name: 'Hebei Spirit (2007)', desc: '태안 · 충돌 · 원유 12,547kL 유출', color: 'var(--cyan)' }, { pct: '87%', name: 'Sea Empress (1996)', desc: '밀포드 · 좌초 · 72,000t 유출', color: 'var(--blue)' }, { pct: '82%', name: 'Rena (2011)', desc: '타우랑가 · 좌초 · 350t HFO 유출', color: 'var(--t2)' }, ].map((c, i) => (
{c.pct}
{c.name}
{c.desc}
))}
{/* 위험도 평가 */}
위험도 평가 — 2차사고 시나리오
{[ { label: '침수 확대 → 전복', level: 'HIGH', color: 'var(--red)' }, { label: '유류 대량 유출 → 해양오염', level: 'HIGH', color: 'var(--orange)' }, { label: '선체 절단 (BM 초과)', level: 'MED', color: 'var(--yellow)' }, { label: '화재/폭발', level: 'LOW', color: 'var(--t2)' }, ].map((r, i) => (
{r.label} {r.level}
))}
{/* 해상 e-Call */}
해상 e-Call (GMDSS / VHF-DSC)
{[ { label: 'MMSI', value: '440123456', color: 'var(--t1)' }, { label: 'Nature of Distress', value: 'COLLISION', color: 'var(--red)' }, { label: 'DSC Alert', value: 'SENT ✓', color: 'var(--green)' }, { label: 'EPIRB', value: 'ACTIVATED ✓', color: 'var(--green)' }, { label: 'VTS 인천', value: 'ACK 10:36', color: 'var(--blue)' }, ].map((r, i) => (
{r.label} {r.value}
))}
) } /* ─── 손상 복원성 패널 ─── */ // eslint-disable-next-line @typescript-eslint/no-unused-vars function DamageStabilityPanel(_props: { activeType: AccidentType }) { return (
손상 복원성 (DAMAGE STABILITY)
사고유형에 따른 손상 후 선체 복원력 분석
IMO A.749(18) / SOLAS Ch.II-1 기준 평가
{/* GZ Curve SVG */}
GZ 복원력 곡선 (Righting Lever Curve)
비손상 GZ 손상 GZ IMO MIN GZ(m) 0.6 0.4 0.2 0 15° 30° 45° 60°
{/* 복원성 지표 */}
침수 구획수
2 구획
#1 선수탱크 / #3 좌현탱크
Margin Line 여유
0.12 m
침수 임계점 임박
{/* SOLAS 판정 */}
⚠ SOLAS 손상복원성 판정: 부적합 (FAIL)
· Area(0~θ_f): 0.028 m·rad (기준 0.015 ✓)
· GZ_max ≥ 0.1m: 0.25m ✓ | θ_max ≥ 15°: 28° ✓
· Margin Line 침수: 0.12m — 추가 침수 시 전복 위험
{/* 좌초시 복원성 */}
좌초시 복원성 (Grounded Stability)
{[ { label: '지반반력', value: '1,240 kN', color: 'var(--yellow)' }, { label: '접촉 면적', value: '12.5 m²', color: 'var(--t1)' }, { label: '제거력(Removal)', value: '1,850 kN', color: 'var(--red)' }, { label: '좌초 GM', value: '0.65 m', color: 'var(--yellow)' }, ].map((r, i) => (
{r.label}
{r.value}
))}
{/* 탱크 상태 */}
탱크 상태 (Tank Volume Status)
{[ { name: '#1 FP Tank', pct: 100, status: '침수', color: 'var(--red)' }, { name: '#3 Port Tank', pct: 85, status: '85%', color: 'var(--red)' }, { name: '#2 DB Tank', pct: 45, status: '45%', color: 'var(--green)' }, { name: 'Ballast #4', pct: 72, status: '72%', color: 'var(--blue)' }, { name: 'Fuel Oil #5', pct: 68, status: '68%', color: 'var(--orange)' }, ].map((t, i) => (
{t.name}
{t.status}
))}
) } /* ─── 종강도 패널 ─── */ // eslint-disable-next-line @typescript-eslint/no-unused-vars function LongStrengthPanel(_props: { activeType: AccidentType }) { return (
종강도 분석 (LONGITUDINAL STRENGTH)
선체 종방향 구조 응력 분석
IACS CSR / Classification Society 기준
{/* 전단력 분포 SVG */}
전단력 분포 (Shear Force Distribution)
LIMIT 손상 후 SF ▲ +SF 0 -SF AP MID FP
{/* 굽힘모멘트 분포 SVG */}
굽힘모멘트 분포 (Bending Moment Distribution)
LIMIT 손상 후 BM ▲ +BM 0 AP MID FP
{/* 종강도 지표 */}
SF 최대/허용 비율
88%
BM 최대/허용 비율
92%
Section Modulus 여유
1.08
Req'd: 1.00 이상 ✓
Hull Girder ULS
1.12
Req'd: 1.10 이상 ⚠
{/* 판정 */}
⚠ 종강도 판정: 주의 (CAUTION)
· SF 최대: 허용치의 88% — 주의 구간
· BM 최대: 허용치의 92% — 경고 구간
· 중앙부 Hogging 모멘트 증가 — 추가 침수 시 선체 절단 위험
· 밸러스트 이동으로 BM 분산 필요
) } /* ─── 하단 바: 이벤트 로그 + 타임라인 ─── */ function BottomBar() { return (
{/* 이벤트 로그 */}
이벤트 로그 / 통신 기록 (EVENT LOG / COMMUNICATION TRANSCRIPT)
{[ { label: '전체', color: 'var(--cyan)' }, { label: '긴급', color: 'var(--orange)' }, { label: '통신', color: 'var(--blue)' }, ].map((f, i) => ( ))}
{[ { time: '10:35', msg: 'SOS FROM M/V SEA GUARDIAN', color: 'var(--red)', bold: true }, { time: '10:35', msg: 'OIL LEAK DETECTED SENSOR #3', color: 'var(--t1)', bold: false }, { time: '10:40', msg: 'CG HELO DISPATCHED', color: 'var(--red)', bold: true }, { time: '10:41', msg: 'GM CRITICAL ALERT — DAMAGE STABILITY FAIL', color: 'var(--red)', bold: true }, { time: '10:42', msg: 'Coast Guard 123 en route — ETA 15 min', color: 'var(--blue)', bold: false }, { time: '10:43', msg: 'LONGITUDINAL STRENGTH WARNING — BM 92% of LIMIT', color: 'var(--yellow)', bold: false }, { time: '10:45', msg: 'BALLAST TRANSFER INITIATED — PORT #2 → STBD #3', color: 'var(--t1)', bold: false }, { time: '10:48', msg: 'LIST INCREASING — 12° → 15°', color: 'var(--yellow)', bold: false }, { time: '10:50', msg: 'RESCUE HELO ON SCENE — HOISTING OPS', color: 'var(--blue)', bold: false }, { time: '10:55', msg: '5 SURVIVORS RECOVERED BY HELO', color: 'var(--green)', bold: false }, ].map((e, i) => (
[{e.time}]{' '} {e.msg}
))}
{/* 타임라인 시뮬레이션 */}
시간대별 시뮬레이션 컨트롤 (TIMELINE SIMULATION CONTROL)
[-6h] [CURRENT] [+6H] [+12H] [+24H]
현재 시간: 10:45 KST
) } /* ─── 공통 메트릭 카드 ─── */ function MetricCard({ label, value, unit, color, sub, subColor }: { label: string; value: string; unit: string; color: string; sub: string; subColor: string }) { return (
{label}
{value} {unit}
{sub}
) } /* ─── 긴급구난 목록 탭 ─── */ function RescueListView() { const listData = [ { status: '대응중', statusColor: 'var(--red)', no: 'RSC-2026-001', vessel: 'M/V SEA GUARDIAN', type: '충돌/좌초', date: '2026-02-17 10:30', location: '37°28\'N 126°15\'E', crew: '15/20' }, { status: '대응중', statusColor: 'var(--orange)', no: 'RSC-2026-002', vessel: 'M/V EASTERN GLORY', type: '침수/전복', date: '2026-02-15 14:20', location: '35°05\'N 129°02\'E', crew: '22/28' }, { status: '종료', statusColor: 'var(--green)', no: 'RSC-2025-048', vessel: 'M/V PACIFIC WAVE', type: '충돌', date: '2025-12-03 08:15', location: '34°45\'N 128°30\'E', crew: '18/18' }, { status: '종료', statusColor: 'var(--green)', no: 'RSC-2025-047', vessel: 'M/V HARMONY', type: '좌초', date: '2025-11-20 22:40', location: '36°12\'N 126°50\'E', crew: '25/25' }, { status: '종료', statusColor: 'var(--green)', no: 'RSC-2025-046', vessel: 'M/V GRAND FORTUNE', type: '침몰', date: '2025-10-08 05:30', location: '33°30\'N 127°15\'E', crew: '10/22' }, ] return (
긴급구난 사고 목록
{['상태', '사고번호', '선박명', '사고유형', '발생일시', '위치', '인명'].map(h => ( ))} {listData.map((r, i) => ( ))}
{h}
{r.status} {r.no} {r.vessel} {r.type} {r.date} {r.location} {r.crew}
) } /* ═══ 메인 RescueView ═══ */ export function RescueView() { const { activeSubTab } = useSubMenu('rescue') const [activeType, setActiveType] = useState('collision') const [activeAnalysis, setActiveAnalysis] = useState('rescue') if (activeSubTab === 'list') { return (
) } if (activeSubTab === 'scenario') { return } if (activeSubTab === 'theory') { return } return (
{/* 상단 사고 정보바 */} {/* 3단 레이아웃: 사고유형 | 지도 | 분석 패널 */}
{/* 하단: 이벤트 로그 + 타임라인 */}
) }