Spaces:
Running
on
CPU Upgrade
Running
on
CPU Upgrade
inference-playground
/
src
/lib
/components
/InferencePlayground
/InferencePlaygroundConversation.svelte
| <script lang="ts"> | |
| import type { Conversation } from "$lib/types"; | |
| import { createEventDispatcher } from "svelte"; | |
| import CodeSnippets from "./InferencePlaygroundCodeSnippets.svelte"; | |
| import Message from "./InferencePlaygroundMessage.svelte"; | |
| import IconPlus from "../Icons/IconPlus.svelte"; | |
| export let conversation: Conversation; | |
| export let loading: boolean; | |
| export let viewCode: boolean; | |
| export let hfToken: string; | |
| let shouldScrollToBottom = true; | |
| let isProgrammaticScroll = true; | |
| let conversationLength = conversation.messages.length; | |
| const dispatch = createEventDispatcher<{ | |
| addMessage: void; | |
| deleteMessage: number; | |
| }>(); | |
| let messageContainer: HTMLDivElement | null = null; | |
| function resizeMessageTextAreas() { | |
| // ideally we would use CSS "field-sizing:content". However, it is currently only supported on Chrome. | |
| if (messageContainer) { | |
| const containerScrollTop = messageContainer.scrollTop; | |
| const textareaEls = messageContainer.querySelectorAll("textarea"); | |
| for (const textarea of textareaEls) { | |
| textarea.style.height = "0px"; | |
| textarea.style.height = textarea.scrollHeight + "px"; | |
| } | |
| messageContainer.scrollTop = containerScrollTop; | |
| } | |
| } | |
| function scrollToBottom() { | |
| if (messageContainer) { | |
| isProgrammaticScroll = true; | |
| messageContainer.scrollTop = messageContainer.scrollHeight; | |
| } | |
| } | |
| $: { | |
| if (conversation.messages.at(-1)) { | |
| resizeMessageTextAreas(); | |
| if (shouldScrollToBottom) { | |
| scrollToBottom(); | |
| } | |
| } | |
| } | |
| $: if (conversation.messages.length !== conversationLength) { | |
| // enable automatic scrolling when new message was added | |
| conversationLength = conversation.messages.length; | |
| shouldScrollToBottom = true; | |
| } | |
| </script> | |
| <div | |
| class="flex max-h-[calc(100dvh-5.8rem)] flex-col overflow-y-auto overflow-x-hidden @container" | |
| class:animate-pulse={loading && !conversation.streaming} | |
| bind:this={messageContainer} | |
| on:scroll={() => { | |
| // disable automatic scrolling is user initiates scroll | |
| if (!isProgrammaticScroll) { | |
| shouldScrollToBottom = false; | |
| } | |
| isProgrammaticScroll = false; | |
| }} | |
| > | |
| {#if !viewCode} | |
| {#each conversation.messages as message, messageIdx} | |
| <Message | |
| class="border-b" | |
| {message} | |
| {loading} | |
| on:input={resizeMessageTextAreas} | |
| on:delete={() => dispatch("deleteMessage", messageIdx)} | |
| autofocus={!loading && messageIdx === conversation.messages.length - 1} | |
| /> | |
| {/each} | |
| <button | |
| class="flex px-3.5 py-6 hover:bg-gray-50 md:px-6 dark:hover:bg-gray-800/50" | |
| on:click={() => dispatch("addMessage")} | |
| disabled={loading} | |
| > | |
| <div class="flex items-center gap-2 !p-0 text-sm font-semibold"> | |
| <IconPlus classNames="text-lg" /> Add message | |
| </div> | |
| </button> | |
| {:else} | |
| <CodeSnippets {conversation} {hfToken} /> | |
| {/if} | |
| </div> | |