mirror of
https://github.com/open-webui/open-webui.git
synced 2025-12-12 04:15:25 +00:00
refac
This commit is contained in:
parent
caaaa07232
commit
3ed0a6d11f
3 changed files with 215 additions and 203 deletions
|
|
@ -919,6 +919,7 @@
|
||||||
|
|
||||||
<FilesOverlay show={dragged} />
|
<FilesOverlay show={dragged} />
|
||||||
<ToolServersModal bind:show={showTools} {selectedToolIds} />
|
<ToolServersModal bind:show={showTools} {selectedToolIds} />
|
||||||
|
|
||||||
<InputVariablesModal
|
<InputVariablesModal
|
||||||
bind:show={showInputVariablesModal}
|
bind:show={showInputVariablesModal}
|
||||||
variables={inputVariables}
|
variables={inputVariables}
|
||||||
|
|
|
||||||
|
|
@ -81,7 +81,7 @@
|
||||||
|
|
||||||
<div slot="content">
|
<div slot="content">
|
||||||
<DropdownMenu.Content
|
<DropdownMenu.Content
|
||||||
class="w-full max-w-[200px] rounded-2xl px-1 py-1 border border-gray-100 dark:border-gray-800 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg transition"
|
class="w-full max-w-[240px] rounded-2xl px-1 py-1 border border-gray-100 dark:border-gray-800 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg transition"
|
||||||
sideOffset={4}
|
sideOffset={4}
|
||||||
alignOffset={-6}
|
alignOffset={-6}
|
||||||
side="bottom"
|
side="bottom"
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,7 @@
|
||||||
import Terminal from '$lib/components/icons/Terminal.svelte';
|
import Terminal from '$lib/components/icons/Terminal.svelte';
|
||||||
import ChevronRight from '$lib/components/icons/ChevronRight.svelte';
|
import ChevronRight from '$lib/components/icons/ChevronRight.svelte';
|
||||||
import ChevronLeft from '$lib/components/icons/ChevronLeft.svelte';
|
import ChevronLeft from '$lib/components/icons/ChevronLeft.svelte';
|
||||||
|
import { crossfade, fade, fly, slide } from 'svelte/transition';
|
||||||
|
|
||||||
const i18n = getContext('i18n');
|
const i18n = getContext('i18n');
|
||||||
|
|
||||||
|
|
@ -39,9 +40,10 @@
|
||||||
|
|
||||||
export let onClose: Function;
|
export let onClose: Function;
|
||||||
|
|
||||||
let tools = null;
|
|
||||||
let show = false;
|
let show = false;
|
||||||
let showAllTools = false;
|
let tab = '';
|
||||||
|
|
||||||
|
let tools = null;
|
||||||
|
|
||||||
$: if (show) {
|
$: if (show) {
|
||||||
init();
|
init();
|
||||||
|
|
@ -81,129 +83,114 @@
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<div slot="content">
|
<div slot="content">
|
||||||
<DropdownMenu.Content
|
<DropdownMenu.Content
|
||||||
class="w-full max-w-[240px] rounded-2xl px-1 py-1 border border-gray-100 dark:border-gray-800 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg max-h-72 overflow-y-auto scrollbar-thin"
|
class="w-full max-w-[240px] rounded-2xl px-1 py-1 border border-gray-100 dark:border-gray-800 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg max-h-72 overflow-y-auto overflow-x-hidden scrollbar-thin"
|
||||||
sideOffset={4}
|
sideOffset={4}
|
||||||
alignOffset={-6}
|
alignOffset={-6}
|
||||||
side="bottom"
|
side="bottom"
|
||||||
align="start"
|
align="start"
|
||||||
transition={flyAndScale}
|
transition={flyAndScale}
|
||||||
>
|
>
|
||||||
{#if tools}
|
{#if tab === ''}
|
||||||
{#if Object.keys(tools).length > 0}
|
<div in:fly={{ x: -20, duration: 150 }}>
|
||||||
<button
|
{#if tools}
|
||||||
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
{#if Object.keys(tools).length > 0}
|
||||||
on:click={() => {
|
|
||||||
showAllTools = !showAllTools;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{#if !showAllTools}
|
|
||||||
<Wrench />
|
|
||||||
|
|
||||||
<div class="flex items-center w-full justify-between">
|
|
||||||
<div>
|
|
||||||
{$i18n.t('Tools')}
|
|
||||||
<span class="ml-0.5 text-gray-500">{Object.keys(tools).length}</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="text-gray-500">
|
|
||||||
<ChevronRight />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{:else}
|
|
||||||
<ChevronLeft />
|
|
||||||
|
|
||||||
<div class="flex items-center w-full justify-between">
|
|
||||||
<div>
|
|
||||||
{$i18n.t('Tools')}
|
|
||||||
<span class="ml-0.5 text-gray-500">{Object.keys(tools).length}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
</button>
|
|
||||||
{/if}
|
|
||||||
{:else}
|
|
||||||
<div class="py-4">
|
|
||||||
<Spinner />
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
{#if showAllTools}
|
|
||||||
{#each Object.keys(tools) as toolId}
|
|
||||||
<button
|
|
||||||
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
|
||||||
on:click={() => {
|
|
||||||
tools[toolId].enabled = !tools[toolId].enabled;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div class="flex-1 truncate">
|
|
||||||
<div class="flex flex-1 gap-2 items-center">
|
|
||||||
<Tooltip content={tools[toolId]?.name ?? ''} placement="top">
|
|
||||||
<div class="shrink-0">
|
|
||||||
<Wrench />
|
|
||||||
</div>
|
|
||||||
</Tooltip>
|
|
||||||
<Tooltip content={tools[toolId]?.description ?? ''} placement="top-start">
|
|
||||||
<div class=" truncate">{tools[toolId].name}</div>
|
|
||||||
</Tooltip>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class=" shrink-0">
|
|
||||||
<Switch
|
|
||||||
state={tools[toolId].enabled}
|
|
||||||
on:change={async (e) => {
|
|
||||||
const state = e.detail;
|
|
||||||
await tick();
|
|
||||||
if (state) {
|
|
||||||
selectedToolIds = [...selectedToolIds, toolId];
|
|
||||||
} else {
|
|
||||||
selectedToolIds = selectedToolIds.filter((id) => id !== toolId);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
{/each}
|
|
||||||
{:else}
|
|
||||||
{#if toggleFilters && toggleFilters.length > 0}
|
|
||||||
{#each toggleFilters.sort( (a, b) => a.name.localeCompare( b.name, undefined, { sensitivity: 'base' } ) ) as filter, filterIdx (filter.id)}
|
|
||||||
<Tooltip content={filter?.description} placement="top-start">
|
|
||||||
<button
|
<button
|
||||||
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||||
on:click={() => {
|
on:click={() => {
|
||||||
if (selectedFilterIds.includes(filter.id)) {
|
tab = 'tools';
|
||||||
selectedFilterIds = selectedFilterIds.filter((id) => id !== filter.id);
|
}}
|
||||||
} else {
|
>
|
||||||
selectedFilterIds = [...selectedFilterIds, filter.id];
|
<Wrench />
|
||||||
}
|
|
||||||
|
<div class="flex items-center w-full justify-between">
|
||||||
|
<div>
|
||||||
|
{$i18n.t('Tools')}
|
||||||
|
<span class="ml-0.5 text-gray-500">{Object.keys(tools).length}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="text-gray-500">
|
||||||
|
<ChevronRight />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
{/if}
|
||||||
|
{:else}
|
||||||
|
<div class="py-4">
|
||||||
|
<Spinner />
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if toggleFilters && toggleFilters.length > 0}
|
||||||
|
{#each toggleFilters.sort( (a, b) => a.name.localeCompare( b.name, undefined, { sensitivity: 'base' } ) ) as filter, filterIdx (filter.id)}
|
||||||
|
<Tooltip content={filter?.description} placement="top-start">
|
||||||
|
<button
|
||||||
|
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||||
|
on:click={() => {
|
||||||
|
if (selectedFilterIds.includes(filter.id)) {
|
||||||
|
selectedFilterIds = selectedFilterIds.filter((id) => id !== filter.id);
|
||||||
|
} else {
|
||||||
|
selectedFilterIds = [...selectedFilterIds, filter.id];
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div class="flex-1 truncate">
|
||||||
|
<div class="flex flex-1 gap-2 items-center">
|
||||||
|
<div class="shrink-0">
|
||||||
|
{#if filter?.icon}
|
||||||
|
<div class="size-4 items-center flex justify-center">
|
||||||
|
<img
|
||||||
|
src={filter.icon}
|
||||||
|
class="size-3.5 {filter.icon.includes('svg')
|
||||||
|
? 'dark:invert-[80%]'
|
||||||
|
: ''}"
|
||||||
|
style="fill: currentColor;"
|
||||||
|
alt={filter.name}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{:else}
|
||||||
|
<Sparkles className="size-4" strokeWidth="1.75" />
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=" truncate">{filter?.name}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=" shrink-0">
|
||||||
|
<Switch
|
||||||
|
state={selectedFilterIds.includes(filter.id)}
|
||||||
|
on:change={async (e) => {
|
||||||
|
const state = e.detail;
|
||||||
|
await tick();
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</Tooltip>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if showWebSearchButton}
|
||||||
|
<Tooltip content={$i18n.t('Search the internet')} placement="top-start">
|
||||||
|
<button
|
||||||
|
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||||
|
on:click={() => {
|
||||||
|
webSearchEnabled = !webSearchEnabled;
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div class="flex-1 truncate">
|
<div class="flex-1 truncate">
|
||||||
<div class="flex flex-1 gap-2 items-center">
|
<div class="flex flex-1 gap-2 items-center">
|
||||||
<div class="shrink-0">
|
<div class="shrink-0">
|
||||||
{#if filter?.icon}
|
<GlobeAlt />
|
||||||
<div class="size-4 items-center flex justify-center">
|
|
||||||
<img
|
|
||||||
src={filter.icon}
|
|
||||||
class="size-3.5 {filter.icon.includes('svg')
|
|
||||||
? 'dark:invert-[80%]'
|
|
||||||
: ''}"
|
|
||||||
style="fill: currentColor;"
|
|
||||||
alt={filter.name}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
{:else}
|
|
||||||
<Sparkles className="size-4" strokeWidth="1.75" />
|
|
||||||
{/if}
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class=" truncate">{filter?.name}</div>
|
<div class=" truncate">{$i18n.t('Web Search')}</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class=" shrink-0">
|
<div class=" shrink-0">
|
||||||
<Switch
|
<Switch
|
||||||
state={selectedFilterIds.includes(filter.id)}
|
state={webSearchEnabled}
|
||||||
on:change={async (e) => {
|
on:change={async (e) => {
|
||||||
const state = e.detail;
|
const state = e.detail;
|
||||||
await tick();
|
await tick();
|
||||||
|
|
@ -212,105 +199,129 @@
|
||||||
</div>
|
</div>
|
||||||
</button>
|
</button>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if showImageGenerationButton}
|
||||||
|
<Tooltip content={$i18n.t('Generate an image')} placement="top-start">
|
||||||
|
<button
|
||||||
|
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||||
|
on:click={() => {
|
||||||
|
imageGenerationEnabled = !imageGenerationEnabled;
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div class="flex-1 truncate">
|
||||||
|
<div class="flex flex-1 gap-2 items-center">
|
||||||
|
<div class="shrink-0">
|
||||||
|
<Photo className="size-4" strokeWidth="1.5" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=" truncate">{$i18n.t('Image')}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=" shrink-0">
|
||||||
|
<Switch
|
||||||
|
state={imageGenerationEnabled}
|
||||||
|
on:change={async (e) => {
|
||||||
|
const state = e.detail;
|
||||||
|
await tick();
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</Tooltip>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if showCodeInterpreterButton}
|
||||||
|
<Tooltip content={$i18n.t('Execute code for analysis')} placement="top-start">
|
||||||
|
<button
|
||||||
|
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||||
|
aria-pressed={codeInterpreterEnabled}
|
||||||
|
aria-label={codeInterpreterEnabled
|
||||||
|
? $i18n.t('Disable Code Interpreter')
|
||||||
|
: $i18n.t('Enable Code Interpreter')}
|
||||||
|
on:click={() => {
|
||||||
|
codeInterpreterEnabled = !codeInterpreterEnabled;
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div class="flex-1 truncate">
|
||||||
|
<div class="flex flex-1 gap-2 items-center">
|
||||||
|
<div class="shrink-0">
|
||||||
|
<Terminal className="size-3.5" strokeWidth="1.75" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=" truncate">{$i18n.t('Code Interpreter')}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=" shrink-0">
|
||||||
|
<Switch
|
||||||
|
state={codeInterpreterEnabled}
|
||||||
|
on:change={async (e) => {
|
||||||
|
const state = e.detail;
|
||||||
|
await tick();
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</Tooltip>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
{:else if tab === 'tools' && tools}
|
||||||
|
<div in:fly={{ x: 20, duration: 150 }}>
|
||||||
|
<button
|
||||||
|
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||||
|
on:click={() => {
|
||||||
|
tab = '';
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<ChevronLeft />
|
||||||
|
|
||||||
|
<div class="flex items-center w-full justify-between">
|
||||||
|
<div>
|
||||||
|
{$i18n.t('Tools')}
|
||||||
|
<span class="ml-0.5 text-gray-500">{Object.keys(tools).length}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
{#each Object.keys(tools) as toolId}
|
||||||
|
<button
|
||||||
|
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||||
|
on:click={() => {
|
||||||
|
tools[toolId].enabled = !tools[toolId].enabled;
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div class="flex-1 truncate">
|
||||||
|
<div class="flex flex-1 gap-2 items-center">
|
||||||
|
<Tooltip content={tools[toolId]?.name ?? ''} placement="top">
|
||||||
|
<div class="shrink-0">
|
||||||
|
<Wrench />
|
||||||
|
</div>
|
||||||
|
</Tooltip>
|
||||||
|
<Tooltip content={tools[toolId]?.description ?? ''} placement="top-start">
|
||||||
|
<div class=" truncate">{tools[toolId].name}</div>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class=" shrink-0">
|
||||||
|
<Switch
|
||||||
|
state={tools[toolId].enabled}
|
||||||
|
on:change={async (e) => {
|
||||||
|
const state = e.detail;
|
||||||
|
await tick();
|
||||||
|
if (state) {
|
||||||
|
selectedToolIds = [...selectedToolIds, toolId];
|
||||||
|
} else {
|
||||||
|
selectedToolIds = selectedToolIds.filter((id) => id !== toolId);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
{/each}
|
{/each}
|
||||||
{/if}
|
</div>
|
||||||
|
|
||||||
{#if showWebSearchButton}
|
|
||||||
<Tooltip content={$i18n.t('Search the internet')} placement="top-start">
|
|
||||||
<button
|
|
||||||
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
|
||||||
on:click={() => {
|
|
||||||
webSearchEnabled = !webSearchEnabled;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div class="flex-1 truncate">
|
|
||||||
<div class="flex flex-1 gap-2 items-center">
|
|
||||||
<div class="shrink-0">
|
|
||||||
<GlobeAlt />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class=" truncate">{$i18n.t('Web Search')}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class=" shrink-0">
|
|
||||||
<Switch
|
|
||||||
state={webSearchEnabled}
|
|
||||||
on:change={async (e) => {
|
|
||||||
const state = e.detail;
|
|
||||||
await tick();
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</Tooltip>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
{#if showImageGenerationButton}
|
|
||||||
<Tooltip content={$i18n.t('Generate an image')} placement="top-start">
|
|
||||||
<button
|
|
||||||
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
|
||||||
on:click={() => {
|
|
||||||
imageGenerationEnabled = !imageGenerationEnabled;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div class="flex-1 truncate">
|
|
||||||
<div class="flex flex-1 gap-2 items-center">
|
|
||||||
<div class="shrink-0">
|
|
||||||
<Photo className="size-4" strokeWidth="1.5" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class=" truncate">{$i18n.t('Image')}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class=" shrink-0">
|
|
||||||
<Switch
|
|
||||||
state={imageGenerationEnabled}
|
|
||||||
on:change={async (e) => {
|
|
||||||
const state = e.detail;
|
|
||||||
await tick();
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</Tooltip>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
{#if showCodeInterpreterButton}
|
|
||||||
<Tooltip content={$i18n.t('Execute code for analysis')} placement="top-start">
|
|
||||||
<button
|
|
||||||
class="flex w-full justify-between gap-2 items-center px-3 py-1.5 text-sm cursor-pointer rounded-xl hover:bg-gray-50 dark:hover:bg-gray-800"
|
|
||||||
aria-pressed={codeInterpreterEnabled}
|
|
||||||
aria-label={codeInterpreterEnabled
|
|
||||||
? $i18n.t('Disable Code Interpreter')
|
|
||||||
: $i18n.t('Enable Code Interpreter')}
|
|
||||||
on:click={() => {
|
|
||||||
codeInterpreterEnabled = !codeInterpreterEnabled;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div class="flex-1 truncate">
|
|
||||||
<div class="flex flex-1 gap-2 items-center">
|
|
||||||
<div class="shrink-0">
|
|
||||||
<Terminal className="size-3.5" strokeWidth="1.75" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class=" truncate">{$i18n.t('Code Interpreter')}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class=" shrink-0">
|
|
||||||
<Switch
|
|
||||||
state={codeInterpreterEnabled}
|
|
||||||
on:change={async (e) => {
|
|
||||||
const state = e.detail;
|
|
||||||
await tick();
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</Tooltip>
|
|
||||||
{/if}
|
|
||||||
{/if}
|
{/if}
|
||||||
</DropdownMenu.Content>
|
</DropdownMenu.Content>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue