skills/react-core/references/attachments.md
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.
<CopilotChat>"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}`);
},
}}
/>
);
}
"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.
onUpload replaces the default base64-inline strategy. Return an
Attachment.source describing where the file lives.
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}`);
},
},
});
useAttachments({
config: {
enabled: true,
maxSize: 5 * 1024 * 1024,
onUploadFailed: ({ reason, file, message }) => {
// reason: "file-too-large" | "invalid-type" | "upload-failed"
toast.error(message);
},
},
});
consumeAttachments on submitWrong:
const { attachments } = useAttachments({ config: { enabled: true } });
const onSubmit = () => {
sendMessage({ text, attachments });
// attachments queue never cleared — sticks around for the next message
};
Correct:
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
maxSize in KB or MBWrong:
useAttachments({ config: { enabled: true, maxSize: 10 } });
// 10 bytes! Effectively blocks every file.
Correct:
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
containerRef on the paste-scope elementWrong:
const { enabled } = useAttachments({ config: { enabled: true } });
return (
<div>
<input type="text" />
</div>
); // no containerRef attached
Correct:
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
imageUploadsEnabled on <CopilotChat>Wrong:
<CopilotChat imageUploadsEnabled />
Correct:
<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
onUploadFailedWrong:
useAttachments({ config: { enabled: true } });
// Rejected files silently disappear. User has no idea why.
Correct:
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