Spaces:
Runtime error
Runtime error
| <script lang="ts"> | |
| import { type Conversation } from "$lib/types.js"; | |
| import { ScrollState } from "$lib/spells/scroll-state.svelte"; | |
| import { watch } from "runed"; | |
| import { tick } from "svelte"; | |
| import IconPlus from "~icons/carbon/add"; | |
| import CodeSnippets from "./code-snippets.svelte"; | |
| import Message from "./message.svelte"; | |
| interface Props { | |
| conversation: Conversation; | |
| loading: boolean; | |
| viewCode: boolean; | |
| } | |
| let { conversation = $bindable(), loading, viewCode }: Props = $props(); | |
| let messageContainer: HTMLDivElement | null = $state(null); | |
| const scrollState = new ScrollState({ | |
| element: () => messageContainer, | |
| offset: { bottom: 100 }, | |
| }); | |
| const atBottom = $derived(scrollState.arrived.bottom); | |
| watch( | |
| () => conversation.messages.at(-1)?.content, | |
| () => { | |
| const shouldScroll = atBottom && !scrollState.isScrolling; | |
| if (!shouldScroll) return; | |
| try { | |
| tick().then(() => { | |
| scrollState.scrollToBottom(); | |
| }); | |
| } catch { | |
| // noop | |
| } | |
| } | |
| ); | |
| function addMessage() { | |
| const msgs = conversation.messages.slice(); | |
| conversation.messages = [ | |
| ...msgs, | |
| { | |
| role: msgs.at(-1)?.role === "user" ? "assistant" : "user", | |
| content: "", | |
| }, | |
| ]; | |
| conversation = conversation; | |
| } | |
| function deleteMessage(idx: number) { | |
| conversation.messages = conversation.messages.slice(0, idx); | |
| } | |
| </script> | |
| <div | |
| class="@container flex flex-col overflow-x-hidden overflow-y-auto" | |
| class:animate-pulse={loading && !conversation.streaming} | |
| bind:this={messageContainer} | |
| id="test-this" | |
| > | |
| {#if !viewCode} | |
| {#each conversation.messages as _msg, idx} | |
| <Message | |
| bind:message={conversation.messages[idx]!} | |
| {conversation} | |
| autofocus={idx === conversation.messages.length - 1} | |
| {loading} | |
| onDelete={() => deleteMessage(idx)} | |
| /> | |
| {/each} | |
| <button | |
| class="flex px-3.5 py-6 hover:bg-gray-50 md:px-6 dark:hover:bg-gray-800/50" | |
| onclick={addMessage} | |
| disabled={loading} | |
| > | |
| <div class="flex items-center gap-2 p-0! text-sm font-semibold"> | |
| <div class="text-lg"> | |
| <IconPlus /> | |
| </div> | |
| Add message | |
| </div> | |
| </button> | |
| {:else} | |
| <CodeSnippets {conversation} on:closeCode /> | |
| {/if} | |
| </div> | |