- 768px 이하에서 대시보드 세로 스택 + 에이전트 카드 아코디언 토글 - waiting/알림 있을 때 자동 펼침 및 좌측 강조 바 - 픽셀 오피스 캔버스 모바일 높이 140px로 축소 후 상단 배치 - 최근 작업 시간: completed_at 우선 + 오늘/어제/MM-DD HH:MM 포맷 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
204 lines
7.4 KiB
JavaScript
204 lines
7.4 KiB
JavaScript
import React, { useState, useEffect } from 'react';
|
|
import { getAgentTasks, getAgentTokenUsage } from '../../../api';
|
|
|
|
const STATUS_BADGE = {
|
|
pending: { label: '대기', bg: '#92400e' },
|
|
approved: { label: '승인됨', bg: '#1e40af' },
|
|
working: { label: '진행중', bg: '#3730a3' },
|
|
succeeded: { label: '완료', bg: '#065f46' },
|
|
failed: { label: '실패', bg: '#7f1d1d' },
|
|
rejected: { label: '거절됨', bg: '#9a3412' },
|
|
};
|
|
|
|
const AGENT_COMMANDS = {
|
|
stock: [
|
|
{ action: 'fetch_news', label: '뉴스 수집', icon: '📰' },
|
|
{ action: 'list_alerts', label: '알람 목록', icon: '🔔' },
|
|
{ action: 'test_telegram', label: 'TG 테스트', icon: '📨' },
|
|
],
|
|
music: [
|
|
{ action: 'compose', label: '작곡 시작', icon: '🎵', needsInput: true },
|
|
{ action: 'credits', label: '크레딧', icon: '💳' },
|
|
],
|
|
blog: [
|
|
{ action: 'research', label: '키워드 리서치', icon: '🔍', needsInput: true },
|
|
{ action: 'list_trend_keywords', label: '트렌드 목록', icon: '📋' },
|
|
],
|
|
realestate: [
|
|
{ action: 'fetch_matches', label: '매칭 리포트', icon: '🏢' },
|
|
{ action: 'dashboard', label: '대시보드', icon: '📊' },
|
|
],
|
|
};
|
|
|
|
const AgentColumn = ({ agentId, meta, agentState, notification, onCommand, onApproval, onClearNotification }) => {
|
|
const [tasks, setTasks] = useState([]);
|
|
const [input, setInput] = useState('');
|
|
const [activeCommand, setActiveCommand] = useState(null);
|
|
const [tokenUsage, setTokenUsage] = useState(null);
|
|
const [expanded, setExpanded] = useState(false);
|
|
|
|
const state = agentState || { state: 'offline' };
|
|
const commands = AGENT_COMMANDS[agentId] || [];
|
|
const needsAttention = state.state === 'waiting' || notification > 0;
|
|
const isOpen = expanded || needsAttention;
|
|
|
|
useEffect(() => {
|
|
getAgentTasks(agentId, 10)
|
|
.then(d => setTasks(d.tasks || []))
|
|
.catch(() => setTasks([]));
|
|
}, [agentId]);
|
|
|
|
// Refresh tasks when state changes to idle (task likely completed)
|
|
useEffect(() => {
|
|
if (state.state === 'idle' && state.detail) {
|
|
getAgentTasks(agentId, 10)
|
|
.then(d => setTasks(d.tasks || []))
|
|
.catch(() => {});
|
|
}
|
|
}, [agentId, state.state, state.detail]);
|
|
|
|
// 오늘자 AI 토큰 사용량 폴링 (30초 간격 + 작업 완료 시 즉시 갱신)
|
|
useEffect(() => {
|
|
let cancelled = false;
|
|
const fetchUsage = () => {
|
|
getAgentTokenUsage(agentId, 1)
|
|
.then(d => { if (!cancelled) setTokenUsage(d); })
|
|
.catch(() => {});
|
|
};
|
|
fetchUsage();
|
|
const interval = setInterval(fetchUsage, 30000);
|
|
return () => {
|
|
cancelled = true;
|
|
clearInterval(interval);
|
|
};
|
|
}, [agentId, state.state, state.detail]);
|
|
|
|
const handleQuickAction = (cmd) => {
|
|
if (cmd.needsInput) {
|
|
setActiveCommand(cmd.action);
|
|
} else {
|
|
onCommand(agentId, cmd.action, {});
|
|
}
|
|
onClearNotification();
|
|
};
|
|
|
|
const handleSend = () => {
|
|
if (!input.trim() || !activeCommand) return;
|
|
const params = activeCommand === 'compose' ? { prompt: input }
|
|
: activeCommand === 'research' ? { keyword: input }
|
|
: { message: input };
|
|
onCommand(agentId, activeCommand, params);
|
|
setInput('');
|
|
setActiveCommand(null);
|
|
};
|
|
|
|
const formatTaskTime = (task) => {
|
|
const iso = task.completed_at || task.created_at;
|
|
if (!iso) return '';
|
|
const d = new Date(iso);
|
|
if (isNaN(d.getTime())) return '';
|
|
const now = new Date();
|
|
const pad = (n) => String(n).padStart(2, '0');
|
|
const hm = `${pad(d.getHours())}:${pad(d.getMinutes())}`;
|
|
const sameDay = d.toDateString() === now.toDateString();
|
|
const yesterday = new Date(now); yesterday.setDate(now.getDate() - 1);
|
|
const isYesterday = d.toDateString() === yesterday.toDateString();
|
|
if (sameDay) return `오늘 ${hm}`;
|
|
if (isYesterday) return `어제 ${hm}`;
|
|
return `${pad(d.getMonth() + 1)}-${pad(d.getDate())} ${hm}`;
|
|
};
|
|
|
|
const handleHeaderClick = (e) => {
|
|
e.stopPropagation();
|
|
setExpanded(v => !v);
|
|
onClearNotification();
|
|
};
|
|
|
|
return (
|
|
<div className={`ao-col ${isOpen ? 'ao-col--open' : 'ao-col--collapsed'} ${needsAttention ? 'ao-col--attention' : ''}`} onClick={onClearNotification}>
|
|
<div className="ao-col-header" style={{ borderColor: meta.color }} onClick={handleHeaderClick}>
|
|
<span className="ao-col-name" style={{ color: meta.color }}>{meta.name}</span>
|
|
{tokenUsage && tokenUsage.total_tokens > 0 && (
|
|
<span
|
|
className="ao-col-tokens"
|
|
title={`오늘 ${tokenUsage.task_count}건 작업 · ${tokenUsage.total_tokens.toLocaleString()} 토큰`}
|
|
>
|
|
🧮 {tokenUsage.total_tokens.toLocaleString()}
|
|
</span>
|
|
)}
|
|
<span className={`ao-col-state ao-col-state--${state.state}`}>{state.state}</span>
|
|
{notification > 0 && <span className="ao-col-badge">{notification}</span>}
|
|
<span className="ao-col-chevron" aria-hidden="true">{isOpen ? '▾' : '▸'}</span>
|
|
</div>
|
|
|
|
<div className="ao-col-body">
|
|
|
|
|
|
{state.detail && (
|
|
<div className="ao-col-detail">{state.detail}</div>
|
|
)}
|
|
|
|
{state.state === 'waiting' && state.taskId && (
|
|
<div className="ao-col-approval">
|
|
<span>승인 대기</span>
|
|
<button className="ao-btn ao-btn--approve" onClick={() => onApproval(agentId, state.taskId, true)}>승인</button>
|
|
<button className="ao-btn ao-btn--reject" onClick={() => onApproval(agentId, state.taskId, false)}>거절</button>
|
|
</div>
|
|
)}
|
|
|
|
<div className="ao-col-commands">
|
|
{commands.map(cmd => (
|
|
<button key={cmd.action} className="ao-cmd-btn" onClick={() => handleQuickAction(cmd)}>
|
|
{cmd.icon} {cmd.label}
|
|
</button>
|
|
))}
|
|
</div>
|
|
|
|
{activeCommand && (
|
|
<div className="ao-col-input">
|
|
<input
|
|
className="ao-chat-input"
|
|
value={input}
|
|
onChange={e => setInput(e.target.value)}
|
|
onKeyDown={e => e.key === 'Enter' && handleSend()}
|
|
placeholder="입력..."
|
|
autoFocus
|
|
/>
|
|
<button className="ao-btn ao-btn--send" onClick={handleSend}>전송</button>
|
|
</div>
|
|
)}
|
|
|
|
<div className="ao-col-tasks">
|
|
<div className="ao-col-tasks-title">최근 작업</div>
|
|
{tasks.length === 0 && <div className="ao-col-empty">이력 없음</div>}
|
|
{tasks.map(task => {
|
|
const badge = STATUS_BADGE[task.status] || STATUS_BADGE.pending;
|
|
return (
|
|
<div key={task.id} className="ao-col-task">
|
|
<div className="ao-col-task-row">
|
|
<span className="ao-col-task-type">{task.task_type}</span>
|
|
<span className="ao-col-task-badge" style={{ background: badge.bg }}>{badge.label}</span>
|
|
</div>
|
|
<div className="ao-col-task-time">
|
|
{formatTaskTime(task)}
|
|
{task.result_data?.telegram_sent !== undefined && (
|
|
<span className="ao-doc-tg-status">{task.result_data.telegram_sent ? ' TG OK' : ' TG Fail'}</span>
|
|
)}
|
|
</div>
|
|
{task.result_data && (
|
|
<details className="ao-col-task-detail">
|
|
<summary>결과</summary>
|
|
<pre>{JSON.stringify(task.result_data, null, 2)}</pre>
|
|
</details>
|
|
)}
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default AgentColumn;
|