open-webui/src/lib/components/chat/ChatControls.svelte

283 lines
6.7 KiB
Svelte
Raw Normal View History

2024-07-08 23:55:12 +00:00
<script lang="ts">
2024-09-17 20:05:19 +00:00
import { SvelteFlowProvider } from '@xyflow/svelte';
2024-07-08 23:55:12 +00:00
import { slide } from 'svelte/transition';
2024-10-12 08:10:10 +00:00
import { Pane, PaneResizer } from 'paneforge';
2024-09-17 20:05:19 +00:00
import { onDestroy, onMount, tick } from 'svelte';
import { mobile, showControls, showCallOverlay, showOverview, showArtifacts } from '$lib/stores';
2024-09-17 20:05:19 +00:00
2024-07-08 23:55:12 +00:00
import Modal from '../common/Modal.svelte';
import Controls from './Controls/Controls.svelte';
2024-08-23 14:42:36 +00:00
import CallOverlay from './MessageInput/CallOverlay.svelte';
2024-09-05 15:23:59 +00:00
import Drawer from '../common/Drawer.svelte';
2024-09-17 20:05:19 +00:00
import Overview from './Overview.svelte';
2024-09-21 01:33:06 +00:00
import EllipsisVertical from '../icons/EllipsisVertical.svelte';
import Artifacts from './Artifacts.svelte';
2024-10-12 08:10:10 +00:00
import { min } from '@floating-ui/utils';
2024-07-08 23:55:12 +00:00
2024-09-17 20:05:19 +00:00
export let history;
2024-07-12 00:18:18 +00:00
export let models = [];
2024-07-09 02:26:31 +00:00
export let chatId = null;
2024-07-17 09:39:37 +00:00
export let chatFiles = [];
2024-07-09 02:26:31 +00:00
export let params = {};
2024-07-08 23:55:12 +00:00
2024-08-23 14:42:36 +00:00
export let eventTarget: EventTarget;
export let submitPrompt: Function;
export let stopResponse: Function;
2024-09-18 01:13:37 +00:00
export let showMessage: Function;
2024-08-23 14:42:36 +00:00
export let files;
export let modelId;
2024-09-21 01:33:06 +00:00
export let pane;
let mediaQuery;
let largeScreen = false;
let dragged = false;
2024-07-08 23:55:12 +00:00
2024-10-12 08:10:10 +00:00
let minSize = 0;
export const openPane = () => {
if (parseInt(localStorage?.chatControlsSize)) {
pane.resize(parseInt(localStorage?.chatControlsSize));
} 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);
2024-07-08 23:55:12 +00:00
}
pane = null;
}
};
2024-07-08 23:55:12 +00:00
const onMouseDown = (event) => {
dragged = true;
};
const onMouseUp = (event) => {
dragged = false;
};
2024-07-08 23:55:12 +00:00
onMount(() => {
// listen to resize 1024px
mediaQuery = window.matchMedia('(min-width: 1024px)');
mediaQuery.addEventListener('change', handleMediaQuery);
2024-07-08 23:55:12 +00:00
handleMediaQuery(mediaQuery);
2024-10-12 08:10:10 +00:00
// 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 200px
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);
}
}
}
});
// Start observing the container's size changes
resizeObserver.observe(container);
document.addEventListener('mousedown', onMouseDown);
document.addEventListener('mouseup', onMouseUp);
2024-07-08 23:55:12 +00:00
});
2024-09-17 21:02:41 +00:00
onDestroy(() => {
showControls.set(false);
mediaQuery.removeEventListener('change', handleMediaQuery);
document.removeEventListener('mousedown', onMouseDown);
document.removeEventListener('mouseup', onMouseUp);
2024-09-17 21:02:41 +00:00
});
2024-09-17 23:57:27 +00:00
const closeHandler = () => {
2024-10-06 07:09:37 +00:00
showControls.set(false);
2024-09-17 23:57:27 +00:00
showOverview.set(false);
2024-10-06 07:09:37 +00:00
showArtifacts.set(false);
if ($showCallOverlay) {
showCallOverlay.set(false);
}
};
$: if (!chatId) {
closeHandler();
2024-09-17 23:57:27 +00:00
}
2024-07-08 23:55:12 +00:00
</script>
2024-09-17 20:05:19 +00:00
<SvelteFlowProvider>
{#if !largeScreen}
2024-09-21 01:33:06 +00:00
{#if $showControls}
2024-09-17 20:05:19 +00:00
<Drawer
2024-09-17 22:18:47 +00:00
show={$showControls}
2024-09-17 20:05:19 +00:00
on:close={() => {
showControls.set(false);
}}
>
2024-09-21 01:33:06 +00:00
<div
class=" {$showCallOverlay || $showOverview || $showArtifacts
? ' h-screen w-screen'
: 'px-6 py-4'} h-full"
2024-09-21 01:33:06 +00:00
>
{#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 $showArtifacts}
2024-10-06 08:12:30 +00:00
<Artifacts {history} />
2024-09-21 01:33:06 +00:00
{:else if $showOverview}
2024-09-17 23:25:28 +00:00
<Overview
2024-09-18 01:47:04 +00:00
{history}
on:nodeclick={(e) => {
showMessage(e.detail.node.data.message);
}}
2024-09-17 23:25:28 +00:00
on:close={() => {
showControls.set(false);
}}
/>
{:else}
<Controls
on:close={() => {
showControls.set(false);
}}
{models}
bind:chatFiles
bind:params
/>
{/if}
2024-09-17 20:05:19 +00:00
</div>
</Drawer>
{/if}
2024-09-21 01:33:06 +00:00
{:else}
<!-- if $showControls -->
{#if $showControls}
<PaneResizer class="relative flex w-2 items-center justify-center bg-background group">
<div class="z-10 flex h-7 w-5 items-center justify-center rounded-sm">
<EllipsisVertical className="size-4 invisible group-hover:visible" />
</div>
</PaneResizer>
{/if}
2024-10-12 08:10:10 +00:00
2024-09-21 01:33:06 +00:00
<Pane
bind:pane
2024-10-12 08:10:10 +00:00
defaultSize={0}
2024-09-21 01:33:06 +00:00
onResize={(size) => {
2024-10-12 08:10:10 +00:00
console.log('size', size, minSize);
if ($showControls && pane.isExpanded()) {
if (size < minSize) {
pane.resize(minSize);
}
if (size < minSize) {
localStorage.chatControlsSize = 0;
} else {
localStorage.chatControlsSize = size;
2024-09-21 01:33:06 +00:00
}
}
}}
2024-10-12 08:10:10 +00:00
onCollapse={() => {
showControls.set(false);
}}
collapsible={true}
2024-10-05 21:49:39 +00:00
class="pt-8"
2024-09-21 01:33:06 +00:00
>
2024-09-21 02:10:24 +00:00
{#if $showControls}
<div class="pr-4 pb-8 flex max-h-full min-h-full">
<div
class="w-full {($showOverview || $showArtifacts) && !$showCallOverlay
2024-09-21 02:10:24 +00:00
? ' '
: 'px-4 py-4 bg-white dark:shadow-lg dark:bg-gray-850 border border-gray-50 dark:border-gray-800'} rounded-lg z-40 pointer-events-auto overflow-y-auto scrollbar-hidden"
2024-09-21 02:10:24 +00:00
>
{#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 $showArtifacts}
2024-10-06 08:12:30 +00:00
<Artifacts {history} overlay={dragged} />
2024-09-21 02:10:24 +00:00
{:else if $showOverview}
<Overview
{history}
on:nodeclick={(e) => {
2024-09-22 00:53:38 +00:00
if (e.detail.node.data.message.favorite) {
history.messages[e.detail.node.data.message.id].favorite = true;
} else {
history.messages[e.detail.node.data.message.id].favorite = null;
}
2024-09-21 02:10:24 +00:00
showMessage(e.detail.node.data.message);
}}
2024-09-21 01:44:44 +00:00
on:close={() => {
showControls.set(false);
}}
/>
2024-09-21 02:10:24 +00:00
{:else}
<Controls
on:close={() => {
showControls.set(false);
}}
{models}
bind:chatFiles
bind:params
/>
{/if}
</div>
2024-09-17 20:05:19 +00:00
</div>
2024-09-21 02:10:24 +00:00
{/if}
2024-09-21 01:33:06 +00:00
</Pane>
2024-09-17 20:05:19 +00:00
{/if}
</SvelteFlowProvider>