showcase/shell-docs/src/content/docs/integrations/deepagents/shared-state/in-app-agent-write.mdx
<IframeSwitcher id="shared-state-example" exampleUrl="https://feature-viewer.copilotkit.ai/langgraph/feature/shared_state?sidebar=false&chatDefaultOpen=false" codeUrl="https://feature-viewer.copilotkit.ai/langgraph/feature/shared_state?view=code&sidebar=false&codeLayout=tabs" exampleLabel="Demo" codeLabel="Code" height="700px" />
<Callout type="info"> This example demonstrates writing to shared state in the [CopilotKit Feature Viewer](https://feature-viewer.copilotkit.ai/langgraph/feature/shared_state). </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.
<Tabs groupId="agent_language" items={["Python", "TypeScript"]} persist>
<Tab value="Python">
```python title="agent.py"
from copilotkit import CopilotKitState
from typing import Literal
class AgentState(CopilotKitState):
language: Literal["english", "spanish"] = "english"
```
</Tab>
<Tab value="TypeScript">
```ts title="agent.ts"
import { createMiddleware } from "langchain";
import { copilotkitMiddleware } from "@copilotkit/sdk-js/langgraph"; // [!code highlight]
import { z } from "zod";
export const languageStateMiddleware = createMiddleware({
name: "AgentState",
stateSchema: z.object({
language: z.enum(["english", "spanish"]).default("english"),
}),
});
// Compose with copilotkitMiddleware when constructing the agent:
// createDeepAgent({ middleware: [languageStateMiddleware, copilotkitMiddleware], ... })
```
</Tab>
</Tabs>
```tsx title="ui/app/page.tsx"
import { useAgent } from "@copilotkit/react-core/v2"; // [!code highlight]
// Example usage in a pseudo React component
function YourMainContent() {
// [!code highlight:3]
const { agent } = useAgent({
agentId: "sample_agent",
});
const language = (agent.state.language as string) ?? "english";
// ...
const toggleLanguage = () => {
agent.setState({ language: language === "english" ? "spanish" : "english" }); // [!code highlight]
};
// ...
return (
// style excluded for brevity
<div>
<h1>Your main content</h1>
<p>Language: {language}</p>
<button onClick={toggleLanguage}>Toggle Language</button>
</div>
);
}
```
<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](#implementation) section applied to it!
</Callout>
The new agent state will be used next time the agent runs.
If you want to re-run it manually, use agent.runAgent after updating state.
import { useAgent } from "@copilotkit/react-core/v2"; // [!code highlight]
// ...
function YourMainContent() {
const { agent } = useAgent({
agentId: "sample_agent",
});
const language = (agent.state.language as string) ?? "english";
// setup to be called when some event in the app occurs
const toggleLanguage = () => {
const newLanguage = language === "english" ? "spanish" : "english";
agent.setState({ language: newLanguage });
// [!code highlight:2]
// re-run the agent with updated state
agent.runAgent();
};
return (
// ...
);
}
By default, the Deep Agents agent state will only update between LangGraph node transitions -- which means state updates will be discontinuous and delayed.
You likely want to render the agent state as it updates continuously.