Merge pull request #82 from MkSaaSHQ/dev/ai-elements
feat: AI Chat demo with ai elements
This commit is contained in:
commit
fc53045d99
@ -23,6 +23,7 @@
|
|||||||
"src/components/magicui/*.tsx",
|
"src/components/magicui/*.tsx",
|
||||||
"src/components/animate-ui/*.tsx",
|
"src/components/animate-ui/*.tsx",
|
||||||
"src/components/tailark/*.tsx",
|
"src/components/tailark/*.tsx",
|
||||||
|
"src/components/ai-elements/*.tsx",
|
||||||
"src/app/[[]locale]/preview/**",
|
"src/app/[[]locale]/preview/**",
|
||||||
"src/payment/types.ts",
|
"src/payment/types.ts",
|
||||||
"src/credits/types.ts",
|
"src/credits/types.ts",
|
||||||
@ -85,6 +86,7 @@
|
|||||||
"src/components/magicui/*.tsx",
|
"src/components/magicui/*.tsx",
|
||||||
"src/components/animate-ui/*.tsx",
|
"src/components/animate-ui/*.tsx",
|
||||||
"src/components/tailark/*.tsx",
|
"src/components/tailark/*.tsx",
|
||||||
|
"src/components/ai-elements/*.tsx",
|
||||||
"src/app/[[]locale]/preview/**",
|
"src/app/[[]locale]/preview/**",
|
||||||
"src/payment/types.ts",
|
"src/payment/types.ts",
|
||||||
"src/credits/types.ts",
|
"src/credits/types.ts",
|
||||||
|
@ -181,6 +181,7 @@ CRON_JOBS_PASSWORD=""
|
|||||||
# AI
|
# AI
|
||||||
# https://mksaas.com/docs/ai
|
# https://mksaas.com/docs/ai
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
|
AI_GATEWAY_API_KEY=""
|
||||||
FAL_API_KEY=""
|
FAL_API_KEY=""
|
||||||
FIREWORKS_API_KEY=""
|
FIREWORKS_API_KEY=""
|
||||||
OPENAI_API_KEY=""
|
OPENAI_API_KEY=""
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
"@ai-sdk/fireworks": "^1.0.0",
|
"@ai-sdk/fireworks": "^1.0.0",
|
||||||
"@ai-sdk/google": "^2.0.0",
|
"@ai-sdk/google": "^2.0.0",
|
||||||
"@ai-sdk/openai": "^2.0.0",
|
"@ai-sdk/openai": "^2.0.0",
|
||||||
|
"@ai-sdk/react": "^2.0.22",
|
||||||
"@ai-sdk/replicate": "^1.0.0",
|
"@ai-sdk/replicate": "^1.0.0",
|
||||||
"@base-ui-components/react": "1.0.0-beta.0",
|
"@base-ui-components/react": "1.0.0-beta.0",
|
||||||
"@better-fetch/fetch": "^1.1.18",
|
"@better-fetch/fetch": "^1.1.18",
|
||||||
@ -74,6 +75,7 @@
|
|||||||
"@radix-ui/react-toggle": "^1.1.2",
|
"@radix-ui/react-toggle": "^1.1.2",
|
||||||
"@radix-ui/react-toggle-group": "^1.1.2",
|
"@radix-ui/react-toggle-group": "^1.1.2",
|
||||||
"@radix-ui/react-tooltip": "^1.1.8",
|
"@radix-ui/react-tooltip": "^1.1.8",
|
||||||
|
"@radix-ui/react-use-controllable-state": "^1.2.2",
|
||||||
"@react-email/components": "0.0.33",
|
"@react-email/components": "0.0.33",
|
||||||
"@react-email/render": "1.0.5",
|
"@react-email/render": "1.0.5",
|
||||||
"@stripe/stripe-js": "^5.6.0",
|
"@stripe/stripe-js": "^5.6.0",
|
||||||
@ -118,6 +120,7 @@
|
|||||||
"react-hook-form": "^7.62.0",
|
"react-hook-form": "^7.62.0",
|
||||||
"react-remove-scroll": "^2.6.3",
|
"react-remove-scroll": "^2.6.3",
|
||||||
"react-resizable-panels": "^2.1.7",
|
"react-resizable-panels": "^2.1.7",
|
||||||
|
"react-syntax-highlighter": "^15.6.3",
|
||||||
"react-tweet": "^3.2.2",
|
"react-tweet": "^3.2.2",
|
||||||
"react-use-measure": "^2.1.7",
|
"react-use-measure": "^2.1.7",
|
||||||
"recharts": "^2.15.1",
|
"recharts": "^2.15.1",
|
||||||
@ -125,6 +128,7 @@
|
|||||||
"s3mini": "^0.2.0",
|
"s3mini": "^0.2.0",
|
||||||
"shiki": "^2.4.2",
|
"shiki": "^2.4.2",
|
||||||
"sonner": "^2.0.0",
|
"sonner": "^2.0.0",
|
||||||
|
"streamdown": "^1.0.12",
|
||||||
"stripe": "^17.6.0",
|
"stripe": "^17.6.0",
|
||||||
"swiper": "^11.2.5",
|
"swiper": "^11.2.5",
|
||||||
"tailwind-merge": "^3.0.2",
|
"tailwind-merge": "^3.0.2",
|
||||||
@ -132,6 +136,7 @@
|
|||||||
"tw-animate-css": "^1.2.4",
|
"tw-animate-css": "^1.2.4",
|
||||||
"use-intl": "^3.26.5",
|
"use-intl": "^3.26.5",
|
||||||
"use-media": "^1.5.0",
|
"use-media": "^1.5.0",
|
||||||
|
"use-stick-to-bottom": "^1.1.1",
|
||||||
"vaul": "^1.1.2",
|
"vaul": "^1.1.2",
|
||||||
"zod": "^4.0.17",
|
"zod": "^4.0.17",
|
||||||
"zustand": "^5.0.3"
|
"zustand": "^5.0.3"
|
||||||
@ -145,6 +150,7 @@
|
|||||||
"@types/pg": "^8.11.11",
|
"@types/pg": "^8.11.11",
|
||||||
"@types/react": "^19",
|
"@types/react": "^19",
|
||||||
"@types/react-dom": "^19",
|
"@types/react-dom": "^19",
|
||||||
|
"@types/react-syntax-highlighter": "^15.5.13",
|
||||||
"drizzle-kit": "^0.30.4",
|
"drizzle-kit": "^0.30.4",
|
||||||
"knip": "^5.61.2",
|
"knip": "^5.61.2",
|
||||||
"postcss": "^8",
|
"postcss": "^8",
|
||||||
|
596
pnpm-lock.yaml
generated
596
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
183
src/app/[locale]/(marketing)/ai/chat/page.tsx
Normal file
183
src/app/[locale]/(marketing)/ai/chat/page.tsx
Normal file
@ -0,0 +1,183 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import {
|
||||||
|
Conversation,
|
||||||
|
ConversationContent,
|
||||||
|
ConversationScrollButton,
|
||||||
|
} from '@/components/ai-elements/conversation';
|
||||||
|
import { Loader } from '@/components/ai-elements/loader';
|
||||||
|
import { Message, MessageContent } from '@/components/ai-elements/message';
|
||||||
|
import {
|
||||||
|
PromptInput,
|
||||||
|
PromptInputButton,
|
||||||
|
PromptInputModelSelect,
|
||||||
|
PromptInputModelSelectContent,
|
||||||
|
PromptInputModelSelectItem,
|
||||||
|
PromptInputModelSelectTrigger,
|
||||||
|
PromptInputModelSelectValue,
|
||||||
|
PromptInputSubmit,
|
||||||
|
PromptInputTextarea,
|
||||||
|
PromptInputToolbar,
|
||||||
|
PromptInputTools,
|
||||||
|
} from '@/components/ai-elements/prompt-input';
|
||||||
|
import {
|
||||||
|
Reasoning,
|
||||||
|
ReasoningContent,
|
||||||
|
ReasoningTrigger,
|
||||||
|
} from '@/components/ai-elements/reasoning';
|
||||||
|
import { Response } from '@/components/ai-elements/response';
|
||||||
|
import {
|
||||||
|
Source,
|
||||||
|
Sources,
|
||||||
|
SourcesContent,
|
||||||
|
SourcesTrigger,
|
||||||
|
} from '@/components/ai-elements/source';
|
||||||
|
import { useChat } from '@ai-sdk/react';
|
||||||
|
import { GlobeIcon } from 'lucide-react';
|
||||||
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
const models = [
|
||||||
|
{
|
||||||
|
name: 'GPT 4o',
|
||||||
|
value: 'openai/gpt-4o',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Deepseek R1',
|
||||||
|
value: 'deepseek/deepseek-r1',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const ChatBotDemo = () => {
|
||||||
|
const [input, setInput] = useState('');
|
||||||
|
const [model, setModel] = useState<string>(models[0].value);
|
||||||
|
const [webSearch, setWebSearch] = useState(false);
|
||||||
|
const { messages, sendMessage, status } = useChat();
|
||||||
|
|
||||||
|
const handleSubmit = (e: React.FormEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
if (input.trim()) {
|
||||||
|
sendMessage(
|
||||||
|
{ text: input },
|
||||||
|
{
|
||||||
|
body: {
|
||||||
|
model: model,
|
||||||
|
webSearch: webSearch,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
setInput('');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="max-w-4xl mx-auto p-6 relative size-full h-screen rounded-lg bg-muted/50">
|
||||||
|
<div className="flex flex-col h-full">
|
||||||
|
<Conversation className="h-full">
|
||||||
|
<ConversationContent>
|
||||||
|
{messages.map((message) => (
|
||||||
|
<div key={message.id}>
|
||||||
|
{message.role === 'assistant' && (
|
||||||
|
<Sources>
|
||||||
|
{message.parts.map((part, i) => {
|
||||||
|
switch (part.type) {
|
||||||
|
case 'source-url':
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<SourcesTrigger
|
||||||
|
count={
|
||||||
|
message.parts.filter(
|
||||||
|
(part) => part.type === 'source-url'
|
||||||
|
).length
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<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>
|
||||||
|
);
|
||||||
|
case 'reasoning':
|
||||||
|
return (
|
||||||
|
<Reasoning
|
||||||
|
key={`${message.id}-${i}`}
|
||||||
|
className="w-full"
|
||||||
|
isStreaming={status === 'streaming'}
|
||||||
|
>
|
||||||
|
<ReasoningTrigger />
|
||||||
|
<ReasoningContent>{part.text}</ReasoningContent>
|
||||||
|
</Reasoning>
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
})}
|
||||||
|
</MessageContent>
|
||||||
|
</Message>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{status === 'submitted' && <Loader />}
|
||||||
|
</ConversationContent>
|
||||||
|
<ConversationScrollButton />
|
||||||
|
</Conversation>
|
||||||
|
|
||||||
|
<PromptInput onSubmit={handleSubmit} className="mt-4">
|
||||||
|
<PromptInputTextarea
|
||||||
|
onChange={(e) => setInput(e.target.value)}
|
||||||
|
value={input}
|
||||||
|
/>
|
||||||
|
<PromptInputToolbar>
|
||||||
|
<PromptInputTools>
|
||||||
|
<PromptInputButton
|
||||||
|
variant={webSearch ? 'default' : 'ghost'}
|
||||||
|
onClick={() => setWebSearch(!webSearch)}
|
||||||
|
>
|
||||||
|
<GlobeIcon size={16} />
|
||||||
|
<span>Search</span>
|
||||||
|
</PromptInputButton>
|
||||||
|
<PromptInputModelSelect
|
||||||
|
onValueChange={(value) => {
|
||||||
|
setModel(value);
|
||||||
|
}}
|
||||||
|
value={model}
|
||||||
|
>
|
||||||
|
<PromptInputModelSelectTrigger>
|
||||||
|
<PromptInputModelSelectValue />
|
||||||
|
</PromptInputModelSelectTrigger>
|
||||||
|
<PromptInputModelSelectContent>
|
||||||
|
{models.map((model) => (
|
||||||
|
<PromptInputModelSelectItem
|
||||||
|
key={model.value}
|
||||||
|
value={model.value}
|
||||||
|
>
|
||||||
|
{model.name}
|
||||||
|
</PromptInputModelSelectItem>
|
||||||
|
))}
|
||||||
|
</PromptInputModelSelectContent>
|
||||||
|
</PromptInputModelSelect>
|
||||||
|
</PromptInputTools>
|
||||||
|
<PromptInputSubmit disabled={!input} status={status} />
|
||||||
|
</PromptInputToolbar>
|
||||||
|
</PromptInput>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ChatBotDemo;
|
26
src/app/api/chat/route.ts
Normal file
26
src/app/api/chat/route.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import { type UIMessage, convertToModelMessages, streamText } from 'ai';
|
||||||
|
|
||||||
|
// Allow streaming responses up to 30 seconds
|
||||||
|
export const maxDuration = 30;
|
||||||
|
|
||||||
|
export async function POST(req: Request) {
|
||||||
|
const {
|
||||||
|
messages,
|
||||||
|
model,
|
||||||
|
webSearch,
|
||||||
|
}: { messages: UIMessage[]; model: string; webSearch: boolean } =
|
||||||
|
await req.json();
|
||||||
|
|
||||||
|
const result = streamText({
|
||||||
|
model: webSearch ? 'perplexity/sonar' : model,
|
||||||
|
messages: convertToModelMessages(messages),
|
||||||
|
system:
|
||||||
|
'You are a helpful assistant that can answer questions and help with tasks',
|
||||||
|
});
|
||||||
|
|
||||||
|
// send sources and reasoning back to the client
|
||||||
|
return result.toUIMessageStreamResponse({
|
||||||
|
sendSources: true,
|
||||||
|
sendReasoning: true,
|
||||||
|
});
|
||||||
|
}
|
65
src/components/ai-elements/actions.tsx
Normal file
65
src/components/ai-elements/actions.tsx
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import {
|
||||||
|
Tooltip,
|
||||||
|
TooltipContent,
|
||||||
|
TooltipProvider,
|
||||||
|
TooltipTrigger,
|
||||||
|
} from '@/components/ui/tooltip';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { ComponentProps } from 'react';
|
||||||
|
|
||||||
|
export type ActionsProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const Actions = ({ className, children, ...props }: ActionsProps) => (
|
||||||
|
<div className={cn('flex items-center gap-1', className)} {...props}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ActionProps = ComponentProps<typeof Button> & {
|
||||||
|
tooltip?: string;
|
||||||
|
label?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Action = ({
|
||||||
|
tooltip,
|
||||||
|
children,
|
||||||
|
label,
|
||||||
|
className,
|
||||||
|
variant = 'ghost',
|
||||||
|
size = 'sm',
|
||||||
|
...props
|
||||||
|
}: ActionProps) => {
|
||||||
|
const button = (
|
||||||
|
<Button
|
||||||
|
className={cn(
|
||||||
|
'size-9 p-1.5 text-muted-foreground hover:text-foreground relative',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
size={size}
|
||||||
|
type="button"
|
||||||
|
variant={variant}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<span className="sr-only">{label || tooltip}</span>
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (tooltip) {
|
||||||
|
return (
|
||||||
|
<TooltipProvider>
|
||||||
|
<Tooltip>
|
||||||
|
<TooltipTrigger asChild>{button}</TooltipTrigger>
|
||||||
|
<TooltipContent>
|
||||||
|
<p>{tooltip}</p>
|
||||||
|
</TooltipContent>
|
||||||
|
</Tooltip>
|
||||||
|
</TooltipProvider>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return button;
|
||||||
|
};
|
212
src/components/ai-elements/branch.tsx
Normal file
212
src/components/ai-elements/branch.tsx
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { UIMessage } from 'ai';
|
||||||
|
import { ChevronLeftIcon, ChevronRightIcon } from 'lucide-react';
|
||||||
|
import type { ComponentProps, HTMLAttributes, ReactElement } from 'react';
|
||||||
|
import { createContext, useContext, useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
type BranchContextType = {
|
||||||
|
currentBranch: number;
|
||||||
|
totalBranches: number;
|
||||||
|
goToPrevious: () => void;
|
||||||
|
goToNext: () => void;
|
||||||
|
branches: ReactElement[];
|
||||||
|
setBranches: (branches: ReactElement[]) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const BranchContext = createContext<BranchContextType | null>(null);
|
||||||
|
|
||||||
|
const useBranch = () => {
|
||||||
|
const context = useContext(BranchContext);
|
||||||
|
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('Branch components must be used within Branch');
|
||||||
|
}
|
||||||
|
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type BranchProps = HTMLAttributes<HTMLDivElement> & {
|
||||||
|
defaultBranch?: number;
|
||||||
|
onBranchChange?: (branchIndex: number) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Branch = ({
|
||||||
|
defaultBranch = 0,
|
||||||
|
onBranchChange,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: BranchProps) => {
|
||||||
|
const [currentBranch, setCurrentBranch] = useState(defaultBranch);
|
||||||
|
const [branches, setBranches] = useState<ReactElement[]>([]);
|
||||||
|
|
||||||
|
const handleBranchChange = (newBranch: number) => {
|
||||||
|
setCurrentBranch(newBranch);
|
||||||
|
onBranchChange?.(newBranch);
|
||||||
|
};
|
||||||
|
|
||||||
|
const goToPrevious = () => {
|
||||||
|
const newBranch =
|
||||||
|
currentBranch > 0 ? currentBranch - 1 : branches.length - 1;
|
||||||
|
handleBranchChange(newBranch);
|
||||||
|
};
|
||||||
|
|
||||||
|
const goToNext = () => {
|
||||||
|
const newBranch =
|
||||||
|
currentBranch < branches.length - 1 ? currentBranch + 1 : 0;
|
||||||
|
handleBranchChange(newBranch);
|
||||||
|
};
|
||||||
|
|
||||||
|
const contextValue: BranchContextType = {
|
||||||
|
currentBranch,
|
||||||
|
totalBranches: branches.length,
|
||||||
|
goToPrevious,
|
||||||
|
goToNext,
|
||||||
|
branches,
|
||||||
|
setBranches,
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BranchContext.Provider value={contextValue}>
|
||||||
|
<div
|
||||||
|
className={cn('grid w-full gap-2 [&>div]:pb-0', className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
</BranchContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type BranchMessagesProps = HTMLAttributes<HTMLDivElement>;
|
||||||
|
|
||||||
|
export const BranchMessages = ({ children, ...props }: BranchMessagesProps) => {
|
||||||
|
const { currentBranch, setBranches, branches } = useBranch();
|
||||||
|
const childrenArray = Array.isArray(children) ? children : [children];
|
||||||
|
|
||||||
|
// Use useEffect to update branches when they change
|
||||||
|
useEffect(() => {
|
||||||
|
if (branches.length !== childrenArray.length) {
|
||||||
|
setBranches(childrenArray);
|
||||||
|
}
|
||||||
|
}, [childrenArray, branches, setBranches]);
|
||||||
|
|
||||||
|
return childrenArray.map((branch, index) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'grid gap-2 overflow-hidden [&>div]:pb-0',
|
||||||
|
index === currentBranch ? 'block' : 'hidden'
|
||||||
|
)}
|
||||||
|
key={branch.key}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{branch}
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
};
|
||||||
|
|
||||||
|
export type BranchSelectorProps = HTMLAttributes<HTMLDivElement> & {
|
||||||
|
from: UIMessage['role'];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const BranchSelector = ({
|
||||||
|
className,
|
||||||
|
from,
|
||||||
|
...props
|
||||||
|
}: BranchSelectorProps) => {
|
||||||
|
const { totalBranches } = useBranch();
|
||||||
|
|
||||||
|
// Don't render if there's only one branch
|
||||||
|
if (totalBranches <= 1) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'flex items-center gap-2 self-end px-10',
|
||||||
|
from === 'assistant' ? 'justify-start' : 'justify-end',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type BranchPreviousProps = ComponentProps<typeof Button>;
|
||||||
|
|
||||||
|
export const BranchPrevious = ({
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: BranchPreviousProps) => {
|
||||||
|
const { goToPrevious, totalBranches } = useBranch();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
aria-label="Previous branch"
|
||||||
|
className={cn(
|
||||||
|
'size-7 shrink-0 rounded-full text-muted-foreground transition-colors',
|
||||||
|
'hover:bg-accent hover:text-foreground',
|
||||||
|
'disabled:pointer-events-none disabled:opacity-50',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
disabled={totalBranches <= 1}
|
||||||
|
onClick={goToPrevious}
|
||||||
|
size="icon"
|
||||||
|
type="button"
|
||||||
|
variant="ghost"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children ?? <ChevronLeftIcon size={14} />}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type BranchNextProps = ComponentProps<typeof Button>;
|
||||||
|
|
||||||
|
export const BranchNext = ({
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: BranchNextProps) => {
|
||||||
|
const { goToNext, totalBranches } = useBranch();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
aria-label="Next branch"
|
||||||
|
className={cn(
|
||||||
|
'size-7 shrink-0 rounded-full text-muted-foreground transition-colors',
|
||||||
|
'hover:bg-accent hover:text-foreground',
|
||||||
|
'disabled:pointer-events-none disabled:opacity-50',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
disabled={totalBranches <= 1}
|
||||||
|
onClick={goToNext}
|
||||||
|
size="icon"
|
||||||
|
type="button"
|
||||||
|
variant="ghost"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children ?? <ChevronRightIcon size={14} />}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type BranchPageProps = HTMLAttributes<HTMLSpanElement>;
|
||||||
|
|
||||||
|
export const BranchPage = ({ className, ...props }: BranchPageProps) => {
|
||||||
|
const { currentBranch, totalBranches } = useBranch();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
className={cn(
|
||||||
|
'font-medium text-muted-foreground text-xs tabular-nums',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{currentBranch + 1} of {totalBranches}
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
};
|
150
src/components/ai-elements/code-block.tsx
Normal file
150
src/components/ai-elements/code-block.tsx
Normal file
@ -0,0 +1,150 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { CheckIcon, CopyIcon } from 'lucide-react';
|
||||||
|
import type { ComponentProps, HTMLAttributes, ReactNode } from 'react';
|
||||||
|
import { createContext, useContext, useState } from 'react';
|
||||||
|
import { Prism as SyntaxHighlighter } from 'react-syntax-highlighter';
|
||||||
|
import {
|
||||||
|
oneDark,
|
||||||
|
oneLight,
|
||||||
|
} from 'react-syntax-highlighter/dist/esm/styles/prism';
|
||||||
|
|
||||||
|
type CodeBlockContextType = {
|
||||||
|
code: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const CodeBlockContext = createContext<CodeBlockContextType>({
|
||||||
|
code: '',
|
||||||
|
});
|
||||||
|
|
||||||
|
export type CodeBlockProps = HTMLAttributes<HTMLDivElement> & {
|
||||||
|
code: string;
|
||||||
|
language: string;
|
||||||
|
showLineNumbers?: boolean;
|
||||||
|
children?: ReactNode;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CodeBlock = ({
|
||||||
|
code,
|
||||||
|
language,
|
||||||
|
showLineNumbers = false,
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: CodeBlockProps) => (
|
||||||
|
<CodeBlockContext.Provider value={{ code }}>
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'relative w-full overflow-hidden rounded-md border bg-background text-foreground',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<div className="relative">
|
||||||
|
{/* @ts-expect-error - SyntaxHighlighter is not a valid JSX component */}
|
||||||
|
<SyntaxHighlighter
|
||||||
|
className="overflow-hidden dark:hidden"
|
||||||
|
codeTagProps={{
|
||||||
|
className: 'font-mono text-sm',
|
||||||
|
}}
|
||||||
|
customStyle={{
|
||||||
|
margin: 0,
|
||||||
|
padding: '1rem',
|
||||||
|
fontSize: '0.875rem',
|
||||||
|
background: 'hsl(var(--background))',
|
||||||
|
color: 'hsl(var(--foreground))',
|
||||||
|
}}
|
||||||
|
language={language}
|
||||||
|
lineNumberStyle={{
|
||||||
|
color: 'hsl(var(--muted-foreground))',
|
||||||
|
paddingRight: '1rem',
|
||||||
|
minWidth: '2.5rem',
|
||||||
|
}}
|
||||||
|
showLineNumbers={showLineNumbers}
|
||||||
|
style={oneLight}
|
||||||
|
>
|
||||||
|
{code}
|
||||||
|
</SyntaxHighlighter>
|
||||||
|
{/* @ts-expect-error - SyntaxHighlighter is not a valid JSX component */}
|
||||||
|
<SyntaxHighlighter
|
||||||
|
className="hidden overflow-hidden dark:block"
|
||||||
|
codeTagProps={{
|
||||||
|
className: 'font-mono text-sm',
|
||||||
|
}}
|
||||||
|
customStyle={{
|
||||||
|
margin: 0,
|
||||||
|
padding: '1rem',
|
||||||
|
fontSize: '0.875rem',
|
||||||
|
background: 'hsl(var(--background))',
|
||||||
|
color: 'hsl(var(--foreground))',
|
||||||
|
}}
|
||||||
|
language={language}
|
||||||
|
lineNumberStyle={{
|
||||||
|
color: 'hsl(var(--muted-foreground))',
|
||||||
|
paddingRight: '1rem',
|
||||||
|
minWidth: '2.5rem',
|
||||||
|
}}
|
||||||
|
showLineNumbers={showLineNumbers}
|
||||||
|
style={oneDark}
|
||||||
|
>
|
||||||
|
{code}
|
||||||
|
</SyntaxHighlighter>
|
||||||
|
{children && (
|
||||||
|
<div className="absolute top-2 right-2 flex items-center gap-2">
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</CodeBlockContext.Provider>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type CodeBlockCopyButtonProps = ComponentProps<typeof Button> & {
|
||||||
|
onCopy?: () => void;
|
||||||
|
onError?: (error: Error) => void;
|
||||||
|
timeout?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CodeBlockCopyButton = ({
|
||||||
|
onCopy,
|
||||||
|
onError,
|
||||||
|
timeout = 2000,
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: CodeBlockCopyButtonProps) => {
|
||||||
|
const [isCopied, setIsCopied] = useState(false);
|
||||||
|
const { code } = useContext(CodeBlockContext);
|
||||||
|
|
||||||
|
const copyToClipboard = async () => {
|
||||||
|
if (typeof window === 'undefined' || !navigator.clipboard.writeText) {
|
||||||
|
onError?.(new Error('Clipboard API not available'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await navigator.clipboard.writeText(code);
|
||||||
|
setIsCopied(true);
|
||||||
|
onCopy?.();
|
||||||
|
setTimeout(() => setIsCopied(false), timeout);
|
||||||
|
} catch (error) {
|
||||||
|
onError?.(error as Error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const Icon = isCopied ? CheckIcon : CopyIcon;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
className={cn('shrink-0', className)}
|
||||||
|
onClick={copyToClipboard}
|
||||||
|
size="icon"
|
||||||
|
variant="ghost"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children ?? <Icon size={14} />}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
};
|
62
src/components/ai-elements/conversation.tsx
Normal file
62
src/components/ai-elements/conversation.tsx
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { ArrowDownIcon } from 'lucide-react';
|
||||||
|
import type { ComponentProps } from 'react';
|
||||||
|
import { useCallback } from 'react';
|
||||||
|
import { StickToBottom, useStickToBottomContext } from 'use-stick-to-bottom';
|
||||||
|
|
||||||
|
export type ConversationProps = ComponentProps<typeof StickToBottom>;
|
||||||
|
|
||||||
|
export const Conversation = ({ className, ...props }: ConversationProps) => (
|
||||||
|
<StickToBottom
|
||||||
|
className={cn('relative flex-1 overflow-y-auto', className)}
|
||||||
|
initial="smooth"
|
||||||
|
resize="smooth"
|
||||||
|
role="log"
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ConversationContentProps = ComponentProps<
|
||||||
|
typeof StickToBottom.Content
|
||||||
|
>;
|
||||||
|
|
||||||
|
export const ConversationContent = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: ConversationContentProps) => (
|
||||||
|
<StickToBottom.Content className={cn('p-4', className)} {...props} />
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ConversationScrollButtonProps = ComponentProps<typeof Button>;
|
||||||
|
|
||||||
|
export const ConversationScrollButton = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: ConversationScrollButtonProps) => {
|
||||||
|
const { isAtBottom, scrollToBottom } = useStickToBottomContext();
|
||||||
|
|
||||||
|
const handleScrollToBottom = useCallback(() => {
|
||||||
|
scrollToBottom();
|
||||||
|
}, [scrollToBottom]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
!isAtBottom && (
|
||||||
|
<Button
|
||||||
|
className={cn(
|
||||||
|
'absolute bottom-4 left-[50%] translate-x-[-50%] rounded-full',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
onClick={handleScrollToBottom}
|
||||||
|
size="icon"
|
||||||
|
type="button"
|
||||||
|
variant="outline"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ArrowDownIcon className="size-4" />
|
||||||
|
</Button>
|
||||||
|
)
|
||||||
|
);
|
||||||
|
};
|
24
src/components/ai-elements/image.tsx
Normal file
24
src/components/ai-elements/image.tsx
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { Experimental_GeneratedImage } from 'ai';
|
||||||
|
|
||||||
|
export type ImageProps = Experimental_GeneratedImage & {
|
||||||
|
className?: string;
|
||||||
|
alt?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Image = ({
|
||||||
|
base64,
|
||||||
|
uint8Array,
|
||||||
|
mediaType,
|
||||||
|
...props
|
||||||
|
}: ImageProps) => (
|
||||||
|
<img
|
||||||
|
{...props}
|
||||||
|
alt={props.alt}
|
||||||
|
className={cn(
|
||||||
|
'h-auto max-w-full overflow-hidden rounded-md',
|
||||||
|
props.className
|
||||||
|
)}
|
||||||
|
src={`data:${mediaType};base64,${base64}`}
|
||||||
|
/>
|
||||||
|
);
|
287
src/components/ai-elements/inline-citation.tsx
Normal file
287
src/components/ai-elements/inline-citation.tsx
Normal file
@ -0,0 +1,287 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Badge } from '@/components/ui/badge';
|
||||||
|
import {
|
||||||
|
Carousel,
|
||||||
|
CarouselContent,
|
||||||
|
CarouselItem,
|
||||||
|
type CarouselApi,
|
||||||
|
} from '@/components/ui/carousel';
|
||||||
|
import {
|
||||||
|
HoverCard,
|
||||||
|
HoverCardContent,
|
||||||
|
HoverCardTrigger,
|
||||||
|
} from '@/components/ui/hover-card';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { ArrowLeftIcon, ArrowRightIcon } from 'lucide-react';
|
||||||
|
import {
|
||||||
|
type ComponentProps,
|
||||||
|
createContext,
|
||||||
|
useCallback,
|
||||||
|
useContext,
|
||||||
|
useEffect,
|
||||||
|
useState,
|
||||||
|
} from 'react';
|
||||||
|
|
||||||
|
export type InlineCitationProps = ComponentProps<'span'>;
|
||||||
|
|
||||||
|
export const InlineCitation = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationProps) => (
|
||||||
|
<span
|
||||||
|
className={cn('group inline items-center gap-1', className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type InlineCitationTextProps = ComponentProps<'span'>;
|
||||||
|
|
||||||
|
export const InlineCitationText = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationTextProps) => (
|
||||||
|
<span
|
||||||
|
className={cn('transition-colors group-hover:bg-accent', className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type InlineCitationCardProps = ComponentProps<typeof HoverCard>;
|
||||||
|
|
||||||
|
export const InlineCitationCard = (props: InlineCitationCardProps) => (
|
||||||
|
<HoverCard closeDelay={0} openDelay={0} {...props} />
|
||||||
|
);
|
||||||
|
|
||||||
|
export type InlineCitationCardTriggerProps = ComponentProps<typeof Badge> & {
|
||||||
|
sources: string[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const InlineCitationCardTrigger = ({
|
||||||
|
sources,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCardTriggerProps) => (
|
||||||
|
<HoverCardTrigger asChild>
|
||||||
|
<Badge
|
||||||
|
className={cn('ml-1 rounded-full', className)}
|
||||||
|
variant="secondary"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{sources.length ? (
|
||||||
|
<>
|
||||||
|
{new URL(sources[0]).hostname}{' '}
|
||||||
|
{sources.length > 1 && `+${sources.length - 1}`}
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
'unknown'
|
||||||
|
)}
|
||||||
|
</Badge>
|
||||||
|
</HoverCardTrigger>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type InlineCitationCardBodyProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const InlineCitationCardBody = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCardBodyProps) => (
|
||||||
|
<HoverCardContent className={cn('relative w-80 p-0', className)} {...props} />
|
||||||
|
);
|
||||||
|
|
||||||
|
const CarouselApiContext = createContext<CarouselApi | undefined>(undefined);
|
||||||
|
|
||||||
|
const useCarouselApi = () => {
|
||||||
|
const context = useContext(CarouselApiContext);
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type InlineCitationCarouselProps = ComponentProps<typeof Carousel>;
|
||||||
|
|
||||||
|
export const InlineCitationCarousel = ({
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCarouselProps) => {
|
||||||
|
const [api, setApi] = useState<CarouselApi>();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<CarouselApiContext.Provider value={api}>
|
||||||
|
<Carousel className={cn('w-full', className)} setApi={setApi} {...props}>
|
||||||
|
{children}
|
||||||
|
</Carousel>
|
||||||
|
</CarouselApiContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type InlineCitationCarouselContentProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const InlineCitationCarouselContent = (
|
||||||
|
props: InlineCitationCarouselContentProps
|
||||||
|
) => <CarouselContent {...props} />;
|
||||||
|
|
||||||
|
export type InlineCitationCarouselItemProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const InlineCitationCarouselItem = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCarouselItemProps) => (
|
||||||
|
<CarouselItem
|
||||||
|
className={cn('w-full space-y-2 p-4 pl-8', className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type InlineCitationCarouselHeaderProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const InlineCitationCarouselHeader = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCarouselHeaderProps) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'flex items-center justify-between gap-2 rounded-t-md bg-secondary p-2',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type InlineCitationCarouselIndexProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const InlineCitationCarouselIndex = ({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCarouselIndexProps) => {
|
||||||
|
const api = useCarouselApi();
|
||||||
|
const [current, setCurrent] = useState(0);
|
||||||
|
const [count, setCount] = useState(0);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!api) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setCount(api.scrollSnapList().length);
|
||||||
|
setCurrent(api.selectedScrollSnap() + 1);
|
||||||
|
|
||||||
|
api.on('select', () => {
|
||||||
|
setCurrent(api.selectedScrollSnap() + 1);
|
||||||
|
});
|
||||||
|
}, [api]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'flex flex-1 items-center justify-end px-3 py-1 text-muted-foreground text-xs',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children ?? `${current}/${count}`}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type InlineCitationCarouselPrevProps = ComponentProps<'button'>;
|
||||||
|
|
||||||
|
export const InlineCitationCarouselPrev = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCarouselPrevProps) => {
|
||||||
|
const api = useCarouselApi();
|
||||||
|
|
||||||
|
const handleClick = useCallback(() => {
|
||||||
|
if (api) {
|
||||||
|
api.scrollPrev();
|
||||||
|
}
|
||||||
|
}, [api]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
aria-label="Previous"
|
||||||
|
className={cn('shrink-0', className)}
|
||||||
|
onClick={handleClick}
|
||||||
|
type="button"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ArrowLeftIcon className="size-4 text-muted-foreground" />
|
||||||
|
</button>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type InlineCitationCarouselNextProps = ComponentProps<'button'>;
|
||||||
|
|
||||||
|
export const InlineCitationCarouselNext = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationCarouselNextProps) => {
|
||||||
|
const api = useCarouselApi();
|
||||||
|
|
||||||
|
const handleClick = useCallback(() => {
|
||||||
|
if (api) {
|
||||||
|
api.scrollNext();
|
||||||
|
}
|
||||||
|
}, [api]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
aria-label="Next"
|
||||||
|
className={cn('shrink-0', className)}
|
||||||
|
onClick={handleClick}
|
||||||
|
type="button"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ArrowRightIcon className="size-4 text-muted-foreground" />
|
||||||
|
</button>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type InlineCitationSourceProps = ComponentProps<'div'> & {
|
||||||
|
title?: string;
|
||||||
|
url?: string;
|
||||||
|
description?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const InlineCitationSource = ({
|
||||||
|
title,
|
||||||
|
url,
|
||||||
|
description,
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: InlineCitationSourceProps) => (
|
||||||
|
<div className={cn('space-y-1', className)} {...props}>
|
||||||
|
{title && (
|
||||||
|
<h4 className="truncate font-medium text-sm leading-tight">{title}</h4>
|
||||||
|
)}
|
||||||
|
{url && (
|
||||||
|
<p className="truncate break-all text-muted-foreground text-xs">{url}</p>
|
||||||
|
)}
|
||||||
|
{description && (
|
||||||
|
<p className="line-clamp-3 text-muted-foreground text-sm leading-relaxed">
|
||||||
|
{description}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type InlineCitationQuoteProps = ComponentProps<'blockquote'>;
|
||||||
|
|
||||||
|
export const InlineCitationQuote = ({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: InlineCitationQuoteProps) => (
|
||||||
|
<blockquote
|
||||||
|
className={cn(
|
||||||
|
'border-muted border-l-2 pl-3 text-muted-foreground text-sm italic',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</blockquote>
|
||||||
|
);
|
96
src/components/ai-elements/loader.tsx
Normal file
96
src/components/ai-elements/loader.tsx
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { HTMLAttributes } from 'react';
|
||||||
|
|
||||||
|
type LoaderIconProps = {
|
||||||
|
size?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
const LoaderIcon = ({ size = 16 }: LoaderIconProps) => (
|
||||||
|
<svg
|
||||||
|
height={size}
|
||||||
|
strokeLinejoin="round"
|
||||||
|
style={{ color: 'currentcolor' }}
|
||||||
|
viewBox="0 0 16 16"
|
||||||
|
width={size}
|
||||||
|
>
|
||||||
|
<title>Loader</title>
|
||||||
|
<g clipPath="url(#clip0_2393_1490)">
|
||||||
|
<path d="M8 0V4" stroke="currentColor" strokeWidth="1.5" />
|
||||||
|
<path
|
||||||
|
d="M8 16V12"
|
||||||
|
opacity="0.5"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M3.29773 1.52783L5.64887 4.7639"
|
||||||
|
opacity="0.9"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M12.7023 1.52783L10.3511 4.7639"
|
||||||
|
opacity="0.1"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M12.7023 14.472L10.3511 11.236"
|
||||||
|
opacity="0.4"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M3.29773 14.472L5.64887 11.236"
|
||||||
|
opacity="0.6"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M15.6085 5.52783L11.8043 6.7639"
|
||||||
|
opacity="0.2"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M0.391602 10.472L4.19583 9.23598"
|
||||||
|
opacity="0.7"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M15.6085 10.4722L11.8043 9.2361"
|
||||||
|
opacity="0.3"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M0.391602 5.52783L4.19583 6.7639"
|
||||||
|
opacity="0.8"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="1.5"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
<defs>
|
||||||
|
<clipPath id="clip0_2393_1490">
|
||||||
|
<rect fill="white" height="16" width="16" />
|
||||||
|
</clipPath>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type LoaderProps = HTMLAttributes<HTMLDivElement> & {
|
||||||
|
size?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Loader = ({ className, size = 16, ...props }: LoaderProps) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'inline-flex animate-spin items-center justify-center',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<LoaderIcon size={size} />
|
||||||
|
</div>
|
||||||
|
);
|
64
src/components/ai-elements/message.tsx
Normal file
64
src/components/ai-elements/message.tsx
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
import {
|
||||||
|
Avatar,
|
||||||
|
AvatarFallback,
|
||||||
|
AvatarImage,
|
||||||
|
} from '@/components/ui/avatar';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { UIMessage } from 'ai';
|
||||||
|
import type { ComponentProps, HTMLAttributes } from 'react';
|
||||||
|
|
||||||
|
export type MessageProps = HTMLAttributes<HTMLDivElement> & {
|
||||||
|
from: UIMessage['role'];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Message = ({ className, from, ...props }: MessageProps) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'group flex w-full items-end justify-end gap-2 py-4',
|
||||||
|
from === 'user' ? 'is-user' : 'is-assistant flex-row-reverse justify-end',
|
||||||
|
'[&>div]:max-w-[80%]',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type MessageContentProps = HTMLAttributes<HTMLDivElement>;
|
||||||
|
|
||||||
|
export const MessageContent = ({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: MessageContentProps) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'flex flex-col gap-2 overflow-hidden rounded-lg px-4 py-3 text-foreground text-sm',
|
||||||
|
'group-[.is-user]:bg-primary group-[.is-user]:text-primary-foreground',
|
||||||
|
'group-[.is-assistant]:bg-secondary group-[.is-assistant]:text-foreground',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<div className="is-user:dark">{children}</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type MessageAvatarProps = ComponentProps<typeof Avatar> & {
|
||||||
|
src: string;
|
||||||
|
name?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const MessageAvatar = ({
|
||||||
|
src,
|
||||||
|
name,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: MessageAvatarProps) => (
|
||||||
|
<Avatar
|
||||||
|
className={cn('size-8 ring ring-1 ring-border', className)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<AvatarImage alt="" className="mt-0 mb-0" src={src} />
|
||||||
|
<AvatarFallback>{name?.slice(0, 2) || 'ME'}</AvatarFallback>
|
||||||
|
</Avatar>
|
||||||
|
);
|
230
src/components/ai-elements/prompt-input.tsx
Normal file
230
src/components/ai-elements/prompt-input.tsx
Normal file
@ -0,0 +1,230 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import {
|
||||||
|
Select,
|
||||||
|
SelectContent,
|
||||||
|
SelectItem,
|
||||||
|
SelectTrigger,
|
||||||
|
SelectValue,
|
||||||
|
} from '@/components/ui/select';
|
||||||
|
import { Textarea } from '@/components/ui/textarea';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { ChatStatus } from 'ai';
|
||||||
|
import { Loader2Icon, SendIcon, SquareIcon, XIcon } from 'lucide-react';
|
||||||
|
import type {
|
||||||
|
ComponentProps,
|
||||||
|
HTMLAttributes,
|
||||||
|
KeyboardEventHandler,
|
||||||
|
} from 'react';
|
||||||
|
import { Children } from 'react';
|
||||||
|
|
||||||
|
export type PromptInputProps = HTMLAttributes<HTMLFormElement>;
|
||||||
|
|
||||||
|
export const PromptInput = ({ className, ...props }: PromptInputProps) => (
|
||||||
|
<form
|
||||||
|
className={cn(
|
||||||
|
'w-full divide-y overflow-hidden rounded-xl border bg-background shadow-sm',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type PromptInputTextareaProps = ComponentProps<typeof Textarea> & {
|
||||||
|
minHeight?: number;
|
||||||
|
maxHeight?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PromptInputTextarea = ({
|
||||||
|
onChange,
|
||||||
|
className,
|
||||||
|
placeholder = 'What would you like to know?',
|
||||||
|
minHeight = 48,
|
||||||
|
maxHeight = 164,
|
||||||
|
...props
|
||||||
|
}: PromptInputTextareaProps) => {
|
||||||
|
const handleKeyDown: KeyboardEventHandler<HTMLTextAreaElement> = (e) => {
|
||||||
|
if (e.key === 'Enter') {
|
||||||
|
// Don't submit if IME composition is in progress
|
||||||
|
if (e.nativeEvent.isComposing) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.shiftKey) {
|
||||||
|
// Allow newline
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Submit on Enter (without Shift)
|
||||||
|
e.preventDefault();
|
||||||
|
const form = e.currentTarget.form;
|
||||||
|
if (form) {
|
||||||
|
form.requestSubmit();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Textarea
|
||||||
|
className={cn(
|
||||||
|
'w-full resize-none rounded-none border-none p-3 shadow-none outline-none ring-0',
|
||||||
|
'field-sizing-content max-h-[6lh] bg-transparent dark:bg-transparent',
|
||||||
|
'focus-visible:ring-0',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
name="message"
|
||||||
|
onChange={(e) => {
|
||||||
|
onChange?.(e);
|
||||||
|
}}
|
||||||
|
onKeyDown={handleKeyDown}
|
||||||
|
placeholder={placeholder}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type PromptInputToolbarProps = HTMLAttributes<HTMLDivElement>;
|
||||||
|
|
||||||
|
export const PromptInputToolbar = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: PromptInputToolbarProps) => (
|
||||||
|
<div
|
||||||
|
className={cn('flex items-center justify-between p-1', className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type PromptInputToolsProps = HTMLAttributes<HTMLDivElement>;
|
||||||
|
|
||||||
|
export const PromptInputTools = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: PromptInputToolsProps) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'flex items-center gap-1',
|
||||||
|
'[&_button:first-child]:rounded-bl-xl',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type PromptInputButtonProps = ComponentProps<typeof Button>;
|
||||||
|
|
||||||
|
export const PromptInputButton = ({
|
||||||
|
variant = 'ghost',
|
||||||
|
className,
|
||||||
|
size,
|
||||||
|
...props
|
||||||
|
}: PromptInputButtonProps) => {
|
||||||
|
const newSize =
|
||||||
|
(size ?? Children.count(props.children) > 1) ? 'default' : 'icon';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
className={cn(
|
||||||
|
'shrink-0 gap-1.5 rounded-lg',
|
||||||
|
variant === 'ghost' && 'text-muted-foreground',
|
||||||
|
newSize === 'default' && 'px-3',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
size={newSize}
|
||||||
|
type="button"
|
||||||
|
variant={variant}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type PromptInputSubmitProps = ComponentProps<typeof Button> & {
|
||||||
|
status?: ChatStatus;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PromptInputSubmit = ({
|
||||||
|
className,
|
||||||
|
variant = 'default',
|
||||||
|
size = 'icon',
|
||||||
|
status,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: PromptInputSubmitProps) => {
|
||||||
|
let Icon = <SendIcon className="size-4" />;
|
||||||
|
|
||||||
|
if (status === 'submitted') {
|
||||||
|
Icon = <Loader2Icon className="size-4 animate-spin" />;
|
||||||
|
} else if (status === 'streaming') {
|
||||||
|
Icon = <SquareIcon className="size-4" />;
|
||||||
|
} else if (status === 'error') {
|
||||||
|
Icon = <XIcon className="size-4" />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
className={cn('gap-1.5 rounded-lg', className)}
|
||||||
|
size={size}
|
||||||
|
type="submit"
|
||||||
|
variant={variant}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children ?? Icon}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type PromptInputModelSelectProps = ComponentProps<typeof Select>;
|
||||||
|
|
||||||
|
export const PromptInputModelSelect = (props: PromptInputModelSelectProps) => (
|
||||||
|
<Select {...props} />
|
||||||
|
);
|
||||||
|
|
||||||
|
export type PromptInputModelSelectTriggerProps = ComponentProps<
|
||||||
|
typeof SelectTrigger
|
||||||
|
>;
|
||||||
|
|
||||||
|
export const PromptInputModelSelectTrigger = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: PromptInputModelSelectTriggerProps) => (
|
||||||
|
<SelectTrigger
|
||||||
|
className={cn(
|
||||||
|
'border-none bg-transparent font-medium text-muted-foreground shadow-none transition-colors',
|
||||||
|
'hover:bg-accent hover:text-foreground [&[aria-expanded="true"]]:bg-accent [&[aria-expanded="true"]]:text-foreground',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type PromptInputModelSelectContentProps = ComponentProps<
|
||||||
|
typeof SelectContent
|
||||||
|
>;
|
||||||
|
|
||||||
|
export const PromptInputModelSelectContent = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: PromptInputModelSelectContentProps) => (
|
||||||
|
<SelectContent className={cn(className)} {...props} />
|
||||||
|
);
|
||||||
|
|
||||||
|
export type PromptInputModelSelectItemProps = ComponentProps<typeof SelectItem>;
|
||||||
|
|
||||||
|
export const PromptInputModelSelectItem = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: PromptInputModelSelectItemProps) => (
|
||||||
|
<SelectItem className={cn(className)} {...props} />
|
||||||
|
);
|
||||||
|
|
||||||
|
export type PromptInputModelSelectValueProps = ComponentProps<
|
||||||
|
typeof SelectValue
|
||||||
|
>;
|
||||||
|
|
||||||
|
export const PromptInputModelSelectValue = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: PromptInputModelSelectValueProps) => (
|
||||||
|
<SelectValue className={cn(className)} {...props} />
|
||||||
|
);
|
180
src/components/ai-elements/reasoning.tsx
Normal file
180
src/components/ai-elements/reasoning.tsx
Normal file
@ -0,0 +1,180 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useControllableState } from '@radix-ui/react-use-controllable-state';
|
||||||
|
import {
|
||||||
|
Collapsible,
|
||||||
|
CollapsibleContent,
|
||||||
|
CollapsibleTrigger,
|
||||||
|
} from '@/components/ui/collapsible';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { BrainIcon, ChevronDownIcon } from 'lucide-react';
|
||||||
|
import type { ComponentProps } from 'react';
|
||||||
|
import { createContext, memo, useContext, useEffect, useState } from 'react';
|
||||||
|
import { Response } from './response';
|
||||||
|
|
||||||
|
type ReasoningContextValue = {
|
||||||
|
isStreaming: boolean;
|
||||||
|
isOpen: boolean;
|
||||||
|
setIsOpen: (open: boolean) => void;
|
||||||
|
duration: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
const ReasoningContext = createContext<ReasoningContextValue | null>(null);
|
||||||
|
|
||||||
|
const useReasoning = () => {
|
||||||
|
const context = useContext(ReasoningContext);
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('Reasoning components must be used within Reasoning');
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ReasoningProps = ComponentProps<typeof Collapsible> & {
|
||||||
|
isStreaming?: boolean;
|
||||||
|
open?: boolean;
|
||||||
|
defaultOpen?: boolean;
|
||||||
|
onOpenChange?: (open: boolean) => void;
|
||||||
|
duration?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
const AUTO_CLOSE_DELAY = 1000;
|
||||||
|
|
||||||
|
export const Reasoning = memo(
|
||||||
|
({
|
||||||
|
className,
|
||||||
|
isStreaming = false,
|
||||||
|
open,
|
||||||
|
defaultOpen = false,
|
||||||
|
onOpenChange,
|
||||||
|
duration: durationProp,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: ReasoningProps) => {
|
||||||
|
const [isOpen, setIsOpen] = useControllableState({
|
||||||
|
prop: open,
|
||||||
|
defaultProp: defaultOpen,
|
||||||
|
onChange: onOpenChange,
|
||||||
|
});
|
||||||
|
const [duration, setDuration] = useControllableState({
|
||||||
|
prop: durationProp,
|
||||||
|
defaultProp: 0,
|
||||||
|
});
|
||||||
|
|
||||||
|
const [hasAutoClosedRef, setHasAutoClosedRef] = useState(false);
|
||||||
|
const [startTime, setStartTime] = useState<number | null>(null);
|
||||||
|
|
||||||
|
// Track duration when streaming starts and ends
|
||||||
|
useEffect(() => {
|
||||||
|
if (isStreaming) {
|
||||||
|
if (startTime === null) {
|
||||||
|
setStartTime(Date.now());
|
||||||
|
}
|
||||||
|
} else if (startTime !== null) {
|
||||||
|
setDuration(Math.round((Date.now() - startTime) / 1000));
|
||||||
|
setStartTime(null);
|
||||||
|
}
|
||||||
|
}, [isStreaming, startTime, setDuration]);
|
||||||
|
|
||||||
|
// Auto-open when streaming starts, auto-close when streaming ends (once only)
|
||||||
|
useEffect(() => {
|
||||||
|
if (isStreaming && !isOpen) {
|
||||||
|
setIsOpen(true);
|
||||||
|
} else if (!isStreaming && isOpen && !defaultOpen && !hasAutoClosedRef) {
|
||||||
|
// Add a small delay before closing to allow user to see the content
|
||||||
|
const timer = setTimeout(() => {
|
||||||
|
setIsOpen(false);
|
||||||
|
setHasAutoClosedRef(true);
|
||||||
|
}, AUTO_CLOSE_DELAY);
|
||||||
|
return () => clearTimeout(timer);
|
||||||
|
}
|
||||||
|
}, [isStreaming, isOpen, defaultOpen, setIsOpen, hasAutoClosedRef]);
|
||||||
|
|
||||||
|
const handleOpenChange = (newOpen: boolean) => {
|
||||||
|
setIsOpen(newOpen);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ReasoningContext.Provider
|
||||||
|
value={{ isStreaming, isOpen, setIsOpen, duration }}
|
||||||
|
>
|
||||||
|
<Collapsible
|
||||||
|
className={cn('not-prose mb-4', className)}
|
||||||
|
onOpenChange={handleOpenChange}
|
||||||
|
open={isOpen}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Collapsible>
|
||||||
|
</ReasoningContext.Provider>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ReasoningTriggerProps = ComponentProps<
|
||||||
|
typeof CollapsibleTrigger
|
||||||
|
> & {
|
||||||
|
title?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ReasoningTrigger = memo(
|
||||||
|
({
|
||||||
|
className,
|
||||||
|
title = 'Reasoning',
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: ReasoningTriggerProps) => {
|
||||||
|
const { isStreaming, isOpen, duration } = useReasoning();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<CollapsibleTrigger
|
||||||
|
className={cn(
|
||||||
|
'flex items-center gap-2 text-muted-foreground text-sm',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children ?? (
|
||||||
|
<>
|
||||||
|
<BrainIcon className="size-4" />
|
||||||
|
{isStreaming || duration === 0 ? (
|
||||||
|
<p>Thinking...</p>
|
||||||
|
) : (
|
||||||
|
<p>Thought for {duration} seconds</p>
|
||||||
|
)}
|
||||||
|
<ChevronDownIcon
|
||||||
|
className={cn(
|
||||||
|
'size-4 text-muted-foreground transition-transform',
|
||||||
|
isOpen ? 'rotate-180' : 'rotate-0'
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</CollapsibleTrigger>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ReasoningContentProps = ComponentProps<
|
||||||
|
typeof CollapsibleContent
|
||||||
|
> & {
|
||||||
|
children: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ReasoningContent = memo(
|
||||||
|
({ className, children, ...props }: ReasoningContentProps) => (
|
||||||
|
<CollapsibleContent
|
||||||
|
className={cn(
|
||||||
|
'mt-4 text-sm',
|
||||||
|
'data-[state=closed]:fade-out-0 data-[state=closed]:slide-out-to-top-2 data-[state=open]:slide-in-from-top-2 text-popover-foreground outline-none data-[state=closed]:animate-out data-[state=open]:animate-in',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<Response className="grid gap-2">{children}</Response>
|
||||||
|
</CollapsibleContent>
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
Reasoning.displayName = 'Reasoning';
|
||||||
|
ReasoningTrigger.displayName = 'ReasoningTrigger';
|
||||||
|
ReasoningContent.displayName = 'ReasoningContent';
|
22
src/components/ai-elements/response.tsx
Normal file
22
src/components/ai-elements/response.tsx
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { type ComponentProps, memo } from 'react';
|
||||||
|
import { Streamdown } from 'streamdown';
|
||||||
|
|
||||||
|
type ResponseProps = ComponentProps<typeof Streamdown>;
|
||||||
|
|
||||||
|
export const Response = memo(
|
||||||
|
({ className, ...props }: ResponseProps) => (
|
||||||
|
<Streamdown
|
||||||
|
className={cn(
|
||||||
|
'size-full [&>*:first-child]:mt-0 [&>*:last-child]:mb-0',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
),
|
||||||
|
(prevProps, nextProps) => prevProps.children === nextProps.children
|
||||||
|
);
|
||||||
|
|
||||||
|
Response.displayName = 'Response';
|
74
src/components/ai-elements/source.tsx
Normal file
74
src/components/ai-elements/source.tsx
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import {
|
||||||
|
Collapsible,
|
||||||
|
CollapsibleContent,
|
||||||
|
CollapsibleTrigger,
|
||||||
|
} from '@/components/ui/collapsible';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { BookIcon, ChevronDownIcon } from 'lucide-react';
|
||||||
|
import type { ComponentProps } from 'react';
|
||||||
|
|
||||||
|
export type SourcesProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const Sources = ({ className, ...props }: SourcesProps) => (
|
||||||
|
<Collapsible
|
||||||
|
className={cn('not-prose mb-4 text-primary text-xs', className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type SourcesTriggerProps = ComponentProps<typeof CollapsibleTrigger> & {
|
||||||
|
count: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SourcesTrigger = ({
|
||||||
|
className,
|
||||||
|
count,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: SourcesTriggerProps) => (
|
||||||
|
<CollapsibleTrigger className="flex items-center gap-2" {...props}>
|
||||||
|
{children ?? (
|
||||||
|
<>
|
||||||
|
<p className="font-medium">Used {count} sources</p>
|
||||||
|
<ChevronDownIcon className="h-4 w-4" />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</CollapsibleTrigger>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type SourcesContentProps = ComponentProps<typeof CollapsibleContent>;
|
||||||
|
|
||||||
|
export const SourcesContent = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: SourcesContentProps) => (
|
||||||
|
<CollapsibleContent
|
||||||
|
className={cn(
|
||||||
|
'mt-3 flex w-fit flex-col gap-2',
|
||||||
|
'data-[state=closed]:fade-out-0 data-[state=closed]:slide-out-to-top-2 data-[state=open]:slide-in-from-top-2 outline-none data-[state=closed]:animate-out data-[state=open]:animate-in',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type SourceProps = ComponentProps<'a'>;
|
||||||
|
|
||||||
|
export const Source = ({ href, title, children, ...props }: SourceProps) => (
|
||||||
|
<a
|
||||||
|
className="flex items-center gap-2"
|
||||||
|
href={href}
|
||||||
|
rel="noreferrer"
|
||||||
|
target="_blank"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children ?? (
|
||||||
|
<>
|
||||||
|
<BookIcon className="h-4 w-4" />
|
||||||
|
<span className="block font-medium">{title}</span>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</a>
|
||||||
|
);
|
56
src/components/ai-elements/suggestion.tsx
Normal file
56
src/components/ai-elements/suggestion.tsx
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import {
|
||||||
|
ScrollArea,
|
||||||
|
ScrollBar,
|
||||||
|
} from '@/components/ui/scroll-area';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { ComponentProps } from 'react';
|
||||||
|
|
||||||
|
export type SuggestionsProps = ComponentProps<typeof ScrollArea>;
|
||||||
|
|
||||||
|
export const Suggestions = ({
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: SuggestionsProps) => (
|
||||||
|
<ScrollArea className="w-full overflow-x-auto whitespace-nowrap" {...props}>
|
||||||
|
<div className={cn('flex w-max flex-nowrap items-center gap-2', className)}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
<ScrollBar className="hidden" orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type SuggestionProps = Omit<ComponentProps<typeof Button>, 'onClick'> & {
|
||||||
|
suggestion: string;
|
||||||
|
onClick?: (suggestion: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Suggestion = ({
|
||||||
|
suggestion,
|
||||||
|
onClick,
|
||||||
|
className,
|
||||||
|
variant = 'outline',
|
||||||
|
size = 'sm',
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: SuggestionProps) => {
|
||||||
|
const handleClick = () => {
|
||||||
|
onClick?.(suggestion);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
className={cn('cursor-pointer rounded-full px-4', className)}
|
||||||
|
onClick={handleClick}
|
||||||
|
size={size}
|
||||||
|
type="button"
|
||||||
|
variant={variant}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children || suggestion}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
};
|
94
src/components/ai-elements/task.tsx
Normal file
94
src/components/ai-elements/task.tsx
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import {
|
||||||
|
Collapsible,
|
||||||
|
CollapsibleContent,
|
||||||
|
CollapsibleTrigger,
|
||||||
|
} from '@/components/ui/collapsible';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { ChevronDownIcon, SearchIcon } from 'lucide-react';
|
||||||
|
import type { ComponentProps } from 'react';
|
||||||
|
|
||||||
|
export type TaskItemFileProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const TaskItemFile = ({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: TaskItemFileProps) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'inline-flex items-center gap-1 rounded-md border bg-secondary px-1.5 py-0.5 text-foreground text-xs',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type TaskItemProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const TaskItem = ({ children, className, ...props }: TaskItemProps) => (
|
||||||
|
<div className={cn('text-muted-foreground text-sm', className)} {...props}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type TaskProps = ComponentProps<typeof Collapsible>;
|
||||||
|
|
||||||
|
export const Task = ({
|
||||||
|
defaultOpen = true,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: TaskProps) => (
|
||||||
|
<Collapsible
|
||||||
|
className={cn(
|
||||||
|
'data-[state=closed]:fade-out-0 data-[state=closed]:slide-out-to-top-2 data-[state=open]:slide-in-from-top-2 data-[state=closed]:animate-out data-[state=open]:animate-in',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
defaultOpen={defaultOpen}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type TaskTriggerProps = ComponentProps<typeof CollapsibleTrigger> & {
|
||||||
|
title: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const TaskTrigger = ({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
title,
|
||||||
|
...props
|
||||||
|
}: TaskTriggerProps) => (
|
||||||
|
<CollapsibleTrigger asChild className={cn('group', className)} {...props}>
|
||||||
|
{children ?? (
|
||||||
|
<div className="flex cursor-pointer items-center gap-2 text-muted-foreground hover:text-foreground">
|
||||||
|
<SearchIcon className="size-4" />
|
||||||
|
<p className="text-sm">{title}</p>
|
||||||
|
<ChevronDownIcon className="size-4 transition-transform group-data-[state=open]:rotate-180" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</CollapsibleTrigger>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type TaskContentProps = ComponentProps<typeof CollapsibleContent>;
|
||||||
|
|
||||||
|
export const TaskContent = ({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: TaskContentProps) => (
|
||||||
|
<CollapsibleContent
|
||||||
|
className={cn(
|
||||||
|
'data-[state=closed]:fade-out-0 data-[state=closed]:slide-out-to-top-2 data-[state=open]:slide-in-from-top-2 text-popover-foreground outline-none data-[state=closed]:animate-out data-[state=open]:animate-in',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<div className="mt-4 space-y-2 border-muted border-l-2 pl-4">
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</CollapsibleContent>
|
||||||
|
);
|
142
src/components/ai-elements/tool.tsx
Normal file
142
src/components/ai-elements/tool.tsx
Normal file
@ -0,0 +1,142 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Badge } from '@/components/ui/badge';
|
||||||
|
import {
|
||||||
|
Collapsible,
|
||||||
|
CollapsibleContent,
|
||||||
|
CollapsibleTrigger,
|
||||||
|
} from '@/components/ui/collapsible';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import type { ToolUIPart } from 'ai';
|
||||||
|
import {
|
||||||
|
CheckCircleIcon,
|
||||||
|
ChevronDownIcon,
|
||||||
|
CircleIcon,
|
||||||
|
ClockIcon,
|
||||||
|
WrenchIcon,
|
||||||
|
XCircleIcon,
|
||||||
|
} from 'lucide-react';
|
||||||
|
import type { ComponentProps, ReactNode } from 'react';
|
||||||
|
import { CodeBlock } from './code-block';
|
||||||
|
|
||||||
|
export type ToolProps = ComponentProps<typeof Collapsible>;
|
||||||
|
|
||||||
|
export const Tool = ({ className, ...props }: ToolProps) => (
|
||||||
|
<Collapsible
|
||||||
|
className={cn('not-prose mb-4 w-full rounded-md border', className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ToolHeaderProps = {
|
||||||
|
type: ToolUIPart['type'];
|
||||||
|
state: ToolUIPart['state'];
|
||||||
|
className?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getStatusBadge = (status: ToolUIPart['state']) => {
|
||||||
|
const labels = {
|
||||||
|
'input-streaming': 'Pending',
|
||||||
|
'input-available': 'Running',
|
||||||
|
'output-available': 'Completed',
|
||||||
|
'output-error': 'Error',
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
const icons = {
|
||||||
|
'input-streaming': <CircleIcon className="size-4" />,
|
||||||
|
'input-available': <ClockIcon className="size-4 animate-pulse" />,
|
||||||
|
'output-available': <CheckCircleIcon className="size-4 text-green-600" />,
|
||||||
|
'output-error': <XCircleIcon className="size-4 text-red-600" />,
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Badge className="rounded-full text-xs" variant="secondary">
|
||||||
|
{icons[status]}
|
||||||
|
{labels[status]}
|
||||||
|
</Badge>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ToolHeader = ({
|
||||||
|
className,
|
||||||
|
type,
|
||||||
|
state,
|
||||||
|
...props
|
||||||
|
}: ToolHeaderProps) => (
|
||||||
|
<CollapsibleTrigger
|
||||||
|
className={cn(
|
||||||
|
'flex w-full items-center justify-between gap-4 p-3',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<WrenchIcon className="size-4 text-muted-foreground" />
|
||||||
|
<span className="font-medium text-sm">{type}</span>
|
||||||
|
{getStatusBadge(state)}
|
||||||
|
</div>
|
||||||
|
<ChevronDownIcon className="size-4 text-muted-foreground transition-transform group-data-[state=open]:rotate-180" />
|
||||||
|
</CollapsibleTrigger>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ToolContentProps = ComponentProps<typeof CollapsibleContent>;
|
||||||
|
|
||||||
|
export const ToolContent = ({ className, ...props }: ToolContentProps) => (
|
||||||
|
<CollapsibleContent
|
||||||
|
className={cn(
|
||||||
|
'data-[state=closed]:fade-out-0 data-[state=closed]:slide-out-to-top-2 data-[state=open]:slide-in-from-top-2 text-popover-foreground outline-none data-[state=closed]:animate-out data-[state=open]:animate-in',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ToolInputProps = ComponentProps<'div'> & {
|
||||||
|
input: ToolUIPart['input'];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ToolInput = ({ className, input, ...props }: ToolInputProps) => (
|
||||||
|
<div className={cn('space-y-2 overflow-hidden p-4', className)} {...props}>
|
||||||
|
<h4 className="font-medium text-muted-foreground text-xs uppercase tracking-wide">
|
||||||
|
Parameters
|
||||||
|
</h4>
|
||||||
|
<div className="rounded-md bg-muted/50">
|
||||||
|
<CodeBlock code={JSON.stringify(input, null, 2)} language="json" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type ToolOutputProps = ComponentProps<'div'> & {
|
||||||
|
output: ReactNode;
|
||||||
|
errorText: ToolUIPart['errorText'];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ToolOutput = ({
|
||||||
|
className,
|
||||||
|
output,
|
||||||
|
errorText,
|
||||||
|
...props
|
||||||
|
}: ToolOutputProps) => {
|
||||||
|
if (!(output || errorText)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('space-y-2 p-4', className)} {...props}>
|
||||||
|
<h4 className="font-medium text-muted-foreground text-xs uppercase tracking-wide">
|
||||||
|
{errorText ? 'Error' : 'Result'}
|
||||||
|
</h4>
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'overflow-x-auto rounded-md text-xs [&_table]:w-full',
|
||||||
|
errorText
|
||||||
|
? 'bg-destructive/10 text-destructive'
|
||||||
|
: 'bg-muted/50 text-foreground'
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{errorText && <div>{errorText}</div>}
|
||||||
|
{output && <div>{output}</div>}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
252
src/components/ai-elements/web-preview.tsx
Normal file
252
src/components/ai-elements/web-preview.tsx
Normal file
@ -0,0 +1,252 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import {
|
||||||
|
Collapsible,
|
||||||
|
CollapsibleContent,
|
||||||
|
CollapsibleTrigger,
|
||||||
|
} from '@/components/ui/collapsible';
|
||||||
|
import { Input } from '@/components/ui/input';
|
||||||
|
import {
|
||||||
|
Tooltip,
|
||||||
|
TooltipContent,
|
||||||
|
TooltipProvider,
|
||||||
|
TooltipTrigger,
|
||||||
|
} from '@/components/ui/tooltip';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { ChevronDownIcon } from 'lucide-react';
|
||||||
|
import type { ComponentProps, ReactNode } from 'react';
|
||||||
|
import { createContext, useContext, useState } from 'react';
|
||||||
|
|
||||||
|
export type WebPreviewContextValue = {
|
||||||
|
url: string;
|
||||||
|
setUrl: (url: string) => void;
|
||||||
|
consoleOpen: boolean;
|
||||||
|
setConsoleOpen: (open: boolean) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const WebPreviewContext = createContext<WebPreviewContextValue | null>(null);
|
||||||
|
|
||||||
|
const useWebPreview = () => {
|
||||||
|
const context = useContext(WebPreviewContext);
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('WebPreview components must be used within a WebPreview');
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WebPreviewProps = ComponentProps<'div'> & {
|
||||||
|
defaultUrl?: string;
|
||||||
|
onUrlChange?: (url: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WebPreview = ({
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
defaultUrl = '',
|
||||||
|
onUrlChange,
|
||||||
|
...props
|
||||||
|
}: WebPreviewProps) => {
|
||||||
|
const [url, setUrl] = useState(defaultUrl);
|
||||||
|
const [consoleOpen, setConsoleOpen] = useState(false);
|
||||||
|
|
||||||
|
const handleUrlChange = (newUrl: string) => {
|
||||||
|
setUrl(newUrl);
|
||||||
|
onUrlChange?.(newUrl);
|
||||||
|
};
|
||||||
|
|
||||||
|
const contextValue: WebPreviewContextValue = {
|
||||||
|
url,
|
||||||
|
setUrl: handleUrlChange,
|
||||||
|
consoleOpen,
|
||||||
|
setConsoleOpen,
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WebPreviewContext.Provider value={contextValue}>
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'flex size-full flex-col rounded-lg border bg-card',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</WebPreviewContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WebPreviewNavigationProps = ComponentProps<'div'>;
|
||||||
|
|
||||||
|
export const WebPreviewNavigation = ({
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: WebPreviewNavigationProps) => (
|
||||||
|
<div
|
||||||
|
className={cn('flex items-center gap-1 border-b p-2', className)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type WebPreviewNavigationButtonProps = ComponentProps<typeof Button> & {
|
||||||
|
tooltip?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WebPreviewNavigationButton = ({
|
||||||
|
onClick,
|
||||||
|
disabled,
|
||||||
|
tooltip,
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: WebPreviewNavigationButtonProps) => (
|
||||||
|
<TooltipProvider>
|
||||||
|
<Tooltip>
|
||||||
|
<TooltipTrigger asChild>
|
||||||
|
<Button
|
||||||
|
className="h-8 w-8 p-0 hover:text-foreground"
|
||||||
|
disabled={disabled}
|
||||||
|
onClick={onClick}
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Button>
|
||||||
|
</TooltipTrigger>
|
||||||
|
<TooltipContent>
|
||||||
|
<p>{tooltip}</p>
|
||||||
|
</TooltipContent>
|
||||||
|
</Tooltip>
|
||||||
|
</TooltipProvider>
|
||||||
|
);
|
||||||
|
|
||||||
|
export type WebPreviewUrlProps = ComponentProps<typeof Input>;
|
||||||
|
|
||||||
|
export const WebPreviewUrl = ({
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
onKeyDown,
|
||||||
|
...props
|
||||||
|
}: WebPreviewUrlProps) => {
|
||||||
|
const { url, setUrl } = useWebPreview();
|
||||||
|
|
||||||
|
const handleKeyDown = (event: React.KeyboardEvent<HTMLInputElement>) => {
|
||||||
|
if (event.key === 'Enter') {
|
||||||
|
const target = event.target as HTMLInputElement;
|
||||||
|
setUrl(target.value);
|
||||||
|
}
|
||||||
|
onKeyDown?.(event);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Input
|
||||||
|
className="h-8 flex-1 text-sm"
|
||||||
|
onChange={onChange}
|
||||||
|
onKeyDown={handleKeyDown}
|
||||||
|
placeholder="Enter URL..."
|
||||||
|
value={value ?? url}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WebPreviewBodyProps = ComponentProps<'iframe'> & {
|
||||||
|
loading?: ReactNode;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WebPreviewBody = ({
|
||||||
|
className,
|
||||||
|
loading,
|
||||||
|
src,
|
||||||
|
...props
|
||||||
|
}: WebPreviewBodyProps) => {
|
||||||
|
const { url } = useWebPreview();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex-1">
|
||||||
|
<iframe
|
||||||
|
className={cn('size-full', className)}
|
||||||
|
sandbox="allow-scripts allow-same-origin allow-forms allow-popups allow-presentation"
|
||||||
|
src={(src ?? url) || undefined}
|
||||||
|
title="Preview"
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
{loading}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WebPreviewConsoleProps = ComponentProps<'div'> & {
|
||||||
|
logs?: Array<{
|
||||||
|
level: 'log' | 'warn' | 'error';
|
||||||
|
message: string;
|
||||||
|
timestamp: Date;
|
||||||
|
}>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WebPreviewConsole = ({
|
||||||
|
className,
|
||||||
|
logs = [],
|
||||||
|
children,
|
||||||
|
...props
|
||||||
|
}: WebPreviewConsoleProps) => {
|
||||||
|
const { consoleOpen, setConsoleOpen } = useWebPreview();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Collapsible
|
||||||
|
className={cn('border-t bg-muted/50 font-mono text-sm', className)}
|
||||||
|
onOpenChange={setConsoleOpen}
|
||||||
|
open={consoleOpen}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<CollapsibleTrigger asChild>
|
||||||
|
<Button
|
||||||
|
className="flex w-full items-center justify-between p-4 text-left font-medium hover:bg-muted/50"
|
||||||
|
variant="ghost"
|
||||||
|
>
|
||||||
|
Console
|
||||||
|
<ChevronDownIcon
|
||||||
|
className={cn(
|
||||||
|
'h-4 w-4 transition-transform duration-200',
|
||||||
|
consoleOpen && 'rotate-180'
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</Button>
|
||||||
|
</CollapsibleTrigger>
|
||||||
|
<CollapsibleContent
|
||||||
|
className={cn(
|
||||||
|
'px-4 pb-4',
|
||||||
|
'data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 outline-none data-[state=closed]:animate-out data-[state=open]:animate-in'
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="max-h-48 space-y-1 overflow-y-auto">
|
||||||
|
{logs.length === 0 ? (
|
||||||
|
<p className="text-muted-foreground">No console output</p>
|
||||||
|
) : (
|
||||||
|
logs.map((log, index) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'text-xs',
|
||||||
|
log.level === 'error' && 'text-destructive',
|
||||||
|
log.level === 'warn' && 'text-yellow-600',
|
||||||
|
log.level === 'log' && 'text-foreground'
|
||||||
|
)}
|
||||||
|
key={`${log.timestamp.getTime()}-${index}`}
|
||||||
|
>
|
||||||
|
<span className="text-muted-foreground">
|
||||||
|
{log.timestamp.toLocaleTimeString()}
|
||||||
|
</span>{' '}
|
||||||
|
{log.message}
|
||||||
|
</div>
|
||||||
|
))
|
||||||
|
)}
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</CollapsibleContent>
|
||||||
|
</Collapsible>
|
||||||
|
);
|
||||||
|
};
|
@ -14,7 +14,7 @@ const badgeVariants = cva(
|
|||||||
secondary:
|
secondary:
|
||||||
"border-transparent bg-secondary text-secondary-foreground [a&]:hover:bg-secondary/90",
|
"border-transparent bg-secondary text-secondary-foreground [a&]:hover:bg-secondary/90",
|
||||||
destructive:
|
destructive:
|
||||||
"border-transparent bg-destructive text-white [a&]:hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/70",
|
"border-transparent bg-destructive text-white [a&]:hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
|
||||||
outline:
|
outline:
|
||||||
"text-foreground [a&]:hover:bg-accent [a&]:hover:text-accent-foreground",
|
"text-foreground [a&]:hover:bg-accent [a&]:hover:text-accent-foreground",
|
||||||
},
|
},
|
||||||
|
@ -18,7 +18,7 @@ function ScrollArea({
|
|||||||
>
|
>
|
||||||
<ScrollAreaPrimitive.Viewport
|
<ScrollAreaPrimitive.Viewport
|
||||||
data-slot="scroll-area-viewport"
|
data-slot="scroll-area-viewport"
|
||||||
className="ring-ring/10 dark:ring-ring/20 dark:outline-ring/40 outline-ring/50 size-full rounded-[inherit] transition-[color,box-shadow] focus-visible:ring-4 focus-visible:outline-1"
|
className="focus-visible:ring-ring/50 size-full rounded-[inherit] transition-[color,box-shadow] outline-none focus-visible:ring-[3px] focus-visible:outline-1"
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</ScrollAreaPrimitive.Viewport>
|
</ScrollAreaPrimitive.Viewport>
|
||||||
|
Loading…
Reference in New Issue
Block a user