Files
gh-nathanonn-claude-skills-…/docs/components/sources.md
2025-11-30 08:41:51 +08:00

7.0 KiB

Sources

URL: /components/sources


title: Sources description: A component that allows a user to view the sources or citations used to generate a response. path: elements/components/sources


The Sources component allows a user to view the sources or citations used to generate a response.

Installation

Usage

import { Source, Sources, SourcesContent, SourcesTrigger } from "@/components/ai-elements/sources";
<Sources>
    <SourcesTrigger count={1} />
    <SourcesContent>
        <Source href="https://ai-sdk.dev" title="AI SDK" />
    </SourcesContent>
</Sources>

Usage with AI SDK

Build a simple web search agent with Perplexity Sonar.

Add the following component to your frontend:

"use client";

import { useChat } from "@ai-sdk/react";
import { Source, Sources, SourcesContent, SourcesTrigger } from "@/components/ai-elements/sources";
import { Input, PromptInputTextarea, PromptInputSubmit } from "@/components/ai-elements/prompt-input";
import { Conversation, ConversationContent, ConversationScrollButton } from "@/components/ai-elements/conversation";
import { Message, MessageContent } from "@/components/ai-elements/message";
import { Response } from "@/components/ai-elements/response";
import { useState } from "react";
import { DefaultChatTransport } from "ai";

const SourceDemo = () => {
    const [input, setInput] = useState("");
    const { messages, sendMessage, status } = useChat({
        transport: new DefaultChatTransport({
            api: "/api/sources",
        }),
    });

    const handleSubmit = (e: React.FormEvent) => {
        e.preventDefault();
        if (input.trim()) {
            sendMessage({ text: input });
            setInput("");
        }
    };

    return (
        <div className="max-w-4xl mx-auto p-6 relative size-full rounded-lg border h-[600px]">
            <div className="flex flex-col h-full">
                <div className="flex-1 overflow-auto mb-4">
                    <Conversation>
                        <ConversationContent>
                            {messages.map((message) => (
                                <div key={message.id}>
                                    {message.role === "assistant" && (
                                        <Sources>
                                            <SourcesTrigger count={message.parts.filter((part) => part.type === "source-url").length} />
                                            {message.parts.map((part, i) => {
                                                switch (part.type) {
                                                    case "source-url":
                                                        return (
                                                            <SourcesContent key={`${message.id}-${i}`}>
                                                                <Source key={`${message.id}-${i}`} href={part.url} title={part.url} />
                                                            </SourcesContent>
                                                        );
                                                }
                                            })}
                                        </Sources>
                                    )}
                                    <Message from={message.role} key={message.id}>
                                        <MessageContent>
                                            {message.parts.map((part, i) => {
                                                switch (part.type) {
                                                    case "text":
                                                        return <Response key={`${message.id}-${i}`}>{part.text}</Response>;
                                                    default:
                                                        return null;
                                                }
                                            })}
                                        </MessageContent>
                                    </Message>
                                </div>
                            ))}
                        </ConversationContent>
                        <ConversationScrollButton />
                    </Conversation>
                </div>

                <Input onSubmit={handleSubmit} className="mt-4 w-full max-w-2xl mx-auto relative">
                    <PromptInputTextarea
                        value={input}
                        placeholder="Ask a question and search the..."
                        onChange={(e) => setInput(e.currentTarget.value)}
                        className="pr-12"
                    />
                    <PromptInputSubmit
                        status={status === "streaming" ? "streaming" : "ready"}
                        disabled={!input.trim()}
                        className="absolute bottom-1 right-1"
                    />
                </Input>
            </div>
        </div>
    );
};

export default SourceDemo;

Add the following route to your backend:

import { convertToModelMessages, streamText, UIMessage } from "ai";
import { perplexity } from "@ai-sdk/perplexity";

// 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: "perplexity/sonar",
        system: "You are a helpful assistant. Keep your responses short (< 100 words) unless you are asked for more details. ALWAYS USE SEARCH.",
        messages: convertToModelMessages(messages),
    });

    return result.toUIMessageStreamResponse({
        sendSources: true,
    });
}

Features

  • Collapsible component that allows a user to view the sources or citations used to generate a response
  • Customizable trigger and content components
  • Support for custom sources or citations
  • Responsive design with mobile-friendly controls
  • Clean, modern styling with customizable themes

Examples

Custom rendering

Props

<Sources />

<TypeTable type={{ '...props': { description: 'Any other props are spread to the root div.', type: 'React.HTMLAttributes', }, }} />

<SourcesTrigger />

<TypeTable type={{ count: { description: 'The number of sources to display in the trigger.', type: 'number', }, '...props': { description: 'Any other props are spread to the trigger button.', type: 'React.ButtonHTMLAttributes', }, }} />

<SourcesContent />

<TypeTable type={{ '...props': { description: 'Any other props are spread to the content container.', type: 'React.HTMLAttributes', }, }} />

<Source />

<TypeTable type={{ '...props': { description: 'Any other props are spread to the anchor element.', type: 'React.AnchorHTMLAttributes', }, }} />