mirror of
https://gitee.com/wanwujie/deer-flow
synced 2026-04-21 05:14:45 +08:00
283 lines
9.1 KiB
TypeScript
283 lines
9.1 KiB
TypeScript
// Copyright (c) 2025 Bytedance Ltd. and/or its affiliates
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
import { PythonOutlined } from "@ant-design/icons";
|
|
import { parse } from "best-effort-json-parser";
|
|
import { motion } from "framer-motion";
|
|
import { LRUCache } from "lru-cache";
|
|
import { BookOpenText, Search } from "lucide-react";
|
|
import { useMemo } from "react";
|
|
import SyntaxHighlighter from "react-syntax-highlighter";
|
|
import { docco } from "react-syntax-highlighter/dist/esm/styles/hljs";
|
|
|
|
import { Skeleton } from "~/components/ui/skeleton";
|
|
import type { ToolCallRuntime } from "~/core/messages";
|
|
import { useMessage, useStore } from "~/core/store";
|
|
import { cn } from "~/lib/utils";
|
|
|
|
import { FavIcon } from "./fav-icon";
|
|
import Image from "./image";
|
|
import { LoadingAnimation } from "./loading-animation";
|
|
import { Markdown } from "./markdown";
|
|
import { RainbowText } from "./rainbow-text";
|
|
|
|
export function ResearchActivitiesBlock({
|
|
className,
|
|
researchId,
|
|
}: {
|
|
className?: string;
|
|
researchId: string;
|
|
}) {
|
|
const activityIds = useStore((state) =>
|
|
state.researchActivityIds.get(researchId),
|
|
)!;
|
|
const ongoing = useStore((state) => state.ongoingResearchId === researchId);
|
|
return (
|
|
<>
|
|
<ul className={cn("flex flex-col py-4", className)}>
|
|
{activityIds.map(
|
|
(activityId, i) =>
|
|
i !== 0 && (
|
|
<motion.li
|
|
key={activityId}
|
|
style={{ transition: "all 0.4s ease-out" }}
|
|
initial={{ opacity: 0, y: 24 }}
|
|
animate={{ opacity: 1, y: 0 }}
|
|
transition={{
|
|
duration: 0.4,
|
|
ease: "easeOut",
|
|
}}
|
|
>
|
|
<ActivityMessage messageId={activityId} />
|
|
<ActivityListItem messageId={activityId} />
|
|
{i !== activityIds.length - 1 && <hr className="my-8" />}
|
|
</motion.li>
|
|
),
|
|
)}
|
|
</ul>
|
|
{ongoing && <LoadingAnimation className="mx-4 my-12" />}
|
|
</>
|
|
);
|
|
}
|
|
|
|
function ActivityMessage({ messageId }: { messageId: string }) {
|
|
const message = useMessage(messageId);
|
|
if (message?.agent && message.content) {
|
|
if (message.agent !== "reporter" && message.agent !== "planner") {
|
|
return (
|
|
<div className="px-4 py-2">
|
|
<Markdown animate>{message.content}</Markdown>
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function ActivityListItem({ messageId }: { messageId: string }) {
|
|
const message = useMessage(messageId);
|
|
if (message) {
|
|
if (!message.isStreaming && message.toolCalls?.length) {
|
|
for (const toolCall of message.toolCalls) {
|
|
if (toolCall.name === "web_search") {
|
|
return <WebSearchToolCall key={toolCall.id} toolCall={toolCall} />;
|
|
} else if (toolCall.name === "crawl_tool") {
|
|
return <CrawlToolCall key={toolCall.id} toolCall={toolCall} />;
|
|
} else if (toolCall.name === "python_repl_tool") {
|
|
return <PythonToolCall key={toolCall.id} toolCall={toolCall} />;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
const __pageCache = new LRUCache<string, string>({ max: 100 });
|
|
type SearchResult =
|
|
| {
|
|
type: "page";
|
|
title: string;
|
|
url: string;
|
|
content: string;
|
|
}
|
|
| {
|
|
type: "image";
|
|
image_url: string;
|
|
image_description: string;
|
|
};
|
|
function WebSearchToolCall({ toolCall }: { toolCall: ToolCallRuntime }) {
|
|
const searching = useMemo(() => {
|
|
return toolCall.result === undefined;
|
|
}, [toolCall.result]);
|
|
const searchResults = useMemo<SearchResult[]>(() => {
|
|
let results: SearchResult[] | undefined = undefined;
|
|
try {
|
|
results = toolCall.result ? parse(toolCall.result) : undefined;
|
|
} catch {
|
|
results = undefined;
|
|
}
|
|
if (Array.isArray(results)) {
|
|
results.forEach((result) => {
|
|
if (result.type === "page") {
|
|
__pageCache.set(result.url, result.title);
|
|
}
|
|
});
|
|
} else {
|
|
results = [];
|
|
}
|
|
return results;
|
|
}, [toolCall.result]);
|
|
const pageResults = useMemo(
|
|
() => searchResults?.filter((result) => result.type === "page"),
|
|
[searchResults],
|
|
);
|
|
const imageResults = useMemo(
|
|
() => searchResults?.filter((result) => result.type === "image"),
|
|
[searchResults],
|
|
);
|
|
return (
|
|
<section className="mt-4 pl-4">
|
|
<div className="font-medium italic">
|
|
<RainbowText
|
|
className="flex items-center"
|
|
animated={searchResults === undefined}
|
|
>
|
|
<Search className={"mr-2"} />
|
|
<span>Searching for </span>
|
|
<span className="max-w-[500px] overflow-hidden text-ellipsis whitespace-nowrap">
|
|
{(toolCall.args as { query: string }).query}
|
|
</span>
|
|
</RainbowText>
|
|
</div>
|
|
<div className="pr-4">
|
|
{pageResults && (
|
|
<ul className="mt-2 flex flex-wrap gap-4">
|
|
{searching &&
|
|
[...Array(6)].map((_, i) => (
|
|
<li
|
|
key={`search-result-${i}`}
|
|
className="flex h-40 w-40 gap-2 rounded-md text-sm"
|
|
>
|
|
<Skeleton
|
|
className="to-accent h-full w-full rounded-md bg-gradient-to-tl from-slate-400"
|
|
style={{ animationDelay: `${i * 0.2}s` }}
|
|
/>
|
|
</li>
|
|
))}
|
|
{pageResults
|
|
.filter((result) => result.type === "page")
|
|
.map((searchResult, i) => (
|
|
<motion.li
|
|
key={`search-result-${i}`}
|
|
className="text-muted-foreground bg-accent flex max-w-40 gap-2 rounded-md px-2 py-1 text-sm"
|
|
initial={{ opacity: 0, y: 10, scale: 0.66 }}
|
|
animate={{ opacity: 1, y: 0, scale: 1 }}
|
|
transition={{
|
|
duration: 0.2,
|
|
delay: i * 0.1,
|
|
ease: "easeOut",
|
|
}}
|
|
>
|
|
<FavIcon
|
|
className="mt-1"
|
|
url={searchResult.url}
|
|
title={searchResult.title}
|
|
/>
|
|
<a href={searchResult.url} target="_blank">
|
|
{searchResult.title}
|
|
</a>
|
|
</motion.li>
|
|
))}
|
|
{imageResults.map((searchResult, i) => (
|
|
<motion.li
|
|
key={`search-result-${i}`}
|
|
initial={{ opacity: 0, y: 10, scale: 0.66 }}
|
|
animate={{ opacity: 1, y: 0, scale: 1 }}
|
|
transition={{
|
|
duration: 0.2,
|
|
delay: i * 0.1,
|
|
ease: "easeOut",
|
|
}}
|
|
>
|
|
<a
|
|
className="flex flex-col gap-2 overflow-hidden rounded-md opacity-75 transition-opacity duration-300 hover:opacity-100"
|
|
href={searchResult.image_url}
|
|
target="_blank"
|
|
>
|
|
<Image
|
|
src={searchResult.image_url}
|
|
alt={searchResult.image_description}
|
|
className="bg-accent h-40 w-40 max-w-full rounded-md bg-cover bg-center bg-no-repeat"
|
|
imageClassName="hover:scale-110"
|
|
imageTransition
|
|
/>
|
|
</a>
|
|
</motion.li>
|
|
))}
|
|
</ul>
|
|
)}
|
|
</div>
|
|
</section>
|
|
);
|
|
}
|
|
|
|
function CrawlToolCall({ toolCall }: { toolCall: ToolCallRuntime }) {
|
|
const url = useMemo(
|
|
() => (toolCall.args as { url: string }).url,
|
|
[toolCall.args],
|
|
);
|
|
const title = useMemo(() => __pageCache.get(url), [url]);
|
|
return (
|
|
<section className="mt-4 pl-4">
|
|
<div className="font-medium italic">
|
|
<RainbowText
|
|
className="flex items-center"
|
|
animated={toolCall.result === undefined}
|
|
>
|
|
<BookOpenText className={"mr-2"} />
|
|
<span>Reading</span>
|
|
</RainbowText>
|
|
</div>
|
|
<ul className="mt-2 flex flex-wrap gap-4">
|
|
<motion.li
|
|
className="text-muted-foreground bg-accent flex h-40 w-40 gap-2 rounded-md px-2 py-1 text-sm"
|
|
initial={{ opacity: 0, y: 10, scale: 0.66 }}
|
|
animate={{ opacity: 1, y: 0, scale: 1 }}
|
|
transition={{
|
|
duration: 0.2,
|
|
ease: "easeOut",
|
|
}}
|
|
>
|
|
<FavIcon className="mt-1" url={url} title={title} />
|
|
<a href={url} target="_blank">
|
|
{title}
|
|
</a>
|
|
</motion.li>
|
|
</ul>
|
|
</section>
|
|
);
|
|
}
|
|
|
|
function PythonToolCall({ toolCall }: { toolCall: ToolCallRuntime }) {
|
|
const code = useMemo<string>(() => {
|
|
return (toolCall.args as { code: string }).code;
|
|
}, [toolCall.args]);
|
|
return (
|
|
<section className="mt-4 pl-4">
|
|
<div className="font-medium italic">
|
|
<PythonOutlined className={"mr-2"} />
|
|
<RainbowText animated={toolCall.result === undefined}>
|
|
Running Python code
|
|
</RainbowText>
|
|
</div>
|
|
<div className="px-5">
|
|
<div className="bg-accent mt-2 rounded-md p-2 text-sm">
|
|
<SyntaxHighlighter language="python" style={docco}>
|
|
{code}
|
|
</SyntaxHighlighter>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
);
|
|
}
|