feat: Gravity Web Phase 1 - CDP remote control dashboard
This commit is contained in:
252
public/js/app.js
Normal file
252
public/js/app.js
Normal file
@@ -0,0 +1,252 @@
|
||||
/**
|
||||
* Gravity Web — App 초기화 및 WebSocket 관리
|
||||
*/
|
||||
|
||||
(function () {
|
||||
// ─── 컴포넌트 초기화 ──────────────────────────────────
|
||||
const sessionPanel = new SessionPanel();
|
||||
const chatPanel = new ChatPanel();
|
||||
|
||||
let ws = null;
|
||||
let reconnectTimer = null;
|
||||
const WS_URL = `ws://${location.host}/ws`;
|
||||
|
||||
// ─── DOM 레퍼런스 ─────────────────────────────────────
|
||||
const connectionStatus = document.getElementById('connectionStatus');
|
||||
const statusDot = connectionStatus.querySelector('.status-dot');
|
||||
const statusText = connectionStatus.querySelector('.status-text');
|
||||
|
||||
const addSessionBtn = document.getElementById('addSessionBtn');
|
||||
const addSessionModal = document.getElementById('addSessionModal');
|
||||
const closeModal = document.getElementById('closeModal');
|
||||
const cancelModal = document.getElementById('cancelModal');
|
||||
const confirmAddSession = document.getElementById('confirmAddSession');
|
||||
const sessionNameInput = document.getElementById('sessionName');
|
||||
const sessionHostInput = document.getElementById('sessionHost');
|
||||
const sessionPortInput = document.getElementById('sessionPort');
|
||||
|
||||
const screenshotBtn = document.getElementById('screenshotBtn');
|
||||
const reconnectBtn = document.getElementById('reconnectBtn');
|
||||
const screenshotOverlay = document.getElementById('screenshotOverlay');
|
||||
const screenshotImage = document.getElementById('screenshotImage');
|
||||
const closeScreenshot = document.getElementById('closeScreenshot');
|
||||
|
||||
// ─── WebSocket 연결 ───────────────────────────────────
|
||||
function connectWebSocket() {
|
||||
ws = new WebSocket(WS_URL);
|
||||
|
||||
ws.onopen = () => {
|
||||
setConnectionStatus('connected', '서버 연결됨');
|
||||
if (reconnectTimer) {
|
||||
clearTimeout(reconnectTimer);
|
||||
reconnectTimer = null;
|
||||
}
|
||||
};
|
||||
|
||||
ws.onclose = () => {
|
||||
setConnectionStatus('error', '연결 끊김');
|
||||
scheduleReconnect();
|
||||
};
|
||||
|
||||
ws.onerror = () => {
|
||||
setConnectionStatus('error', '연결 오류');
|
||||
};
|
||||
|
||||
ws.onmessage = (event) => {
|
||||
try {
|
||||
const msg = JSON.parse(event.data);
|
||||
handleMessage(msg);
|
||||
} catch (e) {
|
||||
console.error('WS 메시지 파싱 오류:', e);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
function scheduleReconnect() {
|
||||
if (reconnectTimer) return;
|
||||
reconnectTimer = setTimeout(() => {
|
||||
reconnectTimer = null;
|
||||
connectWebSocket();
|
||||
}, 3000);
|
||||
}
|
||||
|
||||
function sendWs(msg) {
|
||||
if (ws && ws.readyState === WebSocket.OPEN) {
|
||||
ws.send(JSON.stringify(msg));
|
||||
}
|
||||
}
|
||||
|
||||
// ─── 서버 메시지 핸들러 ───────────────────────────────
|
||||
function handleMessage(msg) {
|
||||
switch (msg.type) {
|
||||
case 'sessions_list':
|
||||
sessionPanel.update(msg.sessions);
|
||||
break;
|
||||
|
||||
case 'session_switched':
|
||||
sessionPanel.setActive(msg.sessionId);
|
||||
chatPanel.showSession(msg.session);
|
||||
break;
|
||||
|
||||
case 'chat_update':
|
||||
if (msg.sessionId === sessionPanel.activeSessionId) {
|
||||
chatPanel.updateChat(msg.html);
|
||||
}
|
||||
break;
|
||||
|
||||
case 'message_sent':
|
||||
if (!msg.success) {
|
||||
showToast(`전송 실패: ${msg.error}`, 'error');
|
||||
}
|
||||
break;
|
||||
|
||||
case 'screenshot':
|
||||
screenshotImage.src = `data:image/jpeg;base64,${msg.data}`;
|
||||
screenshotOverlay.style.display = 'flex';
|
||||
break;
|
||||
|
||||
case 'error':
|
||||
showToast(msg.message, 'error');
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// ─── 세션 패널 이벤트 ─────────────────────────────────
|
||||
sessionPanel.onSessionSelect = (sessionId) => {
|
||||
sendWs({ type: 'switch_session', sessionId });
|
||||
};
|
||||
|
||||
sessionPanel.onSessionRemove = async (sessionId) => {
|
||||
try {
|
||||
const res = await fetch(`/api/sessions/${sessionId}`, { method: 'DELETE' });
|
||||
if (res.ok) {
|
||||
if (sessionPanel.activeSessionId === sessionId) {
|
||||
chatPanel.showEmpty();
|
||||
sessionPanel.setActive(null);
|
||||
}
|
||||
showToast('세션이 제거되었습니다', 'success');
|
||||
}
|
||||
} catch (e) {
|
||||
showToast('세션 제거 실패', 'error');
|
||||
}
|
||||
};
|
||||
|
||||
// ─── 채팅 패널 이벤트 ─────────────────────────────────
|
||||
chatPanel.onSendMessage = (text) => {
|
||||
sendWs({
|
||||
type: 'send_message',
|
||||
sessionId: sessionPanel.activeSessionId,
|
||||
text,
|
||||
});
|
||||
};
|
||||
|
||||
// ─── 모달 ─────────────────────────────────────────────
|
||||
function showModal() {
|
||||
addSessionModal.style.display = 'flex';
|
||||
sessionNameInput.value = '';
|
||||
sessionHostInput.value = 'localhost';
|
||||
sessionPortInput.value = '9000';
|
||||
setTimeout(() => sessionNameInput.focus(), 100);
|
||||
}
|
||||
|
||||
function hideModal() {
|
||||
addSessionModal.style.display = 'none';
|
||||
}
|
||||
|
||||
async function addSession() {
|
||||
const name = sessionNameInput.value.trim();
|
||||
const host = sessionHostInput.value.trim();
|
||||
const port = parseInt(sessionPortInput.value, 10);
|
||||
|
||||
if (!name) {
|
||||
sessionNameInput.focus();
|
||||
return;
|
||||
}
|
||||
|
||||
hideModal();
|
||||
|
||||
try {
|
||||
const res = await fetch('/api/sessions', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ name, host, cdpPort: port }),
|
||||
});
|
||||
|
||||
const session = await res.json();
|
||||
|
||||
if (session.status === 'connected') {
|
||||
showToast(`${name} 연결 성공`, 'success');
|
||||
} else {
|
||||
showToast(`${name} 연결 실패: ${session.error || '알 수 없는 오류'}`, 'error');
|
||||
}
|
||||
} catch (e) {
|
||||
showToast('세션 추가 실패', 'error');
|
||||
}
|
||||
}
|
||||
|
||||
addSessionBtn.addEventListener('click', showModal);
|
||||
closeModal.addEventListener('click', hideModal);
|
||||
cancelModal.addEventListener('click', hideModal);
|
||||
confirmAddSession.addEventListener('click', addSession);
|
||||
|
||||
// Enter로 모달 확인
|
||||
addSessionModal.addEventListener('keydown', (e) => {
|
||||
if (e.key === 'Enter') addSession();
|
||||
if (e.key === 'Escape') hideModal();
|
||||
});
|
||||
|
||||
// 배경 클릭으로 모달 닫기
|
||||
addSessionModal.addEventListener('click', (e) => {
|
||||
if (e.target === addSessionModal) hideModal();
|
||||
});
|
||||
|
||||
// ─── 스크린샷 ─────────────────────────────────────────
|
||||
screenshotBtn.addEventListener('click', () => {
|
||||
sendWs({ type: 'get_screenshot', sessionId: sessionPanel.activeSessionId });
|
||||
});
|
||||
|
||||
closeScreenshot.addEventListener('click', () => {
|
||||
screenshotOverlay.style.display = 'none';
|
||||
});
|
||||
|
||||
screenshotOverlay.addEventListener('click', (e) => {
|
||||
if (e.target === screenshotOverlay) {
|
||||
screenshotOverlay.style.display = 'none';
|
||||
}
|
||||
});
|
||||
|
||||
// ─── 재연결 ───────────────────────────────────────────
|
||||
reconnectBtn.addEventListener('click', async () => {
|
||||
const id = sessionPanel.activeSessionId;
|
||||
if (!id) return;
|
||||
|
||||
try {
|
||||
const res = await fetch(`/api/sessions/${id}/reconnect`, { method: 'POST' });
|
||||
const result = await res.json();
|
||||
if (result.success) {
|
||||
showToast('재연결 성공', 'success');
|
||||
} else {
|
||||
showToast(`재연결 실패: ${result.error}`, 'error');
|
||||
}
|
||||
} catch {
|
||||
showToast('재연결 실패', 'error');
|
||||
}
|
||||
});
|
||||
|
||||
// ─── 유틸리티 ─────────────────────────────────────────
|
||||
function setConnectionStatus(status, text) {
|
||||
statusDot.className = `status-dot ${status}`;
|
||||
statusText.textContent = text;
|
||||
}
|
||||
|
||||
function showToast(message, type = '') {
|
||||
const toast = document.createElement('div');
|
||||
toast.className = `toast ${type}`;
|
||||
toast.textContent = message;
|
||||
document.body.appendChild(toast);
|
||||
setTimeout(() => toast.remove(), 3000);
|
||||
}
|
||||
|
||||
// ─── 시작 ─────────────────────────────────────────────
|
||||
connectWebSocket();
|
||||
})();
|
||||
Reference in New Issue
Block a user