# Conversation URL: /components/conversation --- title: Conversation description: Wraps messages and automatically scrolls to the bottom. Also includes a scroll button that appears when not at the bottom. path: elements/components/conversation --- The `Conversation` component wraps messages and automatically scrolls to the bottom. Also includes a scroll button that appears when not at the bottom. ## Installation ## Usage ```tsx import { Conversation, ConversationContent, ConversationEmptyState, ConversationScrollButton } from "@/components/ai-elements/conversation"; ``` ```tsx {messages.length === 0 ? ( } title="No messages yet" description="Start a conversation to see messages here" /> ) : ( messages.map((message) => ( {message.content} )) )} ``` ## Usage with AI SDK Build a simple conversational UI with `Conversation` and [`PromptInput`](/elements/components/prompt-input): Add the following component to your frontend: ```tsx title="app/page.tsx" "use client"; import { Conversation, ConversationContent, ConversationEmptyState, ConversationScrollButton } from "@/components/ai-elements/conversation"; import { Message, MessageContent } from "@/components/ai-elements/message"; import { Input, PromptInputTextarea, PromptInputSubmit } from "@/components/ai-elements/prompt-input"; import { MessageSquare } from "lucide-react"; import { useState } from "react"; import { useChat } from "@ai-sdk/react"; import { Response } from "@/components/ai-elements/response"; const ConversationDemo = () => { const [input, setInput] = useState(""); const { messages, sendMessage, status } = useChat(); const handleSubmit = (e: React.FormEvent) => { e.preventDefault(); if (input.trim()) { sendMessage({ text: input }); setInput(""); } }; return (
{messages.length === 0 ? ( } title="Start a conversation" description="Type a message below to begin chatting" /> ) : ( messages.map((message) => ( {message.parts.map((part, i) => { switch (part.type) { case "text": // we don't use any reasoning or tool calls in this example return {part.text}; default: return null; } })} )) )} setInput(e.currentTarget.value)} className="pr-12" />
); }; export default ConversationDemo; ``` Add the following route to your backend: ```tsx title="api/chat/route.ts" import { streamText, UIMessage, convertToModelMessages } from "ai"; // Allow streaming responses up to 30 seconds export const maxDuration = 30; export async function POST(req: Request) { const { messages }: { messages: UIMessage[] } = await req.json(); const result = streamText({ model: "openai/gpt-4o", messages: convertToModelMessages(messages), }); return result.toUIMessageStreamResponse(); } ``` ## Features - Automatic scrolling to the bottom when new messages are added - Smooth scrolling behavior with configurable animation - Scroll button that appears when not at the bottom - Responsive design with customizable padding and spacing - Flexible content layout with consistent message spacing - Accessible with proper ARIA roles for screen readers - Customizable styling through className prop - Support for any number of child message components ## Props ### `` ', }, instance: { description: 'Optional instance for controlling the StickToBottom component.', type: 'StickToBottomInstance', }, children: { description: 'Render prop or ReactNode for custom rendering with context.', type: '((context: StickToBottomContext) => ReactNode) | ReactNode', }, '...props': { description: 'Any other props are spread to the root div.', type: 'Omit, "children">', }, }} /> ### `` ReactNode) | ReactNode', }, '...props': { description: 'Any other props are spread to the root div.', type: 'Omit, "children">', }, }} /> ### `` ', }, }} /> ### `` ', }, }} />