123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248 |
- <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, showShortcuts, 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]';
- 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 })}
- >
- <DropdownMenu.Item
- class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition cursor-pointer"
- 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>
- </DropdownMenu.Item>
- <DropdownMenu.Item
- class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition cursor-pointer"
- 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>
- </DropdownMenu.Item>
- {#if role === 'admin'}
- <DropdownMenu.Item
- as="a"
- href="/playground"
- class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition select-none"
- on:click={() => {
- 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>
- </DropdownMenu.Item>
- <DropdownMenu.Item
- as="a"
- href="/admin"
- class="flex rounded-md py-1.5 px-3 w-full hover:bg-gray-50 dark:hover:bg-gray-800 transition select-none"
- on:click={() => {
- 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>
- </DropdownMenu.Item>
- {/if}
- {#if help}
- <hr class=" border-gray-100 dark:border-gray-800 my-1 p-0" />
- <!-- {$i18n.t('Help')} -->
- <DropdownMenu.Item
- as="a"
- 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={() => {
- show = false;
- }}
- href="https://docs.openwebui.com"
- >
- <QuestionMarkCircle className="size-5" />
- <div class="flex items-center">{$i18n.t('Documentation')}</div>
- </DropdownMenu.Item>
- <!-- Releases -->
- <DropdownMenu.Item
- as="a"
- 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={() => {
- show = false;
- }}
- href="https://github.com/open-webui/"
- >
- <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 cursor-pointer"
- id="chat-share-button"
- on:click={() => {
- showShortcuts.set(!$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" />
- <DropdownMenu.Item
- 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>
- </DropdownMenu.Item>
- {#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>
|