- 添加 prompt_locale 参数支持简体中文、繁体中文和英文提示词本地化 - 移除内置 agents 配置以简化系统架构 - 更新 ContextBuilder 使用动态提示词模板而非硬编码内容 - 在 AgentLoop、Web 接口和 AgentService 中传递 locale 参数 - 添加输出语言指令确保用户界面内容按指定语言生成 - 扩展前端 LanguageSwitcher 组件支持三种语言选项 - 优化 Header 和侧边栏组件的响应式布局和文本截断处理 - 更新测试用例验证不同语言环境下的提示词正确性
101 lines
3.6 KiB
TypeScript
101 lines
3.6 KiB
TypeScript
'use client';
|
|
|
|
import React from 'react';
|
|
import { Activity, PanelRightOpen, X } from 'lucide-react';
|
|
|
|
import { TaskTimeline } from '@/components/task-detail';
|
|
import { ScrollArea } from '@/components/ui/scroll-area';
|
|
import { pickAppText } from '@/lib/i18n/core';
|
|
import { useAppI18n } from '@/lib/i18n/provider';
|
|
import type { TaskTimelineCard } from '@/types';
|
|
|
|
function ProgressPanel({
|
|
cards,
|
|
isLive,
|
|
onClose,
|
|
}: {
|
|
cards: TaskTimelineCard[];
|
|
isLive: boolean;
|
|
onClose?: () => void;
|
|
}) {
|
|
const { locale } = useAppI18n();
|
|
|
|
return (
|
|
<div className="flex h-full min-w-0 flex-col overflow-hidden bg-[#FBFAF9]">
|
|
<div className="flex h-16 min-w-0 shrink-0 items-center justify-between gap-3 border-b border-[#E6E1DE] px-5">
|
|
<div className="min-w-0">
|
|
<h2 className="truncate text-base font-semibold text-foreground">
|
|
{pickAppText(locale, '当前会话的运行进度', 'Current Session Progress')}
|
|
</h2>
|
|
<p className="flex min-w-0 items-center gap-1.5 text-xs text-muted-foreground">
|
|
{isLive ? <Activity className="h-3.5 w-3.5" /> : null}
|
|
<span className="truncate">
|
|
{isLive
|
|
? pickAppText(locale, '任务时间线实时更新', 'Task timeline updates live')
|
|
: pickAppText(locale, '与任务详情时间线一致', 'Matches the Task detail timeline')}
|
|
</span>
|
|
</p>
|
|
</div>
|
|
{onClose ? (
|
|
<button
|
|
type="button"
|
|
onClick={onClose}
|
|
className="rounded-full p-2 text-muted-foreground transition-colors hover:bg-[#ECE8E5] hover:text-foreground"
|
|
aria-label={pickAppText(locale, '关闭进度面板', 'Close progress panel')}
|
|
>
|
|
<X className="h-4 w-4" />
|
|
</button>
|
|
) : null}
|
|
</div>
|
|
|
|
<ScrollArea className="min-h-0 min-w-0 flex-1 overflow-hidden px-4 py-4">
|
|
<div className="min-w-0 max-w-full pb-6">
|
|
<TaskTimeline cards={cards} isLive={isLive} showHeader={false} />
|
|
</div>
|
|
</ScrollArea>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export function CurrentSessionProgressSidebar({
|
|
cards,
|
|
isLive,
|
|
}: {
|
|
cards: TaskTimelineCard[];
|
|
isLive: boolean;
|
|
}) {
|
|
const { locale } = useAppI18n();
|
|
const [mobileOpen, setMobileOpen] = React.useState(false);
|
|
|
|
return (
|
|
<>
|
|
<aside className="hidden h-full w-[380px] min-w-0 shrink-0 overflow-hidden border-l border-[#E6E1DE] xl:flex">
|
|
<ProgressPanel cards={cards} isLive={isLive} />
|
|
</aside>
|
|
|
|
<button
|
|
type="button"
|
|
onClick={() => setMobileOpen(true)}
|
|
className="fixed right-3 top-24 z-40 flex h-11 w-11 items-center justify-center rounded-full border border-[#E6E1DE] bg-white text-[#342E2B] shadow-[0_8px_22px_rgba(0,0,0,0.16)] transition-colors hover:bg-[#F7F6F5] xl:hidden"
|
|
aria-label={pickAppText(locale, '查看当前会话运行进度', 'View current session progress')}
|
|
>
|
|
<PanelRightOpen className="h-5 w-5" />
|
|
</button>
|
|
|
|
{mobileOpen ? (
|
|
<div className="fixed inset-0 z-50 xl:hidden">
|
|
<button
|
|
type="button"
|
|
className="absolute inset-0 bg-black/30"
|
|
onClick={() => setMobileOpen(false)}
|
|
aria-label={pickAppText(locale, '关闭进度面板', 'Close progress panel')}
|
|
/>
|
|
<div className="absolute inset-y-0 right-0 w-[min(92vw,390px)] min-w-0 overflow-hidden border-l border-[#E6E1DE] shadow-2xl">
|
|
<ProgressPanel cards={cards} isLive={isLive} onClose={() => setMobileOpen(false)} />
|
|
</div>
|
|
</div>
|
|
) : null}
|
|
</>
|
|
);
|
|
}
|