| title | CopilotChatUserMessage |
|---|---|
| description | User message rendering component with edit and branch navigation |
CopilotChatUserMessage is the default component used by CopilotChatMessageView to render user messages. It handles message display, editing functionality, and branch navigation for conversation history.
The CopilotChatUserMessage component:
- Renders user messages in a styled bubble
- Provides a toolbar with copy and edit buttons
- Supports message editing functionality
- Handles branch navigation for conversation forks
- Built on the slot system for deep customization
CopilotChatUserMessage provides slots for customizing each part of the message:
graph LR
UM[CopilotChatUserMessage] --> messageRenderer
UM --> toolbar
UM --> copyButton
UM --> editButton
UM --> branchNavigation
| Slot | Description |
|---|---|
messageRenderer |
Renders the message text content |
toolbar |
Container for action buttons (appears on hover) |
copyButton |
Button to copy message content |
editButton |
Button to edit the message |
branchNavigation |
Navigation controls for conversation branches |
Customize user messages through the messageView.userMessage slot on CopilotChat:
<CopilotChat
messageView={{
userMessage: {
className: "bg-blue-500 text-white rounded-2xl",
onEditMessage: ({ message }) => handleEdit(message),
},
}}
/>CopilotChatUserMessage provides callbacks for user interactions:
<CopilotChat
messageView={{
userMessage: {
onEditMessage: ({ message }) => {
// Open edit modal or inline editor
setEditingMessage(message);
},
onSwitchToBranch: ({ message, branchIndex, numberOfBranches }) => {
// Switch to a different conversation branch
switchToBranch(message.id, branchIndex);
},
},
}}
/>| Callback | Signature | Description |
|---|---|---|
onEditMessage |
({ message }) => void |
Called when user clicks the edit button |
onSwitchToBranch |
({ message, branchIndex, numberOfBranches }) => void |
Called when user navigates between branches |
CopilotChatUserMessage uses the slot system. Each slot accepts four types of values:
- Tailwind class string - Add or override CSS classes
- Props object - Pass additional props to the default component
- Custom component - Replace the component entirely
- Nested sub-slots - Drill down to customize child components
Style the message bubble:
<CopilotChat
messageView={{
userMessage: {
messageRenderer:
"bg-gradient-to-r from-blue-500 to-purple-500 text-white",
},
}}
/>The toolbar appears on hover and contains action buttons:
<CopilotChat
messageView={{
userMessage: {
toolbar: "bg-gray-50 rounded-lg p-1",
},
}}
/>Customize specific toolbar buttons:
<CopilotChat
messageView={{
userMessage: {
copyButton: "text-gray-500 hover:text-gray-700",
editButton: "text-blue-500 hover:text-blue-700",
},
}}
/>Hide buttons by returning null:
<CopilotChat
messageView={{
userMessage: {
editButton: () => null,
branchNavigation: () => null,
},
}}
/>Note: The editButton only shows when onEditMessage callback is provided.
Customize the branch navigation controls:
<CopilotChat
messageView={{
userMessage: {
branchNavigation: "bg-gray-100 rounded-lg px-2",
},
}}
/>Branch navigation only appears when there are multiple branches (conversation forks) available.
To completely replace the user message component:
import { CopilotChatUserMessage } from "@copilotkit/react-core";
function CustomUserMessage({ message, ...props }) {
return (
<div className="flex gap-3 items-start justify-end">
<div className="flex-1">
<CopilotChatUserMessage
message={message}
className="bg-blue-600 text-white"
{...props}
/>
</div>
<Avatar src="/user-avatar.png" />
</div>
);
}
<CopilotChat
messageView={{
userMessage: CustomUserMessage,
}}
/>;For full layout control, use the children render function:
function CustomUserMessage(props) {
return (
<CopilotChatUserMessage {...props}>
{({ messageRenderer, toolbar, message }) => (
<div className="flex flex-col items-end gap-1">
<div className="flex items-center gap-2">
<span className="text-xs text-gray-400">You</span>
<span className="text-xs text-gray-400">
{new Date(message.createdAt).toLocaleTimeString()}
</span>
</div>
{messageRenderer}
{toolbar}
</div>
)}
</CopilotChatUserMessage>
);
}
<CopilotChat
messageView={{
userMessage: CustomUserMessage,
}}
/>;The render function receives:
| Property | Type | Description |
|---|---|---|
messageRenderer |
ReactElement |
The rendered message content |
toolbar |
ReactElement |
The action buttons toolbar |
copyButton |
ReactElement |
Copy button |
editButton |
ReactElement |
Edit button |
branchNavigation |
ReactElement |
Branch navigation controls |
message |
UserMessage |
The message data |
branchIndex |
number |
Current branch index |
numberOfBranches |
number |
Total number of branches |
When users edit messages and regenerate responses, CopilotKit creates conversation branches. The branch navigation allows users to switch between these alternative conversation paths:
<CopilotChat
messageView={{
userMessage: {
onSwitchToBranch: ({ message, branchIndex, numberOfBranches }) => {
console.log(
`Switching to branch ${branchIndex + 1} of ${numberOfBranches}`,
);
// Your branch switching logic
},
},
}}
/>The branch navigation shows:
- Previous/Next arrows to navigate between branches
- Current branch indicator (e.g., "2/3")
<CopilotChat
messageView={{
userMessage: {
className: "items-end",
messageRenderer:
"bg-blue-600 text-white rounded-2xl px-4 py-2 max-w-[75%]",
toolbar: "opacity-0 group-hover:opacity-100 transition-opacity",
},
}}
/>function ChatWithEdit() {
const [editingMessage, setEditingMessage] = useState(null);
return (
<>
<CopilotChat
messageView={{
userMessage: {
onEditMessage: ({ message }) => setEditingMessage(message),
editButton: "text-blue-500 hover:text-blue-700",
},
}}
/>
{editingMessage && (
<EditMessageModal
message={editingMessage}
onClose={() => setEditingMessage(null)}
/>
)}
</>
);
}Hide all toolbar elements for a clean look:
<CopilotChat
messageView={{
userMessage: {
toolbar: () => null,
},
}}
/>function UserMessageWithAvatar(props) {
return (
<CopilotChatUserMessage {...props}>
{({ messageRenderer, toolbar }) => (
<div className="flex items-start gap-3 justify-end">
<div className="flex flex-col items-end">
{messageRenderer}
{toolbar}
</div>
<img
src="/user-avatar.png"
alt="You"
className="w-8 h-8 rounded-full"
/>
</div>
)}
</CopilotChatUserMessage>
);
}
<CopilotChat
messageView={{
userMessage: UserMessageWithAvatar,
}}
/>;<CopilotChat
messageView={{
userMessage: {
messageRenderer: "bg-blue-600 text-white dark:bg-blue-500",
toolbar: "text-gray-400 dark:text-gray-500",
copyButton: "hover:text-white dark:hover:text-gray-300",
editButton: "hover:text-white dark:hover:text-gray-300",
},
}}
/>- CopilotChat - Parent component
- CopilotChatMessageView - Message list component that uses user messages
- CopilotChatAssistantMessage - Counterpart for AI messages
- Slot System - Deep dive into slot customization