| | <script lang="ts"> |
| | import { SvelteFlowProvider } from '@xyflow/svelte'; |
| | import { slide } from 'svelte/transition'; |
| | import { Pane, PaneResizer } from 'paneforge'; |
| | |
| | import { onDestroy, onMount, tick } from 'svelte'; |
| | import { |
| | mobile, |
| | showControls, |
| | showCallOverlay, |
| | showOverview, |
| | showArtifacts, |
| | showEmbeds |
| | } from '$lib/stores'; |
| | |
| | import Controls from './Controls/Controls.svelte'; |
| | import CallOverlay from './MessageInput/CallOverlay.svelte'; |
| | import Drawer from '../common/Drawer.svelte'; |
| | import Artifacts from './Artifacts.svelte'; |
| | import Embeds from './ChatControls/Embeds.svelte'; |
| | |
| | export let history; |
| | export let models = []; |
| | |
| | export let chatId = null; |
| | |
| | export let chatFiles = []; |
| | export let params = {}; |
| | |
| | export let eventTarget: EventTarget; |
| | export let submitPrompt: Function; |
| | export let stopResponse: Function; |
| | export let showMessage: Function; |
| | export let files; |
| | export let modelId; |
| | |
| | export let pane; |
| | |
| | let mediaQuery; |
| | let largeScreen = false; |
| | let dragged = false; |
| | |
| | let minSize = 0; |
| | |
| | export const openPane = () => { |
| | if (parseInt(localStorage?.chatControlsSize)) { |
| | const container = document.getElementById('chat-container'); |
| | let size = Math.floor( |
| | (parseInt(localStorage?.chatControlsSize) / container.clientWidth) * 100 |
| | ); |
| | pane.resize(size); |
| | } else { |
| | pane.resize(minSize); |
| | } |
| | }; |
| | |
| | const handleMediaQuery = async (e) => { |
| | if (e.matches) { |
| | largeScreen = true; |
| | |
| | if ($showCallOverlay) { |
| | showCallOverlay.set(false); |
| | await tick(); |
| | showCallOverlay.set(true); |
| | } |
| | } else { |
| | largeScreen = false; |
| | |
| | if ($showCallOverlay) { |
| | showCallOverlay.set(false); |
| | await tick(); |
| | showCallOverlay.set(true); |
| | } |
| | pane = null; |
| | } |
| | }; |
| | |
| | const onMouseDown = (event) => { |
| | dragged = true; |
| | }; |
| | |
| | const onMouseUp = (event) => { |
| | dragged = false; |
| | }; |
| | |
| | onMount(() => { |
| | // listen to resize 1024px |
| | mediaQuery = window.matchMedia('(min-width: 1024px)'); |
| | |
| | mediaQuery.addEventListener('change', handleMediaQuery); |
| | handleMediaQuery(mediaQuery); |
| | |
| | // Select the container element you want to observe |
| | const container = document.getElementById('chat-container'); |
| | |
| | // initialize the minSize based on the container width |
| | minSize = Math.floor((350 / container.clientWidth) * 100); |
| | |
| | // Create a new ResizeObserver instance |
| | const resizeObserver = new ResizeObserver((entries) => { |
| | for (let entry of entries) { |
| | const width = entry.contentRect.width; |
| | // calculate the percentage of 350px |
| | const percentage = (350 / width) * 100; |
| | // set the minSize to the percentage, must be an integer |
| | minSize = Math.floor(percentage); |
| | |
| | if ($showControls) { |
| | if (pane && pane.isExpanded() && pane.getSize() < minSize) { |
| | pane.resize(minSize); |
| | } else { |
| | let size = Math.floor( |
| | (parseInt(localStorage?.chatControlsSize) / container.clientWidth) * 100 |
| | ); |
| | if (size < minSize) { |
| | pane.resize(minSize); |
| | } |
| | } |
| | } |
| | } |
| | }); |
| | |
| | |
| | resizeObserver.observe(container); |
| | |
| | document.addEventListener('mousedown', onMouseDown); |
| | document.addEventListener('mouseup', onMouseUp); |
| | }); |
| | |
| | onDestroy(() => { |
| | showControls.set(false); |
| | |
| | mediaQuery.removeEventListener('change', handleMediaQuery); |
| | document.removeEventListener('mousedown', onMouseDown); |
| | document.removeEventListener('mouseup', onMouseUp); |
| | }); |
| | |
| | const closeHandler = () => { |
| | showControls.set(false); |
| | showOverview.set(false); |
| | showArtifacts.set(false); |
| | showEmbeds.set(false); |
| | |
| | if ($showCallOverlay) { |
| | showCallOverlay.set(false); |
| | } |
| | }; |
| | |
| | $: if (!chatId) { |
| | closeHandler(); |
| | } |
| | </script> |
| |
|
| | {#if !largeScreen} |
| | {#if $showControls} |
| | <Drawer |
| | show={$showControls} |
| | onClose={() => { |
| | showControls.set(false); |
| | }} |
| | > |
| | <div |
| | class=" {$showCallOverlay || $showOverview || $showArtifacts || $showEmbeds |
| | ? ' h-screen w-full' |
| | : 'px-4 py-3'} h-full" |
| | > |
| | {#if $showCallOverlay} |
| | <div |
| | class=" h-full max-h-[100dvh] bg-white text-gray-700 dark:bg-black dark:text-gray-300 flex justify-center" |
| | > |
| | <CallOverlay |
| | bind:files |
| | {submitPrompt} |
| | {stopResponse} |
| | {modelId} |
| | {chatId} |
| | {eventTarget} |
| | on:close={() => { |
| | showControls.set(false); |
| | }} |
| | /> |
| | </div> |
| | {:else if $showEmbeds} |
| | <Embeds /> |
| | {:else if $showArtifacts} |
| | <Artifacts {history} /> |
| | {:else if $showOverview} |
| | {#await import('./Overview.svelte') then { default: Overview }} |
| | <Overview |
| | {history} |
| | onNodeClick={(e) => { |
| | const node = e.node; |
| | showMessage(node.data.message, true); |
| | }} |
| | onClose={() => { |
| | showControls.set(false); |
| | }} |
| | /> |
| | {/await} |
| | {:else} |
| | <Controls |
| | on:close={() => { |
| | showControls.set(false); |
| | }} |
| | {models} |
| | bind:chatFiles |
| | bind:params |
| | /> |
| | {/if} |
| | </div> |
| | </Drawer> |
| | {/if} |
| | {:else} |
| | <!-- if $showControls --> |
| |
|
| | {#if $showControls} |
| | <PaneResizer |
| | class="relative flex items-center justify-center group border-l border-gray-50 dark:border-gray-850/30 hover:border-gray-200 dark:hover:border-gray-800 transition z-20" |
| | id="controls-resizer" |
| | > |
| | <div |
| | class=" absolute -left-1.5 -right-1.5 -top-0 -bottom-0 z-20 cursor-col-resize bg-transparent" |
| | /> |
| | </PaneResizer> |
| | {/if} |
| |
|
| | <Pane |
| | bind:pane |
| | defaultSize={0} |
| | onResize={(size) => { |
| | if ($showControls && pane.isExpanded()) { |
| | if (size < minSize) { |
| | pane.resize(minSize); |
| | } |
| |
|
| | if (size < minSize) { |
| | localStorage.chatControlsSize = 0; |
| | } else { |
| | // save the size in pixels to localStorage |
| | const container = document.getElementById('chat-container'); |
| | localStorage.chatControlsSize = Math.floor((size / 100) * container.clientWidth); |
| | } |
| | } |
| | }} |
| | onCollapse={() => { |
| | showControls.set(false); |
| | }} |
| | collapsible={true} |
| | class=" z-10 bg-white dark:bg-gray-850" |
| | > |
| | {#if $showControls} |
| | <div class="flex max-h-full min-h-full"> |
| | <div |
| | class="w-full {($showOverview || $showArtifacts || $showEmbeds) && !$showCallOverlay |
| | ? ' ' |
| | : 'px-4 py-3 bg-white dark:shadow-lg dark:bg-gray-850 '} z-40 pointer-events-auto overflow-y-auto scrollbar-hidden" |
| | id="controls-container" |
| | > |
| | {#if $showCallOverlay} |
| | <div class="w-full h-full flex justify-center"> |
| | <CallOverlay |
| | bind:files |
| | {submitPrompt} |
| | {stopResponse} |
| | {modelId} |
| | {chatId} |
| | {eventTarget} |
| | on:close={() => { |
| | showControls.set(false); |
| | }} |
| | /> |
| | </div> |
| | {:else if $showEmbeds} |
| | <Embeds overlay={dragged} /> |
| | {:else if $showArtifacts} |
| | <Artifacts {history} overlay={dragged} /> |
| | {:else if $showOverview} |
| | {#await import('./Overview.svelte') then { default: Overview }} |
| | <Overview |
| | {history} |
| | onNodeClick={(e) => { |
| | const node = e.node; |
| | if (node?.data?.message?.favorite) { |
| | history.messages[node.data.message.id].favorite = true; |
| | } else { |
| | history.messages[node.data.message.id].favorite = null; |
| | } |
| | |
| | showMessage(node.data.message, true); |
| | }} |
| | onClose={() => { |
| | showControls.set(false); |
| | }} |
| | /> |
| | {/await} |
| | {:else} |
| | <Controls |
| | on:close={() => { |
| | showControls.set(false); |
| | }} |
| | {models} |
| | bind:chatFiles |
| | bind:params |
| | /> |
| | {/if} |
| | </div> |
| | </div> |
| | {/if} |
| | </Pane> |
| | {/if} |
| |
|