f42e29fdf1
* wip: server_tools * feat: Integrate with `/tools` endpoint * feat: Builtin + MCP + JSON Schema Tools WIP * refactor * displayName -> display_name * snake_case everywhere * rm redundant field * feat: Improvements * chore: update webui build output * refactor: Updates after server updates * chore: update webui build output * change arg to --tools all * feat: UI improvements * chore: update webui build output * add readme mention * llama-gen-docs * chore: update webui build output * chore: update webui build output * chore: update webui build output * feat: Reorganize settings sections * feat: Separate dialogs for MCP Servers Settings and Import/Export * feat: WIP * feat: WIP * feat: WIP * feat: WIP * feat: WIP * feat: WIP * WIP on allozaur/20677-webui-server-tools * feat: UI improvements * chore: Update package lock * chore: Run `npm audit fix` * feat: UI WIP * feat: UI * refactor: Desktop Icon Strip DRY * feat: Cleaner rendering and transition for ChatScreen * feat: UI improvements * feat: UI improvement * feat: Remove MCP Server "enable" switch from Tools submenu * chore: Run `npm audit fix` * feat: WIP * feat: Logic improvements * refactor: Cleanup * refactor: DRY * test: Fix Chat Sidebar UI Tests * chore: Update package lock * refactor: Cleanup * feat: Chat Message Action Card with Continue and Permission flow implementations * feat: Add agentic steering messages, draft messages and improve chat UX * fix: Search results UI * test: Fix unit test * feat: UI/UX improvements * refactor: Simplify `useToolsPanel` access in components * feat: Implement Processing Info Context API * feat: Implement 'Go back to chat' functionality for settings * feat: Enhance MCP Server management in Chat Form Attachments * style: Minor UI and branding adjustments * chore: Update webui static build output * chore: Formatting, linting & type checks * feat: Draft messages logic * feat: UI improvements * feat: Steering Messages improvements * refactor: Cleanup * refactor: Cleanup * feat: Improve UI * refactor: Settings navigation hook * refactor: DRY code * refactor: DRY ChatMessageUser UI components * refactor: Desktop Icon Strip DRY * refactor: Tools & permissions * fix: Navigation condition * refactor: Cleanup * refactor: Cleanup * refactor: Cleanup * fix: preserve reasoning_content in agentic flow --------- Co-authored-by: Xuan Son Nguyen <son@huggingface.co>
233 lines
6.5 KiB
Svelte
233 lines
6.5 KiB
Svelte
<script lang="ts">
|
|
import '../app.css';
|
|
import { base } from '$app/paths';
|
|
import { browser } from '$app/environment';
|
|
import { page } from '$app/state';
|
|
import { untrack } from 'svelte';
|
|
import { onMount } from 'svelte';
|
|
import { fade } from 'svelte/transition';
|
|
import {
|
|
ChatSidebar,
|
|
DesktopIconStrip,
|
|
DialogConversationTitleUpdate
|
|
} from '$lib/components/app';
|
|
import { conversationsStore } from '$lib/stores/conversations.svelte';
|
|
import * as Sidebar from '$lib/components/ui/sidebar/index.js';
|
|
import * as Tooltip from '$lib/components/ui/tooltip';
|
|
import { isRouterMode, serverStore } from '$lib/stores/server.svelte';
|
|
import { config, settingsStore } from '$lib/stores/settings.svelte';
|
|
import { ModeWatcher } from 'mode-watcher';
|
|
import { Toaster } from 'svelte-sonner';
|
|
import { modelsStore } from '$lib/stores/models.svelte';
|
|
import { mcpStore } from '$lib/stores/mcp.svelte';
|
|
import { TOOLTIP_DELAY_DURATION } from '$lib/constants';
|
|
import { IsMobile } from '$lib/hooks/is-mobile.svelte';
|
|
import { useKeyboardShortcuts } from '$lib/hooks/use-keyboard-shortcuts.svelte';
|
|
import { useSettingsNavigation } from '$lib/hooks/use-settings-navigation.svelte';
|
|
|
|
let { children } = $props();
|
|
|
|
let alwaysShowSidebarOnDesktop = $derived(config().alwaysShowSidebarOnDesktop);
|
|
let isMobile = new IsMobile();
|
|
let isDesktop = $derived(!isMobile.current);
|
|
let sidebarOpen = $state(false);
|
|
let mounted = $state(false);
|
|
let innerHeight = $state<number | undefined>();
|
|
let chatSidebar:
|
|
| { activateSearchMode?: () => void; editActiveConversation?: () => void }
|
|
| undefined = $state();
|
|
|
|
// Conversation title update dialog state
|
|
let titleUpdateDialogOpen = $state(false);
|
|
let titleUpdateCurrentTitle = $state('');
|
|
let titleUpdateNewTitle = $state('');
|
|
let titleUpdateResolve: ((value: boolean) => void) | null = null;
|
|
|
|
const panelNav = useSettingsNavigation();
|
|
|
|
// Global keyboard shortcuts
|
|
const { handleKeydown } = useKeyboardShortcuts({
|
|
editActiveConversation: () => chatSidebar?.editActiveConversation?.()
|
|
});
|
|
|
|
function handleTitleUpdateCancel() {
|
|
titleUpdateDialogOpen = false;
|
|
if (titleUpdateResolve) {
|
|
titleUpdateResolve(false);
|
|
titleUpdateResolve = null;
|
|
}
|
|
}
|
|
|
|
function handleTitleUpdateConfirm() {
|
|
titleUpdateDialogOpen = false;
|
|
if (titleUpdateResolve) {
|
|
titleUpdateResolve(true);
|
|
titleUpdateResolve = null;
|
|
}
|
|
}
|
|
|
|
onMount(() => {
|
|
mounted = true;
|
|
});
|
|
|
|
$effect(() => {
|
|
if (alwaysShowSidebarOnDesktop && isDesktop) {
|
|
sidebarOpen = true;
|
|
return;
|
|
}
|
|
});
|
|
|
|
// Initialize server properties on app load (run once)
|
|
$effect(() => {
|
|
// Only fetch if we don't already have props
|
|
if (!serverStore.props) {
|
|
untrack(() => {
|
|
serverStore.fetch();
|
|
});
|
|
}
|
|
});
|
|
|
|
// Sync settings when server props are loaded
|
|
$effect(() => {
|
|
const serverProps = serverStore.props;
|
|
|
|
if (serverProps) {
|
|
settingsStore.syncWithServerDefaults();
|
|
}
|
|
});
|
|
|
|
// Fetch router models when in router mode (for status and modalities)
|
|
// Wait for models to be loaded first, run only once
|
|
let routerModelsFetched = false;
|
|
|
|
$effect(() => {
|
|
const isRouter = isRouterMode();
|
|
const modelsCount = modelsStore.models.length;
|
|
|
|
// Only fetch router models once when we have models loaded and in router mode
|
|
if (isRouter && modelsCount > 0 && !routerModelsFetched) {
|
|
routerModelsFetched = true;
|
|
untrack(() => {
|
|
modelsStore.fetchRouterModels();
|
|
});
|
|
}
|
|
});
|
|
|
|
// Background MCP server health checks on app load
|
|
// Fetch enabled servers from settings and run health checks in background
|
|
$effect(() => {
|
|
if (!browser) return;
|
|
|
|
const mcpServers = mcpStore.getServers();
|
|
|
|
// Only run health checks if we have enabled servers with URLs
|
|
const enabledServers = mcpServers.filter((s) => s.enabled && s.url.trim());
|
|
|
|
if (enabledServers.length > 0) {
|
|
untrack(() => {
|
|
// Run health checks in background (don't await)
|
|
mcpStore.runHealthChecksForServers(enabledServers, false).catch((error) => {
|
|
console.warn('[layout] MCP health checks failed:', error);
|
|
});
|
|
});
|
|
}
|
|
});
|
|
|
|
// Monitor API key changes and redirect to error page if removed or changed when required
|
|
$effect(() => {
|
|
const apiKey = config().apiKey;
|
|
|
|
if (
|
|
(page.route.id === '/(chat)' || page.route.id === '/(chat)/chat/[id]') &&
|
|
page.status !== 401 &&
|
|
page.status !== 403
|
|
) {
|
|
const headers: Record<string, string> = {
|
|
'Content-Type': 'application/json'
|
|
};
|
|
|
|
if (apiKey && apiKey.trim() !== '') {
|
|
headers.Authorization = `Bearer ${apiKey.trim()}`;
|
|
}
|
|
|
|
fetch(`${base}/props`, { headers })
|
|
.then((response) => {
|
|
if (response.status === 401 || response.status === 403) {
|
|
window.location.reload();
|
|
}
|
|
})
|
|
.catch((e) => {
|
|
console.error('Error checking API key:', e);
|
|
});
|
|
}
|
|
});
|
|
|
|
// Set up title update confirmation callback
|
|
$effect(() => {
|
|
conversationsStore.setTitleUpdateConfirmationCallback(
|
|
async (currentTitle: string, newTitle: string) => {
|
|
return new Promise<boolean>((resolve) => {
|
|
titleUpdateCurrentTitle = currentTitle;
|
|
titleUpdateNewTitle = newTitle;
|
|
titleUpdateResolve = resolve;
|
|
titleUpdateDialogOpen = true;
|
|
});
|
|
}
|
|
);
|
|
});
|
|
</script>
|
|
|
|
<Tooltip.Provider delayDuration={TOOLTIP_DELAY_DURATION}>
|
|
<ModeWatcher />
|
|
|
|
<Toaster richColors />
|
|
|
|
<DialogConversationTitleUpdate
|
|
bind:open={titleUpdateDialogOpen}
|
|
currentTitle={titleUpdateCurrentTitle}
|
|
newTitle={titleUpdateNewTitle}
|
|
onConfirm={handleTitleUpdateConfirm}
|
|
onCancel={handleTitleUpdateCancel}
|
|
/>
|
|
|
|
<Sidebar.Provider bind:open={sidebarOpen}>
|
|
<div class="flex h-screen w-full" style:height="{innerHeight}px">
|
|
<Sidebar.Root variant="floating" class="h-full">
|
|
<ChatSidebar bind:this={chatSidebar} />
|
|
</Sidebar.Root>
|
|
|
|
{#if !(alwaysShowSidebarOnDesktop && isDesktop) && !(panelNav.isSettingsRoute && !isDesktop)}
|
|
{#if mounted}
|
|
<div in:fade={{ duration: 200 }}>
|
|
<Sidebar.Trigger
|
|
class="transition-left absolute left-0 z-[900] duration-200 ease-linear {sidebarOpen
|
|
? 'left-[calc(var(--sidebar-width)+0.75rem)] max-md:hidden'
|
|
: 'left-0!'}"
|
|
style="translate: 1rem 1rem;"
|
|
/>
|
|
</div>
|
|
{/if}
|
|
{/if}
|
|
|
|
{#if isDesktop && !alwaysShowSidebarOnDesktop}
|
|
<DesktopIconStrip
|
|
{sidebarOpen}
|
|
onSearchClick={() => {
|
|
if (chatSidebar?.activateSearchMode) {
|
|
chatSidebar.activateSearchMode();
|
|
}
|
|
|
|
sidebarOpen = true;
|
|
}}
|
|
/>
|
|
{/if}
|
|
|
|
<Sidebar.Inset class="flex flex-1 flex-col overflow-hidden">
|
|
{@render children?.()}
|
|
</Sidebar.Inset>
|
|
</div>
|
|
</Sidebar.Provider>
|
|
</Tooltip.Provider>
|
|
|
|
<svelte:window onkeydown={handleKeydown} bind:innerHeight />
|