deer-flow/web/src/app/landing/components/multi-agent-visualization.tsx
2025-05-02 17:33:36 +08:00

307 lines
7.9 KiB
TypeScript

// Copyright (c) 2025 Bytedance Ltd. and/or its affiliates
// SPDX-License-Identifier: MIT
"use client";
import {
ReactFlow,
Background,
Handle,
Position,
type Edge,
type ReactFlowInstance,
} from "@xyflow/react";
import {
Play,
type LucideIcon,
ChevronRight,
ChevronLeft,
Pause,
Fullscreen,
Minimize,
} from "lucide-react";
import "@xyflow/react/dist/style.css";
import { useCallback, useRef, useState } from "react";
import { Tooltip } from "~/components/deer-flow/tooltip";
import { ShineBorder } from "~/components/magicui/shine-border";
import { Button } from "~/components/ui/button";
import { Slider } from "~/components/ui/slider";
import { useIntersectionObserver } from "~/hooks/use-intersection-observer";
import { cn } from "~/lib/utils";
import { playbook, type GraphNode } from "../store";
import {
activateStep,
nextStep,
play,
prevStep,
togglePlay,
useMAVStore,
} from "../store/mav-store";
const nodeTypes = {
circle: CircleNode,
agent: AgentNode,
default: AgentNode,
};
export function MultiAgentVisualization({ className }: { className?: string }) {
const {
graph: { nodes, edges },
activeStepIndex,
playing,
} = useMAVStore((state) => state);
const flowRef = useRef<ReactFlowInstance<GraphNode, Edge>>(null);
const containerRef = useRef<HTMLDivElement>(null);
const [hasPlayed, setHasPlayed] = useState(false);
const [fullscreen, setFullscreen] = useState(false);
const { ref: anchorRef } = useIntersectionObserver({
rootMargin: "0%",
threshold: 0,
onChange: (isIntersecting) => {
if (isIntersecting && !playing && !hasPlayed) {
setHasPlayed(true);
void play();
}
},
});
const toggleFullscreen = useCallback(async () => {
if (containerRef.current) {
if (!document.fullscreenElement) {
setFullscreen(true);
await containerRef.current.requestFullscreen();
setTimeout(() => {
void flowRef.current?.fitView({ maxZoom: 2.5 });
}, 100);
} else {
setFullscreen(false);
await document.exitFullscreen();
setTimeout(() => {
void flowRef.current?.fitView();
}, 100);
}
}
}, []);
return (
<div
ref={containerRef}
className={cn("flex h-full w-full flex-col pb-4", className)}
>
<ReactFlow
className={cn("flex min-h-0 flex-grow")}
style={{
["--xy-background-color-default" as string]: "transparent",
}}
nodes={nodes}
edges={edges}
nodeTypes={nodeTypes}
fitView
proOptions={{ hideAttribution: true }}
colorMode="dark"
panOnScroll={false}
zoomOnScroll={false}
preventScrolling={false}
panOnDrag={false}
onInit={(instance) => {
flowRef.current = instance;
}}
>
<Background
className="[mask-image:radial-gradient(800px_circle_at_center,white,transparent)]"
bgColor="var(--background)"
/>
<div
ref={anchorRef}
id="auto-run-animation-trigger"
className="absolute bottom-0 left-[50%] h-px w-px"
/>
</ReactFlow>
<div className="flex h-6 w-full shrink-0 items-center justify-center">
<div className="bg-muted/50 z-[200] flex rounded-3xl px-4 py-2">
<Tooltip title="Move to the previous step">
<Button variant="ghost" onClick={prevStep}>
<ChevronLeft className="size-5" />
</Button>
</Tooltip>
<Tooltip title="Play / Pause">
<Button variant="ghost" onClick={togglePlay}>
{playing ? (
<Pause className="size-5" />
) : (
<Play className="size-5" />
)}
</Button>
</Tooltip>
<Tooltip title="Move to the next step">
<Button
variant="ghost"
onClick={() => {
setHasPlayed(true);
nextStep();
}}
>
<ChevronRight className="size-5" />
</Button>
</Tooltip>
<div className="text-muted-foreground ml-2 flex items-center justify-center">
<Slider
className="w-120"
max={playbook.steps.length - 1}
min={0}
step={1}
value={[activeStepIndex >= 0 ? activeStepIndex : 0]}
onValueChange={([value]) => {
setHasPlayed(true);
activateStep(value!);
}}
/>
</div>
<Tooltip title="Toggle fullscreen">
<Button
variant="ghost"
size="icon"
onClick={() => {
setHasPlayed(true);
void toggleFullscreen();
}}
>
{fullscreen ? (
<Minimize className="size-5" />
) : (
<Fullscreen className="size-5" />
)}
</Button>
</Tooltip>
</div>
</div>
</div>
);
}
function CircleNode({ data }: { data: { label: string; active: boolean } }) {
return (
<>
{data.active && (
<ShineBorder
className="rounded-full"
shineColor={["#A07CFE", "#FE8FB5", "#FFBE7B"]}
/>
)}
<div className="flex h-10 w-10 items-center justify-center rounded-full border bg-[var(--xy-node-background-color-default)]">
<p className="text-xs">{data.label}</p>
</div>
{data.label === "Start" && (
<Handle
className="invisible"
type="source"
position={Position.Right}
id="right"
/>
)}
{data.label === "End" && (
<Handle
className="invisible"
type="target"
position={Position.Top}
id="top"
/>
)}
</>
);
}
function AgentNode({
data,
id,
}: {
data: {
icon?: LucideIcon;
label: string;
active: boolean;
stepDescription?: string;
stepTooltipPosition?: "left" | "right" | "top" | "bottom";
};
id: string;
}) {
return (
<>
{data.active && (
<ShineBorder
shineColor={["#A07CFE", "#FE8FB5", "#FFBE7B"]}
className="rounded-[2px]"
/>
)}
<Tooltip
className="max-w-50 text-[15px] opacity-70"
style={{
["--primary" as string]: "#333",
["--primary-foreground" as string]: "white",
}}
open={data.active && !!data.stepDescription}
title={data.stepDescription}
side={data.stepTooltipPosition}
sideOffset={20}
>
<div
id={id}
className="relative flex w-full items-center justify-center text-xs"
>
<div className="flex items-center gap-2">
{data.icon && <data.icon className="h-[1rem] w-[1rem]" />}
<span>{data.label}</span>
</div>
</div>
</Tooltip>
<Handle
className="invisible"
type="source"
position={Position.Left}
id="left"
/>
<Handle
className="invisible"
type="source"
position={Position.Right}
id="right"
/>
<Handle
className="invisible"
type="source"
position={Position.Top}
id="top"
/>
<Handle
className="invisible"
type="source"
position={Position.Bottom}
id="bottom"
/>
<Handle
className="invisible"
type="target"
position={Position.Left}
id="left"
/>
<Handle
className="invisible"
type="target"
position={Position.Right}
id="right"
/>
<Handle
className="invisible"
type="target"
position={Position.Top}
id="top"
/>
<Handle
className="invisible"
type="target"
position={Position.Bottom}
id="bottom"
/>
</>
);
}