diff options
| author | Adam Malczewski <[email protected]> | 2026-05-19 19:40:21 +0900 |
|---|---|---|
| committer | Adam Malczewski <[email protected]> | 2026-05-19 19:40:21 +0900 |
| commit | f78a91c20f658dd404277919a0b872b352c99bb6 (patch) | |
| tree | 58cfffb655da4443f4b7a39543b86f988f15239f /packages/frontend/src/lib/components/ChatMessage.svelte | |
| download | dispatch-f78a91c20f658dd404277919a0b872b352c99bb6.tar.gz dispatch-f78a91c20f658dd404277919a0b872b352c99bb6.zip | |
- Bun monorepo with @dispatch/core, @dispatch/api, @dispatch/frontend
- Agent runtime with Vercel AI SDK, streaming via WebSocket
- Tools: read_file, write_file, list_files (scoped to working directory)
- Hono API server with POST /chat, GET /status, GET /health, WS /ws
- Svelte 5 + DaisyUI frontend with chat UI, theme switcher, copy button
- OpenCode Go (Zen) as LLM provider, deepseek-v4-flash-free model
- Docker setup (dev + prod) with bin/ scripts and gopass secrets
- Biome v2 linting/formatting, Vitest tests (44 passing)
- Debug info attached to error messages for diagnostics
Diffstat (limited to 'packages/frontend/src/lib/components/ChatMessage.svelte')
| -rw-r--r-- | packages/frontend/src/lib/components/ChatMessage.svelte | 26 |
1 files changed, 26 insertions, 0 deletions
diff --git a/packages/frontend/src/lib/components/ChatMessage.svelte b/packages/frontend/src/lib/components/ChatMessage.svelte new file mode 100644 index 0000000..447bb29 --- /dev/null +++ b/packages/frontend/src/lib/components/ChatMessage.svelte @@ -0,0 +1,26 @@ +<script lang="ts"> +import type { ChatMessage } from "../types.js"; +import ToolCallDisplay from "./ToolCallDisplay.svelte"; + +const { message }: { message: ChatMessage } = $props(); + +const isUser = $derived(message.role === "user"); +</script> + +<div class="chat {isUser ? 'chat-end' : 'chat-start'} mb-2"> + <div class="chat-bubble {isUser ? 'chat-bubble-primary' : 'chat-bubble-secondary'} max-w-[80%] break-words"> + {#if message.content} + <span>{message.content}</span> + {/if} + {#if message.isStreaming} + <span class="inline-block w-2 h-4 bg-current animate-pulse ml-0.5 align-middle">▌</span> + {/if} + {#if message.toolCalls && message.toolCalls.length > 0} + <div class="mt-2"> + {#each message.toolCalls as toolCall (toolCall.id)} + <ToolCallDisplay {toolCall} /> + {/each} + </div> + {/if} + </div> +</div> |
