showcase/shell-docs/src/content/docs/integrations/crewai-flows/shared-state/in-app-agent-write.mdx
<video
src="https://cdn.copilotkit.ai/docs/copilotkit/images/coagents/write-agent-state.mp4"
className="rounded-lg shadow-xl"
loop
playsInline
controls
autoPlay
muted
/>
<Callout>
This video shows the result of npx copilotkit@latest init with the
implementation section applied to it!
</Callout>
This guide shows you how to write to your agent's state from your application.
You can use this when you want to provide the user with feedback about what your agent is doing, specifically when your agent is calling tools. CopilotKit allows you to fully customize how these tools are rendered in the chat.
You'll need to run your agent and connect it to CopilotKit before proceeding. If you haven't done so already,
you can follow the instructions in the [Getting Started](/crewai-flows/quickstart) guide.
If you don't already have an agent, you can use the [coagent starter](https://github.com/copilotkit/copilotkit/tree/main/examples/coagents-starter-crewai-flows) as a starting point
as this guide uses it as a starting point.
<Tabs groupId="language_crewai-flows_agent" items={["Python", "TypeScript"]} persist>
<Tab value="Python">
```python title="agent.py"
from copilotkit.crewai import CopilotKitState
from typing import Literal
class AgentState(CopilotKitState):
language: Literal["english", "spanish"] = "english"
```
</Tab>
</Tabs>
```tsx title="ui/app/page.tsx"
// Define the agent state type, should match the actual state of your agent
type AgentState = {
language: "english" | "spanish";
}
// Example usage in a pseudo React component
function YourMainContent() {
const { agent } = useAgent({ // [!code highlight]
agentId: "sample_agent",
initialState: { language: "english" } // optionally provide an initial state
});
// ...
const toggleLanguage = () => {
agent.setState({ language: agent.state?.language === "english" ? "spanish" : "english" }); // [!code highlight]
};
// ...
return (
// style excluded for brevity
<div>
<h1>Your main content</h1>
<p>Language: {agent.state?.language}</p>
<button onClick={toggleLanguage}>Toggle Language</button>
</div>
);
}
```
The new agent state will be used next time the agent runs.
If you want to re-run it manually, use copilotkit.runAgent().
The agent will be re-run with the latest updated state. You can also add a hint message before re-running.
// ...
function YourMainContent() {
const { agent } = useAgent({
agentId: "sample_agent",
});
const { copilotkit } = useCopilotKit(); // [!code highlight]
// setup to be called when some event in the app occurs
const toggleLanguage = async () => {
const newLanguage = agent.state?.language === "english" ? "spanish" : "english";
agent.setState({ language: newLanguage });
// add a hint message and re-run the agent
// [!code highlight:7]
agent.addMessage({
id: crypto.randomUUID(),
role: "user",
content: `the language has been updated to ${newLanguage}`,
});
await copilotkit.runAgent({ agent });
};
return (
// ...
);
}
By default, the CrewAI Flow agent state will only update between CrewAI Flow node transitions -- which means state updates will be discontinuous and delayed.
You likely want to render the agent state as it updates continuously.