promptCompare
Val Town is a collaborative website to build and scale JavaScript apps.
Deploy APIs, crons, & store data – all from the browser, and deployed in milliseconds.
A composable component for previewing the result of a generated UI, with support for live examples and code display.
The WebPreview component provides a flexible way to showcase the result of a
generated UI component, along with its source code. It is designed for
documentation and demo purposes, allowing users to interact with live examples
and view the underlying implementation.
See scripts/web-preview.tsx for this example.
npx ai-elements@latest add web-preview
Build a simple v0 clone using the v0 Platform API.
Install the v0-sdk package:
npm i v0-sdk
Add the following component to your frontend:
"use client";
import {
WebPreview,
WebPreviewBody,
WebPreviewNavigation,
WebPreviewUrl,
} from "@/components/ai-elements/web-preview";
import { useState } from "react";
import {
Input,
PromptInputSubmit,
PromptInputTextarea,
} from "@/components/ai-elements/prompt-input";
import { Spinner } from "@/components/ui/spinner";
const WebPreviewDemo = () => {
const [previewUrl, setPreviewUrl] = useState("");
const [prompt, setPrompt] = useState("");
const [isGenerating, setIsGenerating] = useState(false);
const handleSubmit = async (e: React.FormEvent) => {
e.preventDefault();
if (!prompt.trim()) return;
setPrompt("");
setIsGenerating(true);
try {
const response = await fetch("/api/v0", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ prompt }),
});
const data = await response.json();
setPreviewUrl(data.demo || "/");
console.log("Generation finished:", data);
} catch (error) {
console.error("Generation failed:", error);
} finally {
setIsGenerating(false);
}
};
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 mb-4">
{isGenerating
? (
<div className="flex flex-col items-center justify-center h-full">
<Spinner />
<p className="mt-4 text-muted-foreground">
Generating app, this may take a few seconds...
</p>
</div>
)
: previewUrl
? (
<WebPreview defaultUrl={previewUrl}>
<WebPreviewNavigation>
<WebPreviewUrl />
</WebPreviewNavigation>
<WebPreviewBody src={previewUrl} />
</WebPreview>
)
: (
<div className="flex items-center justify-center h-full text-muted-foreground">
Your generated app will appear here
</div>
)}
</div>
<Input
onSubmit={handleSubmit}
className="w-full max-w-2xl mx-auto relative"
>
<PromptInputTextarea
value={prompt}
placeholder="Describe the app you want to build..."
onChange={(e) => setPrompt(e.currentTarget.value)}
className="pr-12 min-h-[60px]"
/>
<PromptInputSubmit
status={isGenerating ? "streaming" : "ready"}
disabled={!prompt.trim()}
className="absolute bottom-1 right-1"
/>
</Input>
</div>
</div>
);
};
export default WebPreviewDemo;
Add the following route to your backend:
import { v0 } from "v0-sdk";
export async function POST(req: Request) {
const { prompt }: { prompt: string } = await req.json();
const result = await v0.chats.create({
system: "You are an expert coder",
message: prompt,
modelConfiguration: {
modelId: "v0-1.5-sm",
imageGenerations: false,
thinking: false,
},
});
return Response.json({
demo: result.demo,
webUrl: result.webUrl,
});
}
- Live preview of UI components
- Composable architecture with dedicated sub-components
- Responsive design modes (Desktop, Tablet, Mobile)
- Navigation controls with back/forward functionality
- URL input and example selector
- Full screen mode support
- Console logging with timestamps
- Context-based state management
- Consistent styling with the design system
- Easy integration into documentation pages
| Prop | Type | Default | Description |
|---|---|---|---|
defaultUrl | string | - | The initial URL to load in the preview. |
onUrlChange | (url: string) => void | - | Callback fired when the URL changes. |
...props | React.HTMLAttributes<HTMLDivElement> | - | Any other props are spread to the root div. |
| Prop | Type | Default | Description |
|---|---|---|---|
...props | React.HTMLAttributes<HTMLDivElement> | - | Any other props are spread to the navigation container. |
| Prop | Type | Default | Description |
|---|---|---|---|
tooltip | string | - | Tooltip text to display on hover. |
...props | React.ComponentProps<typeof Button> | - | Any other props are spread to the underlying shadcn/ui Button component. |
| Prop | Type | Default | Description |
|---|---|---|---|
...props | React.ComponentProps<typeof Input> | - | Any other props are spread to the underlying shadcn/ui Input component. |
| Prop | Type | Default | Description |
|---|---|---|---|
loading | React.ReactNode | - | Optional loading indicator to display over the preview. |
...props | React.IframeHTMLAttributes<HTMLIFrameElement> | - | Any other props are spread to the underlying iframe. |
| Prop | Type | Default | Description |
|---|---|---|---|
logs | Array<{ level: | - | Console log entries to display in the console panel. |
...props | React.HTMLAttributes<HTMLDivElement> | - | Any other props are spread to the root div. |