"use client";
import {
PromptInput,
PromptInputActionAddAttachments,
PromptInputActionMenu,
PromptInputActionMenuContent,
PromptInputActionMenuTrigger,
PromptInputAttachment,
PromptInputAttachments,
PromptInputBody,
PromptInputButton,
PromptInputFooter,
PromptInputHeader,
type PromptInputMessage,
PromptInputSpeechButton,
PromptInputSubmit,
PromptInputTextarea,
PromptInputTools,
} from "@/components/ai-elements/prompt-input";
import {
ModelSelector,
ModelSelectorContent,
ModelSelectorEmpty,
ModelSelectorGroup,
ModelSelectorInput,
ModelSelectorItem,
ModelSelectorList,
ModelSelectorLogo,
ModelSelectorLogoGroup,
ModelSelectorName,
ModelSelectorTrigger,
} from "@/components/ai-elements/model-selector";
import {
Queue,
QueueItem,
QueueItemAction,
QueueItemActions,
QueueItemContent,
QueueItemDescription,
QueueItemIndicator,
QueueSection,
QueueSectionContent,
type QueueTodo,
} from "@/components/ai-elements/queue";
import { CheckIcon, GlobeIcon, Trash2 } from "lucide-react";
import { useRef, useState } from "react";
const models = [
{
id: "gpt-4o",
name: "GPT-4o",
chef: "OpenAI",
chefSlug: "openai",
providers: ["openai", "azure"],
},
{
id: "gpt-4o-mini",
name: "GPT-4o Mini",
chef: "OpenAI",
chefSlug: "openai",
providers: ["openai", "azure"],
},
{
id: "claude-opus-4-20250514",
name: "Claude 4 Opus",
chef: "Anthropic",
chefSlug: "anthropic",
providers: ["anthropic", "azure", "google", "amazon-bedrock"],
},
{
id: "claude-sonnet-4-20250514",
name: "Claude 4 Sonnet",
chef: "Anthropic",
chefSlug: "anthropic",
providers: ["anthropic", "azure", "google", "amazon-bedrock"],
},
{
id: "gemini-2.0-flash-exp",
name: "Gemini 2.0 Flash",
chef: "Google",
chefSlug: "google",
providers: ["google"],
},
];
const SUBMITTING_TIMEOUT = 200;
const STREAMING_TIMEOUT = 2000;
const sampleTodos: QueueTodo[] = [
{
id: "todo-1",
title: "Write project documentation",
description: "Complete the README and API docs",
status: "completed",
},
{
id: "todo-2",
title: "Implement authentication",
status: "pending",
},
{
id: "todo-3",
title: "Fix bug #42",
description: "Resolve crash on settings page",
status: "pending",
},
{
id: "todo-4",
title: "Refactor queue logic",
description: "Unify queue and todo state management",
status: "pending",
},
{
id: "todo-5",
title: "Add unit tests",
description: "Increase test coverage for hooks",
status: "pending",
},
];
const Example = () => {
const [todos, setTodos] = useState(sampleTodos);
const handleRemoveTodo = (id: string) => {
setTodos((prev) => prev.filter((todo) => todo.id !== id));
};
const [text, setText] = useState<string>("");
const [model, setModel] = useState<string>(models[0].id);
const [modelSelectorOpen, setModelSelectorOpen] = useState(false);
const [status, setStatus] = useState<
"submitted" | "streaming" | "ready" | "error"
>("ready");
const timeoutRef = useRef<NodeJS.Timeout | null>(null);
const textareaRef = useRef<HTMLTextAreaElement>(null);
const selectedModelData = models.find((m) => m.id === model);
const stop = () => {
console.log("Stopping request...");
// Clear any pending timeouts
if (timeoutRef.current) {
clearTimeout(timeoutRef.current);
timeoutRef.current = null;
}
setStatus("ready");
};
const handleSubmit = (message: PromptInputMessage) => {
// If currently streaming or submitted, stop instead of submitting
if (status === "streaming" || status === "submitted") {
stop();
return;
}
const hasText = Boolean(message.text);
const hasAttachments = Boolean(message.files?.length);
if (!(hasText || hasAttachments)) {
return;
}
setStatus("submitted");
console.log("Submitting message:", message);
setTimeout(() => {
setStatus("streaming");
}, SUBMITTING_TIMEOUT);
timeoutRef.current = setTimeout(() => {
setStatus("ready");
timeoutRef.current = null;
}, STREAMING_TIMEOUT);
};
return (
<div className="flex size-full flex-col justify-end">
<Queue className="mx-auto max-h-[150px] w-[95%] overflow-y-auto rounded-b-none border-input border-b-0">
{todos.length > 0 && (
<QueueSection>
<QueueSectionContent>
<div>
{todos.map((todo) => {
const isCompleted = todo.status === "completed";
return (
<QueueItem key={todo.id}>
<div className="flex items-center gap-2">
<QueueItemIndicator completed={isCompleted} />
<QueueItemContent completed={isCompleted}>
{todo.title}
</QueueItemContent>
<QueueItemActions>
<QueueItemAction
aria-label="Remove todo"
onClick={() => handleRemoveTodo(todo.id)}
>
<Trash2 size={12} />
</QueueItemAction>
</QueueItemActions>
</div>
{todo.description && (
<QueueItemDescription completed={isCompleted}>
{todo.description}
</QueueItemDescription>
)}
</QueueItem>
);
})}
</div>
</QueueSectionContent>
</QueueSection>
)}
</Queue>
<PromptInput globalDrop multiple onSubmit={handleSubmit}>
<PromptInputHeader>
<PromptInputAttachments>
{(attachment) => <PromptInputAttachment data={attachment} />}
</PromptInputAttachments>
</PromptInputHeader>
<PromptInputBody>
<PromptInputTextarea
onChange={(e) => setText(e.target.value)}
ref={textareaRef}
value={text}
/>
</PromptInputBody>
<PromptInputFooter>
<PromptInputTools>
<PromptInputActionMenu>
<PromptInputActionMenuTrigger />
<PromptInputActionMenuContent>
<PromptInputActionAddAttachments />
</PromptInputActionMenuContent>
</PromptInputActionMenu>
<PromptInputSpeechButton
onTranscriptionChange={setText}
textareaRef={textareaRef}
/>
<PromptInputButton>
<GlobeIcon size={16} />
<span>Search</span>
</PromptInputButton>
<ModelSelector
onOpenChange={setModelSelectorOpen}
open={modelSelectorOpen}
>
<ModelSelectorTrigger asChild>
<PromptInputButton>
{selectedModelData?.chefSlug && (
<ModelSelectorLogo provider={selectedModelData.chefSlug} />
)}
{selectedModelData?.name && (
<ModelSelectorName>
{selectedModelData.name}
</ModelSelectorName>
)}
</PromptInputButton>
</ModelSelectorTrigger>
<ModelSelectorContent>
<ModelSelectorInput placeholder="Search models..." />
<ModelSelectorList>
<ModelSelectorEmpty>No models found.</ModelSelectorEmpty>
{["OpenAI", "Anthropic", "Google"].map((chef) => (
<ModelSelectorGroup key={chef} heading={chef}>
{models
.filter((m) => m.chef === chef)
.map((m) => (
<ModelSelectorItem
key={m.id}
onSelect={() => {
setModel(m.id);
setModelSelectorOpen(false);
}}
value={m.id}
>
<ModelSelectorLogo provider={m.chefSlug} />
<ModelSelectorName>{m.name}</ModelSelectorName>
<ModelSelectorLogoGroup>
{m.providers.map((provider) => (
<ModelSelectorLogo
key={provider}
provider={provider}
/>
))}
</ModelSelectorLogoGroup>
{model === m.id ? (
<CheckIcon className="ml-auto size-4" />
) : (
<div className="ml-auto size-4" />
)}
</ModelSelectorItem>
))}
</ModelSelectorGroup>
))}
</ModelSelectorList>
</ModelSelectorContent>
</ModelSelector>
</PromptInputTools>
<PromptInputSubmit status={status} />
</PromptInputFooter>
</PromptInput>
</div>
);
};
export default Example;