ai-sdk-next-openai/app/dynamic-tools/page.tsx
2025-09-26 15:46:29 +00:00

86 lines
2.9 KiB
TypeScript

'use client';
import ChatInput from '@/component/chat-input';
import { useChat } from '@ai-sdk/react';
import { DefaultChatTransport } from 'ai';
import { ToolsMessage } from '../api/dynamic-tools/route';
export default function Chat() {
const { messages, sendMessage, status } = useChat<ToolsMessage>({
transport: new DefaultChatTransport({ api: '/api/dynamic-tools' }),
});
return (
<div className="flex flex-col py-24 mx-auto w-full max-w-md stretch">
{messages?.map(message => (
<div key={message.id} className="whitespace-pre-wrap">
<strong>{`${message.role}: `}</strong>
{message.parts.map((part, index) => {
switch (part.type) {
case 'text':
return <div key={index}>{part.text}</div>;
case 'step-start':
return index > 0 ? (
<div key={index} className="text-gray-500">
<hr className="my-2 border-gray-300" />
</div>
) : null;
case 'dynamic-tool': {
switch (part.state) {
case 'input-streaming':
case 'input-available':
case 'output-available':
return (
<pre key={index}>{JSON.stringify(part, null, 2)}</pre>
);
case 'output-error':
return (
<div key={index} className="text-red-500">
Error: {part.errorText}
</div>
);
}
}
case 'tool-getWeatherInformation': {
switch (part.state) {
// example of pre-rendering streaming tool calls:
case 'input-streaming':
return (
<pre key={index}>
{JSON.stringify(part.input, null, 2)}
</pre>
);
case 'input-available':
return (
<div key={index} className="text-gray-500">
Getting weather information for {part.input.city}...
</div>
);
case 'output-available':
return (
<div key={index} className="text-gray-500">
Weather in {part.input.city}: {part.output}
</div>
);
case 'output-error':
return (
<div key={index} className="text-red-500">
Error: {part.errorText}
</div>
);
}
}
}
})}
<br />
</div>
))}
<ChatInput status={status} onSubmit={text => sendMessage({ text })} />
</div>
);
}