Files
web-page/src/pages/agent-office/components/AgentColumn.jsx
gahusb 104a34912f feat(agent-office): 모바일 반응형 세로 스택 + 작업 시간 표기 개선
- 768px 이하에서 대시보드 세로 스택 + 에이전트 카드 아코디언 토글
- waiting/알림 있을 때 자동 펼침 및 좌측 강조 바
- 픽셀 오피스 캔버스 모바일 높이 140px로 축소 후 상단 배치
- 최근 작업 시간: completed_at 우선 + 오늘/어제/MM-DD HH:MM 포맷

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-04-15 00:19:32 +09:00

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;