@copilotkit/react-core
Version:
<img src="https://github.com/user-attachments/assets/0a6b64d9-e193-4940-a3f6-60334ac34084" alt="banner" style="border-radius: 12px; border: 2px solid #d6d4fa;" />
292 lines (239 loc) • 7.22 kB
Markdown
# CopilotKit Attachments (React)
This skill builds on `copilotkit/provider-setup` and
`copilotkit/chat-components`. `useAttachments` is exposed both as an opt-in
prop on `<CopilotChat>` and as a direct hook for custom chat surfaces.
## Setup
### Easiest: turn attachments on via `<CopilotChat>`
```tsx
"use client";
import { CopilotChat } from "@copilotkit/react-core/v2";
import "@copilotkit/react-core/v2/styles.css";
export function ChatPanel() {
return (
<CopilotChat
agentId="default"
attachments={{
enabled: true,
accept: "image/*",
maxSize: 10 * 1024 * 1024, // 10 MB
onUploadFailed: ({ reason, file, message }) => {
console.warn(`[attachments] ${reason}: ${file.name} — ${message}`);
},
}}
/>
);
}
```
### Direct hook usage for custom surfaces
```tsx
"use client";
import {
useAttachments,
useAgent,
useCopilotKit,
} from "@copilotkit/react-core/v2";
import type { InputContent } from "@ag-ui/core";
export function CustomChatInput() {
const { agent } = useAgent({ agentId: "default" });
const { copilotkit } = useCopilotKit();
const {
attachments,
containerRef,
fileInputRef,
handleFileUpload,
handleDragOver,
handleDragLeave,
handleDrop,
removeAttachment,
consumeAttachments,
} = useAttachments({ config: { enabled: true, accept: "*/*" } });
return (
<div
ref={containerRef}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleDrop}
>
<input type="file" ref={fileInputRef} onChange={handleFileUpload} />
{attachments.map((a) => (
<button key={a.id} onClick={() => removeAttachment(a.id)}>
{a.filename} ({a.status})
</button>
))}
<button
onClick={async () => {
const ready = consumeAttachments();
// `ready` is Attachment[] — map each to an AG-UI InputContent part
// before spreading into the message content array.
const contentParts: InputContent[] = [
{ type: "text", text: "See attachments." },
...ready.map(
(att) =>
({
type: att.type,
source: att.source,
metadata: {
...(att.filename ? { filename: att.filename } : {}),
...att.metadata,
},
}) as InputContent,
),
];
agent.addMessage({
id: crypto.randomUUID(),
role: "user",
content: contentParts,
});
await copilotkit.runAgent({ agent });
}}
>
Send
</button>
</div>
);
}
```
`consumeAttachments()` returns the `Attachment[]` queue — each entry has
`{ id, type, source, filename, status, metadata }` and is NOT a valid
AG-UI content part. Map each attachment to an `InputContent` shape
(`{ type, source, metadata }`) before spreading into a message's `content`
array. See `packages/react-core/src/v2/components/chat/CopilotChat.tsx:247-268`
for the canonical transform.
## Core Patterns
### Custom upload backend (S3 / presigned URL)
`onUpload` replaces the default base64-inline strategy. Return an
`Attachment.source` describing where the file lives.
```tsx
useAttachments({
config: {
enabled: true,
accept: "image/*,application/pdf",
maxSize: 50 * 1024 * 1024,
onUpload: async (file) => {
const { url } = await fetch("/api/upload", {
method: "POST",
body: file,
}).then((r) => r.json());
return { type: "url", value: url, mimeType: file.type };
},
onUploadFailed: ({ reason, file, message }) => {
toast.error(`${file.name}: ${message}`);
},
},
});
```
### Feedback on failed uploads
```tsx
useAttachments({
config: {
enabled: true,
maxSize: 5 * 1024 * 1024,
onUploadFailed: ({ reason, file, message }) => {
// reason: "file-too-large" | "invalid-type" | "upload-failed"
toast.error(message);
},
},
});
```
## Common Mistakes
### HIGH — Forgetting to call `consumeAttachments` on submit
Wrong:
```tsx
const { attachments } = useAttachments({ config: { enabled: true } });
const onSubmit = () => {
sendMessage({ text, attachments });
// attachments queue never cleared — sticks around for the next message
};
```
Correct:
```tsx
const { consumeAttachments } = useAttachments({ config: { enabled: true } });
const onSubmit = () => {
const ready = consumeAttachments();
sendMessage({ text, attachments: ready });
};
```
`consumeAttachments()` returns ready attachments AND drains the internal
queue. If you submit without calling it, attachments stay in state and
accompany every subsequent message.
Source: `packages/react-core/src/v2/hooks/use-attachments.tsx:40-46`
### HIGH — Passing `maxSize` in KB or MB
Wrong:
```tsx
useAttachments({ config: { enabled: true, maxSize: 10 } });
// 10 bytes! Effectively blocks every file.
```
Correct:
```tsx
useAttachments({
config: { enabled: true, maxSize: 10 * 1024 * 1024 }, // 10 MB
});
```
`maxSize` is bytes. The default is `20 * 1024 * 1024` (20 MB). Passing a
small number without the multiplier silently rejects every file via
`onUploadFailed({ reason: "file-too-large" })`.
Source: `packages/react-core/src/v2/hooks/use-attachments.tsx:73-74`
### HIGH — Missing `containerRef` on the paste-scope element
Wrong:
```tsx
const { enabled } = useAttachments({ config: { enabled: true } });
return (
<div>
<input type="text" />
</div>
); // no containerRef attached
```
Correct:
```tsx
const { containerRef, handleDragOver, handleDrop } = useAttachments({
config: { enabled: true },
});
return (
<div ref={containerRef} onDragOver={handleDragOver} onDrop={handleDrop}>
<input type="text" />
</div>
);
```
Clipboard paste is scoped to the element `containerRef` points at. Without
attaching the ref, `Ctrl+V` / `Cmd+V` never reaches the paste handler and
users silently can't paste images from screenshots.
Source: `packages/react-core/src/v2/hooks/use-attachments.tsx:207-239`
### MEDIUM — Using `imageUploadsEnabled` on `<CopilotChat>`
Wrong:
```tsx
<CopilotChat imageUploadsEnabled />
```
Correct:
```tsx
<CopilotChat
attachments={{
enabled: true,
accept: "image/*",
maxSize: 5 * 1024 * 1024,
}}
/>
```
`imageUploadsEnabled` was the v1 flag. v2 replaces it with the `attachments`
config object, which is powered by `useAttachments` internally and supports
any MIME type, not only images.
Source: `docs/content/docs/(root)/migration-guides/migrate-attachments.mdx`
### MEDIUM — Ignoring `onUploadFailed`
Wrong:
```tsx
useAttachments({ config: { enabled: true } });
// Rejected files silently disappear. User has no idea why.
```
Correct:
```tsx
useAttachments({
config: {
enabled: true,
onUploadFailed: ({ reason, file, message }) => {
toast.error(message);
},
},
});
```
Size violations, MIME mismatches, and `onUpload` throws all drop the file
from the queue with no UI feedback unless `onUploadFailed` is wired.
Source: `packages/react-core/src/v2/hooks/use-attachments.tsx:79-157`