mirror of
https://gitee.com/wanwujie/deer-flow
synced 2026-04-25 15:04:46 +08:00
* feat: add agent management functionality with creation, editing, and deletion * feat: enhance agent creation and chat experience - Added AgentWelcome component to display agent description on new thread creation. - Improved agent name validation with availability check during agent creation. - Updated NewAgentPage to handle agent creation flow more effectively, including enhanced error handling and user feedback. - Refactored chat components to streamline message handling and improve user experience. - Introduced new bootstrap skill for personalized onboarding conversations, including detailed conversation phases and a structured SOUL.md template. - Updated localization files to reflect new features and error messages. - General code cleanup and optimizations across various components and hooks. * Refactor workspace layout and agent management components - Updated WorkspaceLayout to use useLayoutEffect for sidebar state initialization. - Removed unused AgentFormDialog and related edit functionality from AgentCard. - Introduced ArtifactTrigger component to manage artifact visibility. - Enhanced ChatBox to handle artifact selection and display. - Improved message list rendering logic to avoid loading states. - Updated localization files to remove deprecated keys and add new translations. - Refined hooks for local settings and thread management to improve performance and clarity. - Added temporal awareness guidelines to deep research skill documentation. * feat: refactor chat components and introduce thread management hooks * feat: improve artifact file detail preview logic and clean up console logs * feat: refactor lead agent creation logic and improve logging details * feat: validate agent name format and enhance error handling in agent setup * feat: simplify thread search query by removing unnecessary metadata * feat: update query key in useDeleteThread and useRenameThread for consistency * feat: add isMock parameter to thread and artifact handling for improved testing * fix: reorder import of setup_agent for consistency in builtins module * feat: append mock parameter to thread links in CaseStudySection for testing purposes * fix: update load_agent_soul calls to use cfg.name for improved clarity * fix: update date format in apply_prompt_template for consistency * feat: integrate isMock parameter into artifact content loading for enhanced testing * docs: add license section to SKILL.md for clarity and attribution * feat(agent): enhance model resolution and agent configuration handling * chore: remove unused import of _resolve_model_name from agents * feat(agent): remove unused field * fix(agent): set default value for requested_model_name in _resolve_model_name function * feat(agent): update get_available_tools call to handle optional agent_config and improve middleware function signature --------- Co-authored-by: Willem Jiang <willem.jiang@gmail.com>
252 lines
9.0 KiB
TypeScript
252 lines
9.0 KiB
TypeScript
"use client";
|
|
|
|
import { ArrowLeftIcon, BotIcon, CheckCircleIcon } from "lucide-react";
|
|
import { useRouter } from "next/navigation";
|
|
import { useCallback, useMemo, useState } from "react";
|
|
|
|
import {
|
|
PromptInput,
|
|
PromptInputFooter,
|
|
PromptInputSubmit,
|
|
PromptInputTextarea,
|
|
} from "@/components/ai-elements/prompt-input";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Input } from "@/components/ui/input";
|
|
import { ArtifactsProvider } from "@/components/workspace/artifacts";
|
|
import { MessageList } from "@/components/workspace/messages";
|
|
import { ThreadContext } from "@/components/workspace/messages/context";
|
|
import type { Agent } from "@/core/agents";
|
|
import { checkAgentName, getAgent } from "@/core/agents/api";
|
|
import { useI18n } from "@/core/i18n/hooks";
|
|
import { useThreadStream } from "@/core/threads/hooks";
|
|
import { uuid } from "@/core/utils/uuid";
|
|
import { cn } from "@/lib/utils";
|
|
|
|
type Step = "name" | "chat";
|
|
|
|
const NAME_RE = /^[A-Za-z0-9-]+$/;
|
|
|
|
export default function NewAgentPage() {
|
|
const { t } = useI18n();
|
|
const router = useRouter();
|
|
|
|
// ── Step 1: name form ──────────────────────────────────────────────────────
|
|
const [step, setStep] = useState<Step>("name");
|
|
const [nameInput, setNameInput] = useState("");
|
|
const [nameError, setNameError] = useState("");
|
|
const [isCheckingName, setIsCheckingName] = useState(false);
|
|
const [agentName, setAgentName] = useState("");
|
|
const [agent, setAgent] = useState<Agent | null>(null);
|
|
// ── Step 2: chat ───────────────────────────────────────────────────────────
|
|
|
|
// Stable thread ID — all turns belong to the same thread
|
|
const threadId = useMemo(() => uuid(), []);
|
|
|
|
const [thread, sendMessage] = useThreadStream({
|
|
context: {
|
|
mode: "flash",
|
|
is_bootstrap: true,
|
|
},
|
|
onToolEnd({ name }) {
|
|
if (name !== "setup_agent" || !agentName) return;
|
|
getAgent(agentName)
|
|
.then((fetched) => setAgent(fetched))
|
|
.catch(() => {
|
|
// agent write may not be flushed yet — ignore silently
|
|
});
|
|
},
|
|
});
|
|
|
|
// ── Handlers ───────────────────────────────────────────────────────────────
|
|
|
|
const handleConfirmName = useCallback(async () => {
|
|
const trimmed = nameInput.trim();
|
|
if (!trimmed) return;
|
|
if (!NAME_RE.test(trimmed)) {
|
|
setNameError(t.agents.nameStepInvalidError);
|
|
return;
|
|
}
|
|
setNameError("");
|
|
setIsCheckingName(true);
|
|
try {
|
|
const result = await checkAgentName(trimmed);
|
|
if (!result.available) {
|
|
setNameError(t.agents.nameStepAlreadyExistsError);
|
|
return;
|
|
}
|
|
} catch {
|
|
setNameError(t.agents.nameStepCheckError);
|
|
return;
|
|
} finally {
|
|
setIsCheckingName(false);
|
|
}
|
|
setAgentName(trimmed);
|
|
setStep("chat");
|
|
await sendMessage(threadId, {
|
|
text: t.agents.nameStepBootstrapMessage.replace("{name}", trimmed),
|
|
files: [],
|
|
});
|
|
}, [
|
|
nameInput,
|
|
sendMessage,
|
|
threadId,
|
|
t.agents.nameStepBootstrapMessage,
|
|
t.agents.nameStepInvalidError,
|
|
t.agents.nameStepAlreadyExistsError,
|
|
t.agents.nameStepCheckError,
|
|
]);
|
|
|
|
const handleNameKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => {
|
|
if (e.key === "Enter") {
|
|
e.preventDefault();
|
|
void handleConfirmName();
|
|
}
|
|
};
|
|
|
|
const handleChatSubmit = useCallback(
|
|
async (text: string) => {
|
|
const trimmed = text.trim();
|
|
if (!trimmed || thread.isLoading) return;
|
|
await sendMessage(
|
|
threadId,
|
|
{ text: trimmed, files: [] },
|
|
{ agent_name: agentName },
|
|
);
|
|
},
|
|
[thread.isLoading, sendMessage, threadId, agentName],
|
|
);
|
|
|
|
// ── Shared header ──────────────────────────────────────────────────────────
|
|
|
|
const header = (
|
|
<header className="flex shrink-0 items-center gap-3 border-b px-4 py-3">
|
|
<Button
|
|
variant="ghost"
|
|
size="icon-sm"
|
|
onClick={() => router.push("/workspace/agents")}
|
|
>
|
|
<ArrowLeftIcon className="h-4 w-4" />
|
|
</Button>
|
|
<h1 className="text-sm font-semibold">{t.agents.createPageTitle}</h1>
|
|
</header>
|
|
);
|
|
|
|
// ── Step 1: name form ──────────────────────────────────────────────────────
|
|
|
|
if (step === "name") {
|
|
return (
|
|
<div className="flex size-full flex-col">
|
|
{header}
|
|
<main className="flex flex-1 flex-col items-center justify-center px-4">
|
|
<div className="w-full max-w-sm space-y-8">
|
|
<div className="space-y-3 text-center">
|
|
<div className="bg-primary/10 mx-auto flex h-14 w-14 items-center justify-center rounded-full">
|
|
<BotIcon className="text-primary h-7 w-7" />
|
|
</div>
|
|
<div className="space-y-1">
|
|
<h2 className="text-xl font-semibold">
|
|
{t.agents.nameStepTitle}
|
|
</h2>
|
|
<p className="text-muted-foreground text-sm">
|
|
{t.agents.nameStepHint}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="space-y-3">
|
|
<Input
|
|
autoFocus
|
|
placeholder={t.agents.nameStepPlaceholder}
|
|
value={nameInput}
|
|
onChange={(e) => {
|
|
setNameInput(e.target.value);
|
|
setNameError("");
|
|
}}
|
|
onKeyDown={handleNameKeyDown}
|
|
className={cn(nameError && "border-destructive")}
|
|
/>
|
|
{nameError && (
|
|
<p className="text-destructive text-sm">{nameError}</p>
|
|
)}
|
|
<Button
|
|
className="w-full"
|
|
onClick={() => void handleConfirmName()}
|
|
disabled={!nameInput.trim() || isCheckingName}
|
|
>
|
|
{t.agents.nameStepContinue}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</main>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// ── Step 2: chat ───────────────────────────────────────────────────────────
|
|
|
|
return (
|
|
<ThreadContext.Provider value={{ thread }}>
|
|
<ArtifactsProvider>
|
|
<div className="flex size-full flex-col">
|
|
{header}
|
|
|
|
<main className="flex min-h-0 flex-1 flex-col">
|
|
{/* ── Message area ── */}
|
|
<div className="flex min-h-0 flex-1 justify-center">
|
|
<MessageList
|
|
className="size-full pt-10"
|
|
threadId={threadId}
|
|
thread={thread}
|
|
/>
|
|
</div>
|
|
|
|
{/* ── Bottom action area ── */}
|
|
<div className="bg-background flex shrink-0 justify-center border-t px-4 py-4">
|
|
<div className="w-full max-w-(--container-width-md)">
|
|
{agent ? (
|
|
// ✅ Success card
|
|
<div className="flex flex-col items-center gap-4 rounded-2xl border py-8 text-center">
|
|
<CheckCircleIcon className="text-primary h-10 w-10" />
|
|
<p className="font-semibold">{t.agents.agentCreated}</p>
|
|
<div className="flex gap-2">
|
|
<Button
|
|
onClick={() =>
|
|
router.push(
|
|
`/workspace/agents/${agentName}/chats/new`,
|
|
)
|
|
}
|
|
>
|
|
{t.agents.startChatting}
|
|
</Button>
|
|
<Button
|
|
variant="outline"
|
|
onClick={() => router.push("/workspace/agents")}
|
|
>
|
|
{t.agents.backToGallery}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
) : (
|
|
// 📝 Normal input
|
|
<PromptInput
|
|
onSubmit={({ text }) => void handleChatSubmit(text)}
|
|
>
|
|
<PromptInputTextarea
|
|
autoFocus
|
|
placeholder={t.agents.createPageSubtitle}
|
|
disabled={thread.isLoading}
|
|
/>
|
|
<PromptInputFooter className="justify-end">
|
|
<PromptInputSubmit disabled={thread.isLoading} />
|
|
</PromptInputFooter>
|
|
</PromptInput>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</main>
|
|
</div>
|
|
</ArtifactsProvider>
|
|
</ThreadContext.Provider>
|
|
);
|
|
}
|