Install this skill
npx skills add jezweb/claude-skillsWorks across Claude Code, Cursor, Codex, Copilot & Antigravity
The AI SDK UI provides specialized React hooks for building intelligent interfaces using Vercel AI SDK v6. This toolkit moves beyond simple string-based messaging by introducing a granular .parts array structure to handle text, tool invocations, file attachments, and model reasoning as distinct entities. It enforces type-safe communications between React frontends and agents via the InferAgentUIMessage utility. A central architectural shift in the v6 release requires developers to manage local input states manually, as the library no longer provides automatic input handling. Additionally, the SDK simplifies human-in-the-loop interactions through built-in tool approval workflows, allowing apps to pause, confirm, or deny agent actions mid-stream before automatically resuming execution. This library serves as the primary bridge for developers upgrading to modern, state-managed conversational interfaces.
When to Use This Skill
- β’Building interactive chat interfaces with complex multi-modal file attachments
- β’Implementing agents that require manual user verification before calling external APIs
- β’Developing dashboards that display model reasoning and source citations alongside text
- β’Creating chat apps that output structured JSON objects alongside conversational responses
How to Invoke This Skill
Example prompts that trigger this skill in Claude Code, Cursor, or Antigravity:
- βSetup a chat interface with AI SDK v6
- βHow do I handle tool calls in Vercel AI SDK?
- βImplement manual tool approval in my React chat
- βMigrate useChat hook to the latest AI SDK version
- βDisplay message parts and file attachments in React
- βType-check my chat component with an AI agent
Pro Tips
- π‘Always iterate through the `m.parts` array when rendering messages to ensure all content types (text, tool-invocation, file) are properly displayed.
- π‘Design specific UI components for `tool-invocation` parts to provide clear visual feedback when the AI calls external functions.
- π‘Regularly check the Vercel AI SDK documentation for updates on message structure and new part types to keep your UI current.
What this skill does
- β’Parses multi-part message payloads containing text, tools, and files
- β’Integrates human-in-the-loop tool approval workflows
- β’Supports automatic message submission after tool confirmation
- β’Provides strict TypeScript inference for agent-specific message schemas
- β’Enables structured data streaming directly within chat conversations
When not to use it
- βIf your project requires simple, legacy string-based chat implementation without tool interaction
- βIf you are not using React as your frontend framework
- βIf you prefer to maintain the deprecated v4/v5 auto-input state management pattern
Example workflow
- Initialize useChat hook and define your agent schema
- Map incoming messages to render text, tool-invocations, and file types
- Capture user input in a local React state variable
- Trigger the sendMessage function on form submission
- Display confirmation buttons for tools in the awaiting-approval state
- Execute addToolApprovalResponse to resolve the human-in-the-loop step
Prerequisites
- βReact 18 or 19
- βNext.js 14 or 15
- βVercel AI SDK v6.x
Pitfalls & limitations
- !Attempting to use deprecated v4 input state management like handleInputChange
- !Accessing message content directly instead of iterating through the .parts array
- !Forgetting to handle the awaiting-approval state for tools, causing silent failures
FAQ
How it compares
Unlike manual fetching or generic API calls, this SDK automates the complex state transitions between streaming text, tool execution, and approval logic that otherwise require significant custom boilerplate.
π Full skill instructions β original source: jezweb/claude-skills
Frontend React hooks for AI-powered user interfaces with Vercel AI SDK v6.
**Version**: AI SDK v6.0.42 (Stable)
**Framework**: React 18+/19, Next.js 14+/15+
**Last Updated**: 2026-01-20
---
## AI SDK v6 Stable (January 2026)
**Status:** Stable Release
**Latest:** [email protected], @ai-sdk/[email protected], @ai-sdk/[email protected]
**Migration:** Minimal breaking changes from v5 β v6
### New UI Features in v6
**1. Message Parts Structure (Breaking Change)**
In v6, message content is now accessed via
.parts array instead of .content:// β v5 (OLD)
{messages.map(m => (
<div key={m.id}>{m.content}</div>
))}
// β
v6 (NEW)
{messages.map(m => (
<div key={m.id}>
{m.parts.map((part, i) => {
if (part.type === 'text') return <span key={i}>{part.text}</span>;
if (part.type === 'tool-invocation') return <ToolCall key={i} tool={part} />;
if (part.type === 'file') return <FilePreview key={i} file={part} />;
return null;
})}
</div>
))}**Part Types:**
-
text - Text content with .text property-
tool-invocation - Tool calls with .toolName, .args, .result-
file - File attachments with .mimeType, .data-
reasoning - Model reasoning (when available)-
source - Source citations**3. Agent Integration**
Type-safe messaging with agents using
InferAgentUIMessage<typeof agent>:import { useChat } from '@ai-sdk/react';
import type { InferAgentUIMessage } from 'ai';
import { myAgent } from './agent';
export default function AgentChat() {
const { messages, sendMessage } = useChat<InferAgentUIMessage<typeof myAgent>>({
api: '/api/chat',
});
// messages are now type-checked against agent schema
}**4. Tool Approval Workflows (Human-in-the-Loop)**
Request user confirmation before executing tools:
import { useChat } from '@ai-sdk/react';
import { useState } from 'react';
export default function ChatWithApproval() {
const { messages, sendMessage, addToolApprovalResponse } = useChat({
api: '/api/chat',
});
const handleApprove = (toolCallId: string) => {
addToolApprovalResponse({
toolCallId,
approved: true, // or false to deny
});
};
return (
<div>
{messages.map(message => (
<div key={message.id}>
{message.toolInvocations?.map(tool => (
tool.state === 'awaiting-approval' && (
<div key={tool.toolCallId}>
<p>Approve tool call: {tool.toolName}?</p>
<button onClick={() => handleApprove(tool.toolCallId)}>
Approve
</button>
<button onClick={() => addToolApprovalResponse({
toolCallId: tool.toolCallId,
approved: false
})}>
Deny
</button>
</div>
)
))}
</div>
))}
</div>
);
}**5. Auto-Submit Capability**
Automatically continue conversation after handling approvals:
import { useChat, lastAssistantMessageIsCompleteWithApprovalResponses } from '@ai-sdk/react';
export default function AutoSubmitChat() {
const { messages, sendMessage } = useChat({
api: '/api/chat',
sendAutomaticallyWhen: lastAssistantMessageIsCompleteWithApprovalResponses,
// Automatically resubmit after all approval responses provided
});
}**6. Structured Output in Chat**
Generate structured data alongside tool calling (previously only available in
useObject):import { useChat } from '@ai-sdk/react';
import { z } from 'zod';
const schema = z.object({
summary: z.string(),
sentiment: z.enum(['positive', 'neutral', 'negative']),
});
export default function StructuredChat() {
const { messages, sendMessage } = useChat({
api: '/api/chat',
// Server can now stream structured output with chat messages
});
}---
## useChat Hook - v4 β v5 Breaking Changes
**CRITICAL: useChat no longer manages input state in v5!**
**v4 (OLD - DON'T USE):**
const { messages, input, handleInputChange, handleSubmit, append } = useChat();
<form onSubmit={handleSubmit}>
<input value={input} onChange={handleInputChange} />
</form>**v5 (NEW - CORRECT):**
const { messages, sendMessage } = useChat();
const [input, setInput] = useState('');
<form onSubmit={(e) => {
e.preventDefault();
sendMessage({ content: input });
setInput('');
}}>
<input value={input} onChange={(e) => setInput(e.target.value)} />
</form>**Summary of v5 Changes:**
1. **Input management removed**:
input, handleInputChange, handleSubmit no longer exist2. **
append() β sendMessage()**: New method for sending messages3. **
onResponse removed**: Use onFinish instead4. **
initialMessages β controlled mode**: Use messages prop for full control5. **
maxSteps removed**: Handle on server-side onlySee
references/use-chat-migration.md for complete migration guide.---
## useAssistant Hook (Deprecated)
> **β οΈ Deprecation Notice**:
useAssistant is deprecated as of AI SDK v5. OpenAI Assistants API v2> will sunset on August 26, 2026. For new projects, use
useChat with custom backend logic instead.> See the **openai-assistants** skill for migration guidance.
Interact with OpenAI-compatible assistant APIs with automatic UI state management.
**Import:**
import { useAssistant } from '@ai-sdk/react';**Basic Usage:**
'use client';
import { useAssistant } from '@ai-sdk/react';
import { useState, FormEvent } from 'react';
export default function AssistantChat() {
const { messages, sendMessage, isLoading, error } = useAssistant({
api: '/api/assistant',
});
const [input, setInput] = useState('');
const handleSubmit = (e: FormEvent) => {
e.preventDefault();
sendMessage({ content: input });
setInput('');
};
return (
<div>
{messages.map(m => (
<div key={m.id}>
<strong>{m.role}:</strong> {m.content}
</div>
))}
<form onSubmit={handleSubmit}>
<input
value={input}
onChange={(e) => setInput(e.target.value)}
disabled={isLoading}
/>
</form>
{error && <div>{error.message}</div>}
</div>
);
}**Use Cases:**
- Building OpenAI Assistant-powered UIs
- Managing assistant threads and runs
- Streaming assistant responses with UI state management
- File search and code interpreter integrations
See official docs for complete API reference: https://ai-sdk.dev/docs/reference/ai-sdk-ui/use-assistant
---
## Top UI Errors & Solutions
See
references/top-ui-errors.md for complete documentation. Quick reference:### 1. useChat Failed to Parse Stream
**Error**:
SyntaxError: Unexpected token in JSON at position X**Cause**: API route not returning proper stream format.
**Solution**:
// β
CORRECT
return result.toDataStreamResponse();
// β WRONG
return new Response(result.textStream);### 2. useChat No Response
**Cause**: API route not streaming correctly.
**Solution**:
// App Router - use toDataStreamResponse()
export async function POST(req: Request) {
const result = streamText({ /* ... */ });
return result.toDataStreamResponse(); // β
}
// Pages Router - use pipeDataStreamToResponse()
export default async function handler(req, res) {
const result = streamText({ /* ... */ });
return result.pipeDataStreamToResponse(res); // β
}### 3. Streaming Not Working When Deployed
**Cause**: Deployment platform buffering responses.
**Solution**: Vercel auto-detects streaming. Other platforms may need configuration.
### 4. Stale Body Values with useChat
**Cause**:
body option captured at first render only.**Solution**:
// β WRONG - body captured once
const { userId } = useUser();
const { messages } = useChat({
body: { userId }, // Stale!
});
// β
CORRECT - use controlled mode
const { userId } = useUser();
const { messages, sendMessage } = useChat();
sendMessage({
content: input,
data: { userId }, // Fresh on each send
});### 5. React Maximum Update Depth
**Cause**: Infinite loop in useEffect.
**Solution**:
// β WRONG
useEffect(() => {
saveMessages(messages);
}, [messages, saveMessages]); // saveMessages triggers re-render!
// β
CORRECT
useEffect(() => {
saveMessages(messages);
}, [messages]); // Only depend on messagesSee
references/top-ui-errors.md for 13 more common errors (18 total documented).---
## Streaming Best Practices
### Performance
**Always use streaming for better UX:**
// β
GOOD - Streaming (shows tokens as they arrive)
const { messages } = useChat({ api: '/api/chat' });
// β BAD - Non-streaming (user waits for full response)
const response = await fetch('/api/chat', { method: 'POST' });### UX Patterns
**Show loading states:**
{isLoading && <div>AI is typing...</div>}**Provide stop button:**
{isLoading && <button onClick={stop}>Stop</button>}**Auto-scroll to latest message:**
useEffect(() => {
messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' });
}, [messages]);**Disable input while loading:**
<input disabled={isLoading} />See
references/streaming-patterns.md for comprehensive best practices.---
## React Strict Mode Considerations
React Strict Mode intentionally double-invokes effects to catch bugs. When using
useChat or useCompletion in effects (auto-resume, initial messages), guard against double execution to prevent duplicate API calls and token waste.**Problem:**
'use client';
import { useChat } from '@ai-sdk/react';
import { useEffect } from 'react';
export default function Chat() {
const { messages, sendMessage, resumeStream } = useChat({
api: '/api/chat',
resume: true,
});
useEffect(() => {
// β Triggers twice in strict mode β two concurrent streams
sendMessage({ content: 'Hello' });
// or
resumeStream();
}, []);
}**Solution:**
// β
Use ref to track execution
import { useRef } from 'react';
const hasSentRef = useRef(false);
useEffect(() => {
if (hasSentRef.current) return;
hasSentRef.current = true;
sendMessage({ content: 'Hello' });
}, []);
// For resumeStream specifically:
const hasResumedRef = useRef(false);
useEffect(() => {
if (!autoResume || hasResumedRef.current || status === 'streaming') return;
hasResumedRef.current = true;
resumeStream();
}, [autoResume, resumeStream, status]);**Why It Happens:** React Strict Mode double-invokes effects to surface side effects. The SDK doesn't guard against concurrent requests, so both invocations create separate streams that fight for state updates.
**Impact:** Duplicate messages, doubled token usage, race conditions causing TypeError: "Cannot read properties of undefined (reading 'state')".
**Source:** [GitHub Issue #7891](https://github.com/vercel/ai/issues/7891), [Issue #6166](https://github.com/vercel/ai/issues/6166)
---
## When to Use This Skill
### Use ai-sdk-ui When:
- Building React chat interfaces
- Implementing AI completions in UI
- Streaming AI responses to frontend
- Building Next.js AI applications
- Handling chat message state
- Displaying tool calls in UI
- Managing file attachments with AI
- Migrating from v4 to v5 (UI hooks)
- Encountering useChat/useCompletion errors
### Don't Use When:
- Need backend AI functionality β Use **ai-sdk-core** instead
- Building non-React frontends (Svelte, Vue) β Check official docs
- Need Generative UI / RSC β See https://ai-sdk.dev/docs/ai-sdk-rsc
- Building native apps β Different SDK required
### Related Skills:
- **ai-sdk-core** - Backend text generation, structured output, tools, agents
- Compose both for full-stack AI applications
---
## Package Versions
**Stable (v6 - Recommended):**
{
"dependencies": {
"ai": "^6.0.8",
"@ai-sdk/react": "^3.0.6",
"@ai-sdk/openai": "^3.0.2",
"react": "^18.3.0",
"zod": "^3.24.2"
}
}**Legacy (v5):**
{
"dependencies": {
"ai": "^5.0.99",
"@ai-sdk/react": "^1.0.0",
"@ai-sdk/openai": "^2.0.68"
}
}**Version Notes:**
- AI SDK v6.0.6 (stable, Jan 2026) - recommended for new projects
- AI SDK v5.x (legacy) - still supported but not receiving new features
- React 18.3+ / React 19 supported
- Next.js 14+/15+ recommended
- Zod 3.24.2+ for schema validation
---
## Links to Official Documentation
**Core UI Hooks:**
- AI SDK UI Overview: https://ai-sdk.dev/docs/ai-sdk-ui/overview
- useChat: https://ai-sdk.dev/docs/ai-sdk-ui/chatbot
- useCompletion: https://ai-sdk.dev/docs/ai-sdk-ui/completion
- useObject: https://ai-sdk.dev/docs/ai-sdk-ui/object-generation
**Advanced Topics (Link Only):**
- Generative UI (RSC): https://ai-sdk.dev/docs/ai-sdk-rsc/overview
- Stream Protocols: https://ai-sdk.dev/docs/ai-sdk-ui/stream-protocols
- Message Metadata: https://ai-sdk.dev/docs/ai-sdk-ui/message-metadata
**Next.js Integration:**
- Next.js App Router: https://ai-sdk.dev/docs/getting-started/nextjs-app-router
- Next.js Pages Router: https://ai-sdk.dev/docs/getting-started/nextjs-pages-router
**Migration & Troubleshooting:**
- v4βv5 Migration: https://ai-sdk.dev/docs/migration-guides/migration-guide-5-0
- Troubleshooting: https://ai-sdk.dev/docs/troubleshooting
- Common Issues: https://ai-sdk.dev/docs/troubleshooting/common-issues
**Vercel Deployment:**
- Vercel Functions: https://vercel.com/docs/functions
- Streaming on Vercel: https://vercel.com/docs/functions/streaming
---
## Templates
This skill includes the following templates in
templates/:1. **use-chat-basic.tsx** - Basic chat with manual input (v5 pattern)
2. **use-chat-tools.tsx** - Chat with tool calling UI rendering
3. **use-chat-attachments.tsx** - File attachments support
4. **use-completion-basic.tsx** - Basic text completion
5. **use-object-streaming.tsx** - Streaming structured data
6. **nextjs-chat-app-router.tsx** - Next.js App Router complete example
7. **nextjs-chat-pages-router.tsx** - Next.js Pages Router complete example
8. **nextjs-api-route.ts** - API route for both App and Pages Router
9. **message-persistence.tsx** - Save/load chat history
10. **custom-message-renderer.tsx** - Custom message components with markdown
11. **package.json** - Dependencies template
## Reference Documents
See
references/ for:- **use-chat-migration.md** - Complete v4βv5 migration guide
- **streaming-patterns.md** - UI streaming best practices
- **top-ui-errors.md** - 18 common UI errors with solutions
- **nextjs-integration.md** - Next.js setup patterns
- **links-to-official-docs.md** - Organized links to official docs
---
**Production Tested**: WordPress Auditor (https://wordpress-auditor.webfonts.workers.dev)
**Last verified**: 2026-01-20 | **Skill version**: 3.1.0 | **Changes**: Updated to AI SDK v6.0.42 (+19 patches). Added React Strict Mode section. Expanded Issue #7 (stale body) with 3 workarounds. Added 6 new issues: TypeError with resume+onFinish (#13), concurrent sendMessage state corruption (#14), tool approval callback edge case (#15), ZodError on early stop (#16), convertToModelMessages tool approval bug (#17), undefined id infinite loop (#18). Error count: 12β18.
---
---
paths: "**/*.ts", "**/*.tsx", "**/chat*.ts", "**/ai*.ts"
---
# AI SDK UI v5 Corrections
Claude's training may reference AI SDK v4 UI patterns. This project uses **AI SDK v5+**.
## Hook Import Changes
/* β v4 imports */
import { useChat, useCompletion } from 'ai/react'
import { useActions, useUIState } from 'ai/rsc'
/* β
v5 imports */
import { useChat, useCompletion } from '@ai-sdk/react'
import { useActions, useUIState } from '@ai-sdk/rsc'## useChat Options
/* β v4 options */
const { messages, input, handleSubmit } = useChat({
api: '/api/chat',
initialMessages: [],
body: { model: 'gpt-4' },
})
/* β
v5 options */
const { messages, input, handleSubmit } = useChat({
api: '/api/chat',
initialMessages: [],
body: { model: 'gpt-5' },
maxSteps: 5, // v5: multi-step by default
})## Message Types
/* β v4 message type */
import type { Message } from 'ai'
const msgs: Message[] = []
/* β
v5 message type */
import type { UIMessage } from '@ai-sdk/react'
const msgs: UIMessage[] = []## Streaming Text
/* β v4 streaming */
import { StreamingTextResponse } from 'ai'
return new StreamingTextResponse(stream)
/* β
v5 streaming */
import { pipeTextStreamToResponse } from 'ai'
return pipeTextStreamToResponse(result.toTextStream(), response)
// Or use result.toDataStreamResponse() for Next.js
return result.toDataStreamResponse()## Tool Results in UI
/* β v4 tool results */
{messages.map((m) => (
m.toolInvocations?.map((t) => (
<ToolResult key={t.toolCallId} result={t.result} />
))
))}
/* β
v5 tool results (check state) */
{messages.map((m) => (
m.toolInvocations?.map((t) => (
t.state === 'result' && (
<ToolResult key={t.toolCallId} result={t.result} />
)
))
))}## Quick Fixes
| If Claude suggests... | Use instead... |
|----------------------|----------------|
|
from 'ai/react' | from '@ai-sdk/react' ||
from 'ai/rsc' | from '@ai-sdk/rsc' ||
Message type | UIMessage type ||
StreamingTextResponse | toDataStreamResponse() or pipeTextStreamToResponse() ||
useChat without maxSteps | Add maxSteps for multi-step || Direct tool result access | Check
t.state === 'result' first |How to Use This Skill Unit
Option A: Project-Specific (Recommended)
- Click "Download" above
- In your project, create the directory:
.agent/skills/ai-sdk-ui/ - Save the file as
SKILL.md - The agent will automatically discover the skill based on its description.
Option B: Global Installation (All Agents)
Save the file to these locations to make it available across all projects:
- Claude Code:
~/.claude/skills/jezweb/claude-skills/ai-sdk-ui/SKILL.md - Cursor:
~/.cursor/skills/jezweb/claude-skills/ai-sdk-ui/SKILL.md - Antigravity:
~/.gemini/antigravity/skills/jezweb/claude-skills/ai-sdk-ui/SKILL.md
π Install with CLI:npx skills add jezweb/claude-skills
