Spaces:
Build error
Build error
| <script lang="ts"> | |
| import { DropdownMenu } from 'bits-ui'; | |
| import { createEventDispatcher, getContext, onMount } from 'svelte'; | |
| import { flyAndScale } from '$lib/utils/transitions'; | |
| import { goto } from '$app/navigation'; | |
| import { fade, slide } from 'svelte/transition'; | |
| import { getUsage } from '$lib/apis'; | |
| import { userSignOut } from '$lib/apis/auths'; | |
| import { showSettings, mobile, showSidebar, user } from '$lib/stores'; | |
| import Tooltip from '$lib/components/common/Tooltip.svelte'; | |
| import ArchiveBox from '$lib/components/icons/ArchiveBox.svelte'; | |
| import QuestionMarkCircle from '$lib/components/icons/QuestionMarkCircle.svelte'; | |
| import Map from '$lib/components/icons/Map.svelte'; | |
| import Keyboard from '$lib/components/icons/Keyboard.svelte'; | |
| import ShortcutsModal from '$lib/components/chat/ShortcutsModal.svelte'; | |
| import Settings from '$lib/components/icons/Settings.svelte'; | |
| import Code from '$lib/components/icons/Code.svelte'; | |
| import UserGroup from '$lib/components/icons/UserGroup.svelte'; | |
| import SignOut from '$lib/components/icons/SignOut.svelte'; | |
| const i18n = getContext('i18n'); | |
| export let show = false; | |
| export let role = ''; | |
| export let help = false; | |
| export let className = 'max-w-[240px]'; | |
| let showShortcuts = false; | |
| const dispatch = createEventDispatcher(); | |
| let usage = null; | |
| const getUsageInfo = async () => { | |
| const res = await getUsage(localStorage.token).catch((error) => { | |
| console.error('Error fetching usage info:', error); | |
| }); | |
| if (res) { | |
| usage = res; | |
| } else { | |
| usage = null; | |
| } | |
| }; | |
| $: if (show) { | |
| getUsageInfo(); | |
| } | |
| </script> | |
| <ShortcutsModal bind:show={showShortcuts} /> | |
| <!-- svelte-ignore a11y-no-static-element-interactions --> | |
| <DropdownMenu.Root | |
| bind:open={show} | |
| onOpenChange={(state) => { | |
| dispatch('change', state); | |
| }} | |
| > | |
| <DropdownMenu.Trigger> | |
| <slot /> | |
| </DropdownMenu.Trigger> | |
| <slot name="content"> | |
| <DropdownMenu.Content | |
| class="w-full {className} text-sm rounded-xl px-1 py-1.5 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg font-primary" | |
| sideOffset={4} | |
| side="bottom" | |
| align="start" | |
| transition={(e) => fade(e, { duration: 100 })} | |
| > | |
| <button | |
| class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition" | |
| on:click={async () => { | |
| await showSettings.set(true); | |
| show = false; | |
| if ($mobile) { | |
| showSidebar.set(false); | |
| } | |
| }} | |
| > | |
| <div class=" self-center mr-3"> | |
| <Settings className="w-5 h-5" strokeWidth="1.5" /> | |
| </div> | |
| <div class=" self-center truncate">{$i18n.t('Settings')}</div> | |
| </button> | |
| <button | |
| class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition" | |
| on:click={() => { | |
| dispatch('show', 'archived-chat'); | |
| show = false; | |
| if ($mobile) { | |
| showSidebar.set(false); | |
| } | |
| }} | |
| > | |
| <div class=" self-center mr-3"> | |
| <ArchiveBox className="size-5" strokeWidth="1.5" /> | |
| </div> | |
| <div class=" self-center truncate">{$i18n.t('Archived Chats')}</div> | |
| </button> | |
| {#if role === 'admin'} | |
| <button | |
| class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition" | |
| on:click={() => { | |
| goto('/playground'); | |
| show = false; | |
| if ($mobile) { | |
| showSidebar.set(false); | |
| } | |
| }} | |
| > | |
| <div class=" self-center mr-3"> | |
| <Code className="size-5" strokeWidth="1.5" /> | |
| </div> | |
| <div class=" self-center truncate">{$i18n.t('Playground')}</div> | |
| </button> | |
| <button | |
| class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition" | |
| on:click={() => { | |
| goto('/admin'); | |
| show = false; | |
| if ($mobile) { | |
| showSidebar.set(false); | |
| } | |
| }} | |
| > | |
| <div class=" self-center mr-3"> | |
| <UserGroup className="w-5 h-5" strokeWidth="1.5" /> | |
| </div> | |
| <div class=" self-center truncate">{$i18n.t('Admin Panel')}</div> | |
| </button> | |
| {/if} | |
| {#if help} | |
| <hr class=" border-gray-100 dark:border-gray-800 my-1 p-0" /> | |
| <!-- {$i18n.t('Help')} --> | |
| <DropdownMenu.Item | |
| class="flex gap-2 items-center py-1.5 px-3 text-sm select-none w-full cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-md transition" | |
| id="chat-share-button" | |
| on:click={() => { | |
| window.open('https://docs.openwebui.com', '_blank'); | |
| show = false; | |
| }} | |
| > | |
| <QuestionMarkCircle className="size-5" /> | |
| <div class="flex items-center">{$i18n.t('Documentation')}</div> | |
| </DropdownMenu.Item> | |
| <!-- Releases --> | |
| <DropdownMenu.Item | |
| class="flex gap-2 items-center py-1.5 px-3 text-sm select-none w-full cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-md transition" | |
| id="menu-item-releases" | |
| on:click={() => { | |
| window.open('https://github.com/open-webui/open-webui/releases', '_blank'); | |
| show = false; | |
| }} | |
| > | |
| <Map className="size-5" /> | |
| <div class="flex items-center">{$i18n.t('Releases')}</div> | |
| </DropdownMenu.Item> | |
| <DropdownMenu.Item | |
| class="flex gap-2 items-center py-1.5 px-3 text-sm select-none w-full cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-md transition" | |
| id="chat-share-button" | |
| on:click={() => { | |
| showShortcuts = !showShortcuts; | |
| show = false; | |
| }} | |
| > | |
| <Keyboard className="size-5" /> | |
| <div class="flex items-center">{$i18n.t('Keyboard shortcuts')}</div> | |
| </DropdownMenu.Item> | |
| {/if} | |
| <hr class=" border-gray-100 dark:border-gray-800 my-1 p-0" /> | |
| <button | |
| class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition" | |
| on:click={async () => { | |
| const res = await userSignOut(); | |
| user.set(null); | |
| localStorage.removeItem('token'); | |
| location.href = res?.redirect_url ?? '/auth'; | |
| show = false; | |
| }} | |
| > | |
| <div class=" self-center mr-3"> | |
| <SignOut className="w-5 h-5" strokeWidth="1.5" /> | |
| </div> | |
| <div class=" self-center truncate">{$i18n.t('Sign Out')}</div> | |
| </button> | |
| {#if usage} | |
| {#if usage?.user_ids?.length > 0} | |
| <hr class=" border-gray-100 dark:border-gray-800 my-1 p-0" /> | |
| <Tooltip | |
| content={usage?.model_ids && usage?.model_ids.length > 0 | |
| ? `${$i18n.t('Running')}: ${usage.model_ids.join(', ')} ✨` | |
| : ''} | |
| > | |
| <div | |
| class="flex rounded-md py-1 px-3 text-xs gap-2.5 items-center" | |
| on:mouseenter={() => { | |
| getUsageInfo(); | |
| }} | |
| > | |
| <div class=" flex items-center"> | |
| <span class="relative flex size-2"> | |
| <span | |
| class="animate-ping absolute inline-flex h-full w-full rounded-full bg-green-400 opacity-75" | |
| /> | |
| <span class="relative inline-flex rounded-full size-2 bg-green-500" /> | |
| </span> | |
| </div> | |
| <div class=" "> | |
| <span class=""> | |
| {$i18n.t('Active Users')}: | |
| </span> | |
| <span class=" font-semibold"> | |
| {usage?.user_ids?.length} | |
| </span> | |
| </div> | |
| </div> | |
| </Tooltip> | |
| {/if} | |
| {/if} | |
| <!-- <DropdownMenu.Item class="flex items-center py-1.5 px-3 text-sm "> | |
| <div class="flex items-center">Profile</div> | |
| </DropdownMenu.Item> --> | |
| </DropdownMenu.Content> | |
| </slot> | |
| </DropdownMenu.Root> | |