Spaces:
Build error
Build error
| <script lang="ts"> | |
| import { marked } from 'marked'; | |
| import fileSaver from 'file-saver'; | |
| const { saveAs } = fileSaver; | |
| import { onMount, getContext, tick } from 'svelte'; | |
| const i18n = getContext('i18n'); | |
| import { WEBUI_NAME, config, mobile, models as _models, settings, user } from '$lib/stores'; | |
| import { | |
| createNewModel, | |
| deleteAllModels, | |
| getBaseModels, | |
| toggleModelById, | |
| updateModelById, | |
| importModels | |
| } from '$lib/apis/models'; | |
| import { copyToClipboard } from '$lib/utils'; | |
| import { page } from '$app/stores'; | |
| import { updateUserSettings } from '$lib/apis/users'; | |
| import { getModels } from '$lib/apis'; | |
| import Search from '$lib/components/icons/Search.svelte'; | |
| import Tooltip from '$lib/components/common/Tooltip.svelte'; | |
| import Switch from '$lib/components/common/Switch.svelte'; | |
| import Spinner from '$lib/components/common/Spinner.svelte'; | |
| import XMark from '$lib/components/icons/XMark.svelte'; | |
| import ModelEditor from '$lib/components/workspace/Models/ModelEditor.svelte'; | |
| import { toast } from 'svelte-sonner'; | |
| import ConfirmDialog from '$lib/components/common/ConfirmDialog.svelte'; | |
| import Cog6 from '$lib/components/icons/Cog6.svelte'; | |
| import ConfigureModelsModal from './Models/ConfigureModelsModal.svelte'; | |
| import Wrench from '$lib/components/icons/Wrench.svelte'; | |
| import Download from '$lib/components/icons/Download.svelte'; | |
| import ManageModelsModal from './Models/ManageModelsModal.svelte'; | |
| import ModelMenu from '$lib/components/admin/Settings/Models/ModelMenu.svelte'; | |
| import EllipsisHorizontal from '$lib/components/icons/EllipsisHorizontal.svelte'; | |
| import EyeSlash from '$lib/components/icons/EyeSlash.svelte'; | |
| import Eye from '$lib/components/icons/Eye.svelte'; | |
| import { WEBUI_API_BASE_URL, WEBUI_BASE_URL } from '$lib/constants'; | |
| import { goto } from '$app/navigation'; | |
| import { DropdownMenu } from 'bits-ui'; | |
| import { flyAndScale } from '$lib/utils/transitions'; | |
| import Dropdown from '$lib/components/common/Dropdown.svelte'; | |
| import AdminViewSelector from './Models/AdminViewSelector.svelte'; | |
| import Pagination from '$lib/components/common/Pagination.svelte'; | |
| let shiftKey = false; | |
| let modelsImportInProgress = false; | |
| let importFiles; | |
| let modelsImportInputElement: HTMLInputElement; | |
| let models = null; | |
| let workspaceModels = null; | |
| let baseModels = null; | |
| let filteredModels = []; | |
| let selectedModelId = null; | |
| let showConfigModal = false; | |
| let showManageModal = false; | |
| let viewOption = ''; // '' = All, 'enabled', 'disabled', 'visible', 'hidden' | |
| const perPage = 30; | |
| let currentPage = 1; | |
| $: if (models) { | |
| filteredModels = models | |
| .filter((m) => searchValue === '' || m.name.toLowerCase().includes(searchValue.toLowerCase())) | |
| .filter((m) => { | |
| if (viewOption === 'enabled') return m?.is_active ?? true; | |
| if (viewOption === 'disabled') return !(m?.is_active ?? true); | |
| if (viewOption === 'visible') return !(m?.meta?.hidden ?? false); | |
| if (viewOption === 'hidden') return m?.meta?.hidden === true; | |
| return true; // All | |
| }) | |
| .sort((a, b) => { | |
| return (a?.name ?? a?.id ?? '').localeCompare(b?.name ?? b?.id ?? ''); | |
| }); | |
| } | |
| let searchValue = ''; | |
| $: if (searchValue || viewOption !== undefined) { | |
| currentPage = 1; | |
| } | |
| const enableAllHandler = async () => { | |
| const modelsToEnable = filteredModels.filter((m) => !(m.is_active ?? true)); | |
| // Optimistic UI update | |
| modelsToEnable.forEach((m) => (m.is_active = true)); | |
| models = models; | |
| // Sync with server | |
| await Promise.all(modelsToEnable.map((model) => toggleModelById(localStorage.token, model.id))); | |
| }; | |
| const disableAllHandler = async () => { | |
| const modelsToDisable = filteredModels.filter((m) => m.is_active ?? true); | |
| // Optimistic UI update | |
| modelsToDisable.forEach((m) => (m.is_active = false)); | |
| models = models; | |
| // Sync with server | |
| await Promise.all( | |
| modelsToDisable.map((model) => toggleModelById(localStorage.token, model.id)) | |
| ); | |
| }; | |
| const downloadModels = async (models) => { | |
| let blob = new Blob([JSON.stringify(models)], { | |
| type: 'application/json' | |
| }); | |
| saveAs(blob, `models-export-${Date.now()}.json`); | |
| }; | |
| const init = async () => { | |
| models = null; | |
| workspaceModels = await getBaseModels(localStorage.token); | |
| baseModels = await getModels(localStorage.token, null, true); | |
| models = baseModels.map((m) => { | |
| const workspaceModel = workspaceModels.find((wm) => wm.id === m.id); | |
| if (workspaceModel) { | |
| return { | |
| ...m, | |
| ...workspaceModel | |
| }; | |
| } else { | |
| return { | |
| ...m, | |
| id: m.id, | |
| name: m.name, | |
| is_active: true | |
| }; | |
| } | |
| }); | |
| }; | |
| const upsertModelHandler = async (model) => { | |
| model.base_model_id = null; | |
| if (workspaceModels.find((m) => m.id === model.id)) { | |
| const res = await updateModelById(localStorage.token, model.id, model).catch((error) => { | |
| return null; | |
| }); | |
| if (res) { | |
| toast.success($i18n.t('Model updated successfully')); | |
| } | |
| } else { | |
| const res = await createNewModel(localStorage.token, { | |
| meta: {}, | |
| id: model.id, | |
| name: model.name, | |
| base_model_id: null, | |
| params: {}, | |
| access_grants: [], | |
| ...model | |
| }).catch((error) => { | |
| return null; | |
| }); | |
| if (res) { | |
| toast.success($i18n.t('Model updated successfully')); | |
| } | |
| } | |
| await init(); | |
| _models.set( | |
| await getModels( | |
| localStorage.token, | |
| $config?.features?.enable_direct_connections && ($settings?.directConnections ?? null) | |
| ) | |
| ); | |
| }; | |
| const toggleModelHandler = async (model) => { | |
| if (!Object.keys(model).includes('base_model_id')) { | |
| await createNewModel(localStorage.token, { | |
| id: model.id, | |
| name: model.name, | |
| base_model_id: null, | |
| meta: {}, | |
| params: {}, | |
| access_grants: [], | |
| is_active: model.is_active | |
| }).catch((error) => { | |
| return null; | |
| }); | |
| } else { | |
| await toggleModelById(localStorage.token, model.id); | |
| } | |
| // await init(); | |
| _models.set( | |
| await getModels( | |
| localStorage.token, | |
| $config?.features?.enable_direct_connections && ($settings?.directConnections ?? null) | |
| ) | |
| ); | |
| }; | |
| const hideModelHandler = async (model) => { | |
| model.meta = { | |
| ...model.meta, | |
| hidden: !(model?.meta?.hidden ?? false) | |
| }; | |
| console.debug(model); | |
| toast.success( | |
| model.meta.hidden | |
| ? $i18n.t(`Model {{name}} is now hidden`, { | |
| name: model.id | |
| }) | |
| : $i18n.t(`Model {{name}} is now visible`, { | |
| name: model.id | |
| }) | |
| ); | |
| upsertModelHandler(model); | |
| }; | |
| const copyLinkHandler = async (model) => { | |
| const baseUrl = window.location.origin; | |
| const res = await copyToClipboard(`${baseUrl}/?model=${encodeURIComponent(model.id)}`); | |
| if (res) { | |
| toast.success($i18n.t('Copied link to clipboard')); | |
| } else { | |
| toast.error($i18n.t('Failed to copy link')); | |
| } | |
| }; | |
| const cloneHandler = async (model) => { | |
| sessionStorage.model = JSON.stringify({ | |
| ...model, | |
| base_model_id: model.id, | |
| id: `${model.id}-clone`, | |
| name: `${model.name} (Clone)` | |
| }); | |
| goto('/workspace/models/create'); | |
| }; | |
| const exportModelHandler = async (model) => { | |
| let blob = new Blob([JSON.stringify([model])], { | |
| type: 'application/json' | |
| }); | |
| saveAs(blob, `${model.id}-${Date.now()}.json`); | |
| }; | |
| const pinModelHandler = async (modelId) => { | |
| let pinnedModels = $settings?.pinnedModels ?? []; | |
| if (pinnedModels.includes(modelId)) { | |
| pinnedModels = pinnedModels.filter((id) => id !== modelId); | |
| } else { | |
| pinnedModels = [...new Set([...pinnedModels, modelId])]; | |
| } | |
| settings.set({ ...$settings, pinnedModels: pinnedModels }); | |
| await updateUserSettings(localStorage.token, { ui: $settings }); | |
| }; | |
| onMount(async () => { | |
| await init(); | |
| const id = $page.url.searchParams.get('id'); | |
| if (id) { | |
| selectedModelId = id; | |
| } | |
| const onKeyDown = (event) => { | |
| if (event.key === 'Shift') { | |
| shiftKey = true; | |
| } | |
| }; | |
| const onKeyUp = (event) => { | |
| if (event.key === 'Shift') { | |
| shiftKey = false; | |
| } | |
| }; | |
| const onBlur = () => { | |
| shiftKey = false; | |
| }; | |
| window.addEventListener('keydown', onKeyDown); | |
| window.addEventListener('keyup', onKeyUp); | |
| window.addEventListener('blur-sm', onBlur); | |
| return () => { | |
| window.removeEventListener('keydown', onKeyDown); | |
| window.removeEventListener('keyup', onKeyUp); | |
| window.removeEventListener('blur-sm', onBlur); | |
| }; | |
| }); | |
| </script> | |
| <ConfigureModelsModal bind:show={showConfigModal} initHandler={init} /> | |
| <ManageModelsModal bind:show={showManageModal} /> | |
| {#if models !== null} | |
| {#if selectedModelId === null} | |
| <div class="flex flex-col gap-1 mt-1.5 mb-2"> | |
| <div class="flex justify-between items-center"> | |
| <div class="flex items-center md:self-center text-xl font-medium px-0.5 gap-2 shrink-0"> | |
| <div> | |
| {$i18n.t('Models')} | |
| </div> | |
| <div class="text-lg font-medium text-gray-500 dark:text-gray-500"> | |
| {filteredModels.length} | |
| </div> | |
| </div> | |
| <div class="flex w-full justify-end gap-1.5"> | |
| <button | |
| class="flex text-xs items-center space-x-1 px-3 py-1.5 rounded-xl bg-gray-50 hover:bg-gray-100 dark:bg-gray-850 dark:hover:bg-gray-800 dark:text-gray-200 transition" | |
| type="button" | |
| on:click={() => { | |
| showManageModal = true; | |
| }} | |
| > | |
| <div class=" self-center font-medium line-clamp-1"> | |
| {$i18n.t('Manage')} | |
| </div> | |
| </button> | |
| <button | |
| class="flex text-xs items-center space-x-1 px-3 py-1.5 rounded-xl bg-gray-50 hover:bg-gray-100 dark:bg-gray-850 dark:hover:bg-gray-800 dark:text-gray-200 transition" | |
| type="button" | |
| on:click={() => { | |
| showConfigModal = true; | |
| }} | |
| > | |
| <div class=" self-center font-medium line-clamp-1"> | |
| {$i18n.t('Settings')} | |
| </div> | |
| </button> | |
| </div> | |
| </div> | |
| </div> | |
| <div | |
| class="py-2 bg-white dark:bg-gray-900 rounded-3xl border border-gray-100/30 dark:border-gray-850/30" | |
| > | |
| <div class="px-3.5 flex flex-1 items-center w-full space-x-2 py-0.5 pb-2"> | |
| <div class="flex flex-1 items-center"> | |
| <div class=" self-center ml-1 mr-3"> | |
| <Search className="size-3.5" /> | |
| </div> | |
| <input | |
| class=" w-full text-sm py-1 rounded-r-xl outline-hidden bg-transparent" | |
| bind:value={searchValue} | |
| placeholder={$i18n.t('Search Models')} | |
| /> | |
| {#if searchValue} | |
| <div class="self-center pl-1.5 translate-y-[0.5px] rounded-l-xl bg-transparent"> | |
| <button | |
| class="p-0.5 rounded-full hover:bg-gray-100 dark:hover:bg-gray-900 transition" | |
| on:click={() => { | |
| searchValue = ''; | |
| }} | |
| > | |
| <XMark className="size-3" strokeWidth="2" /> | |
| </button> | |
| </div> | |
| {/if} | |
| </div> | |
| </div> | |
| <div class="px-3 flex w-full items-center bg-transparent overflow-x-auto scrollbar-none"> | |
| <div | |
| class="flex gap-0.5 w-fit text-center text-sm rounded-full bg-transparent whitespace-nowrap" | |
| > | |
| <AdminViewSelector bind:value={viewOption} /> | |
| </div> | |
| <div class="flex-1"></div> | |
| <Dropdown> | |
| <Tooltip content={$i18n.t('Actions')}> | |
| <button | |
| class="p-1 rounded-lg hover:bg-gray-100 dark:hover:bg-gray-800 transition" | |
| type="button" | |
| > | |
| <EllipsisHorizontal className="size-4" /> | |
| </button> | |
| </Tooltip> | |
| <div slot="content"> | |
| <DropdownMenu.Content | |
| class="w-full max-w-[170px] rounded-xl p-1 border border-gray-100 dark:border-gray-800 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-sm" | |
| sideOffset={-2} | |
| side="bottom" | |
| align="end" | |
| transition={flyAndScale} | |
| > | |
| <DropdownMenu.Item | |
| class="flex gap-2 items-center px-3 py-1.5 text-sm font-medium cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-md" | |
| on:click={() => { | |
| enableAllHandler(); | |
| }} | |
| > | |
| <Eye className="size-4" /> | |
| <div class="flex items-center">{$i18n.t('Enable All')}</div> | |
| </DropdownMenu.Item> | |
| <DropdownMenu.Item | |
| class="flex gap-2 items-center px-3 py-1.5 text-sm font-medium cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-md" | |
| on:click={() => { | |
| disableAllHandler(); | |
| }} | |
| > | |
| <EyeSlash className="size-4" /> | |
| <div class="flex items-center">{$i18n.t('Disable All')}</div> | |
| </DropdownMenu.Item> | |
| </DropdownMenu.Content> | |
| </div> | |
| </Dropdown> | |
| </div> | |
| <div class="px-3 my-2" id="model-list"> | |
| {#if filteredModels.length > 0} | |
| {#each filteredModels.slice((currentPage - 1) * perPage, currentPage * perPage) as model, modelIdx (`${model.id}-${modelIdx}`)} | |
| <div | |
| class=" flex space-x-4 cursor-pointer w-full px-3 py-2 dark:hover:bg-white/5 hover:bg-black/5 rounded-xl transition {model | |
| ?.meta?.hidden | |
| ? 'opacity-50 dark:opacity-50' | |
| : ''}" | |
| id="model-item-{model.id}" | |
| > | |
| <button | |
| class=" flex flex-1 text-left space-x-3.5 cursor-pointer w-full" | |
| type="button" | |
| on:click={() => { | |
| selectedModelId = model.id; | |
| }} | |
| > | |
| <div class=" self-center w-8"> | |
| <div | |
| class=" rounded-full object-cover {(model?.is_active ?? true) | |
| ? '' | |
| : 'opacity-50 dark:opacity-50'} " | |
| > | |
| <img | |
| src={`${WEBUI_API_BASE_URL}/models/model/profile/image?id=${model.id}`} | |
| alt="modelfile profile" | |
| class=" rounded-full w-full h-auto object-cover" | |
| /> | |
| </div> | |
| </div> | |
| <div | |
| class=" flex-1 self-center {(model?.is_active ?? true) ? '' : 'text-gray-500'}" | |
| > | |
| <Tooltip | |
| content={marked.parse( | |
| !!model?.meta?.description | |
| ? model?.meta?.description | |
| : model?.ollama?.digest | |
| ? `${model?.ollama?.digest} **(${model?.ollama?.modified_at})**` | |
| : model.id | |
| )} | |
| className=" w-fit" | |
| placement="top-start" | |
| > | |
| <div class=" font-semibold line-clamp-1">{model.name}</div> | |
| </Tooltip> | |
| <div class=" text-xs overflow-hidden text-ellipsis line-clamp-1 text-gray-500"> | |
| <span class=" line-clamp-1"> | |
| {!!model?.meta?.description | |
| ? model?.meta?.description | |
| : model?.ollama?.digest | |
| ? `${model.id} (${model?.ollama?.digest})` | |
| : model.id} | |
| </span> | |
| </div> | |
| </div> | |
| </button> | |
| <div class="flex flex-row gap-0.5 items-center self-center"> | |
| {#if shiftKey} | |
| <Tooltip content={model?.meta?.hidden ? $i18n.t('Show') : $i18n.t('Hide')}> | |
| <button | |
| class="self-center w-fit text-sm px-2 py-2 dark:text-gray-300 dark:hover:text-white hover:bg-black/5 dark:hover:bg-white/5 rounded-xl" | |
| type="button" | |
| on:click={() => { | |
| hideModelHandler(model); | |
| }} | |
| > | |
| {#if model?.meta?.hidden} | |
| <EyeSlash /> | |
| {:else} | |
| <Eye /> | |
| {/if} | |
| </button> | |
| </Tooltip> | |
| {:else} | |
| <button | |
| class="self-center w-fit text-sm px-2 py-2 dark:text-gray-300 dark:hover:text-white hover:bg-black/5 dark:hover:bg-white/5 rounded-xl" | |
| type="button" | |
| on:click={() => { | |
| selectedModelId = model.id; | |
| }} | |
| > | |
| <svg | |
| xmlns="http://www.w3.org/2000/svg" | |
| fill="none" | |
| viewBox="0 0 24 24" | |
| stroke-width="1.5" | |
| stroke="currentColor" | |
| class="w-4 h-4" | |
| > | |
| <path | |
| stroke-linecap="round" | |
| stroke-linejoin="round" | |
| d="m16.862 4.487 1.687-1.688a1.875 1.875 0 1 1 2.652 2.652L6.832 19.82a4.5 4.5 0 0 1-1.897 1.13l-2.685.8.8-2.685a4.5 4.5 0 0 1 1.13-1.897L16.863 4.487Zm0 0L19.5 7.125" | |
| /> | |
| </svg> | |
| </button> | |
| <ModelMenu | |
| user={$user} | |
| {model} | |
| exportHandler={() => { | |
| exportModelHandler(model); | |
| }} | |
| hideHandler={() => { | |
| hideModelHandler(model); | |
| }} | |
| pinModelHandler={() => { | |
| pinModelHandler(model.id); | |
| }} | |
| copyLinkHandler={() => { | |
| copyLinkHandler(model); | |
| }} | |
| cloneHandler={() => { | |
| cloneHandler(model); | |
| }} | |
| onClose={() => {}} | |
| > | |
| <button | |
| class="self-center w-fit text-sm p-1.5 dark:text-gray-300 dark:hover:text-white hover:bg-black/5 dark:hover:bg-white/5 rounded-xl" | |
| type="button" | |
| > | |
| <EllipsisHorizontal className="size-5" /> | |
| </button> | |
| </ModelMenu> | |
| <div class="ml-1"> | |
| <Tooltip | |
| content={(model?.is_active ?? true) | |
| ? $i18n.t('Enabled') | |
| : $i18n.t('Disabled')} | |
| > | |
| <Switch | |
| bind:state={model.is_active} | |
| on:change={async () => { | |
| toggleModelHandler(model); | |
| }} | |
| /> | |
| </Tooltip> | |
| </div> | |
| {/if} | |
| </div> | |
| </div> | |
| {/each} | |
| {:else} | |
| <div class=" w-full h-full flex flex-col justify-center items-center my-16 mb-24"> | |
| <div class="max-w-md text-center"> | |
| <div class=" text-3xl mb-3">😕</div> | |
| <div class=" text-lg font-medium mb-1">{$i18n.t('No models found')}</div> | |
| <div class=" text-gray-500 text-center text-xs"> | |
| {$i18n.t('Try adjusting your search or filter to find what you are looking for.')} | |
| </div> | |
| </div> | |
| </div> | |
| {/if} | |
| </div> | |
| {#if filteredModels.length > perPage} | |
| <Pagination bind:page={currentPage} count={filteredModels.length} {perPage} /> | |
| {/if} | |
| </div> | |
| {#if $user?.role === 'admin'} | |
| <div class=" flex justify-end w-full mb-3"> | |
| <div class="flex space-x-1"> | |
| <input | |
| id="models-import-input" | |
| bind:this={modelsImportInputElement} | |
| bind:files={importFiles} | |
| type="file" | |
| accept=".json" | |
| hidden | |
| on:change={() => { | |
| if (importFiles.length > 0) { | |
| const reader = new FileReader(); | |
| reader.onload = async (event) => { | |
| modelsImportInProgress = true; | |
| try { | |
| const models = JSON.parse(String(event.target.result)); | |
| const res = await importModels(localStorage.token, models); | |
| if (res) { | |
| toast.success($i18n.t('Models imported successfully')); | |
| await init(); | |
| } else { | |
| toast.error($i18n.t('Failed to import models')); | |
| } | |
| } catch (e) { | |
| toast.error(e?.detail ?? $i18n.t('Invalid JSON file')); | |
| console.error(e); | |
| } | |
| modelsImportInProgress = false; | |
| }; | |
| reader.readAsText(importFiles[0]); | |
| } | |
| }} | |
| /> | |
| <button | |
| class="flex text-xs items-center space-x-1 px-3 py-1.5 rounded-xl bg-gray-50 hover:bg-gray-100 dark:bg-gray-800 dark:hover:bg-gray-700 dark:text-gray-200 transition" | |
| disabled={modelsImportInProgress} | |
| on:click={() => { | |
| modelsImportInputElement.click(); | |
| }} | |
| > | |
| {#if modelsImportInProgress} | |
| <Spinner className="size-3" /> | |
| {/if} | |
| <div class=" self-center mr-2 font-medium line-clamp-1"> | |
| {$i18n.t('Import Presets')} | |
| </div> | |
| <div class=" self-center"> | |
| <svg | |
| xmlns="http://www.w3.org/2000/svg" | |
| viewBox="0 0 16 16" | |
| fill="currentColor" | |
| class="w-3.5 h-3.5" | |
| > | |
| <path | |
| fill-rule="evenodd" | |
| d="M4 2a1.5 1.5 0 0 0-1.5 1.5v9A1.5 1.5 0 0 0 4 14h8a1.5 1.5 0 0 0 1.5-1.5V6.621a1.5 1.5 0 0 0-.44-1.06L9.94 2.439A1.5 1.5 0 0 0 8.878 2H4Zm4 9.5a.75.75 0 0 1-.75-.75V8.06l-.72.72a.75.75 0 0 1-1.06-1.06l2-2a.75.75 0 0 1 1.06 0l2 2a.75.75 0 1 1-1.06 1.06l-.72-.72v2.69a.75.75 0 0 1-.75.75Z" | |
| clip-rule="evenodd" | |
| /> | |
| </svg> | |
| </div> | |
| </button> | |
| <button | |
| class="flex text-xs items-center space-x-1 px-3 py-1.5 rounded-xl bg-gray-50 hover:bg-gray-100 dark:bg-gray-800 dark:hover:bg-gray-700 dark:text-gray-200 transition" | |
| on:click={async () => { | |
| downloadModels(models); | |
| }} | |
| > | |
| <div class=" self-center mr-2 font-medium line-clamp-1"> | |
| {$i18n.t('Export Presets')} ({models.length}) | |
| </div> | |
| <div class=" self-center"> | |
| <svg | |
| xmlns="http://www.w3.org/2000/svg" | |
| viewBox="0 0 16 16" | |
| fill="currentColor" | |
| class="w-3.5 h-3.5" | |
| > | |
| <path | |
| fill-rule="evenodd" | |
| d="M4 2a1.5 1.5 0 0 0-1.5 1.5v9A1.5 1.5 0 0 0 4 14h8a1.5 1.5 0 0 0 1.5-1.5V6.621a1.5 1.5 0 0 0-.44-1.06L9.94 2.439A1.5 1.5 0 0 0 8.878 2H4Zm4 3.5a.75.75 0 0 1 .75.75v2.69l.72-.72a.75.75 0 1 1 1.06 1.06l-2 2a.75.75 0 0 1-1.06 0l-2-2a.75.75 0 0 1 1.06-1.06l.72.72V6.25A.75.75 0 0 1 8 5.5Z" | |
| clip-rule="evenodd" | |
| /> | |
| </svg> | |
| </div> | |
| </button> | |
| </div> | |
| </div> | |
| {/if} | |
| {:else} | |
| <ModelEditor | |
| edit | |
| model={models.find((m) => m.id === selectedModelId)} | |
| preset={false} | |
| onSubmit={(model) => { | |
| console.log(model); | |
| upsertModelHandler(model); | |
| selectedModelId = null; | |
| }} | |
| onBack={() => { | |
| selectedModelId = null; | |
| }} | |
| /> | |
| {/if} | |
| {:else} | |
| <div class=" h-full w-full flex justify-center items-center"> | |
| <Spinner className="size-5" /> | |
| </div> | |
| {/if} | |