123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737 |
- <script lang="ts">
- import { toast } from 'svelte-sonner';
- import { createEventDispatcher, onMount, getContext } from 'svelte';
- import { config as backendConfig, user } from '$lib/stores';
- import { getBackendConfig } from '$lib/apis';
- import {
- getImageGenerationModels,
- getImageGenerationConfig,
- updateImageGenerationConfig,
- getConfig,
- updateConfig,
- verifyConfigUrl
- } from '$lib/apis/images';
- import SensitiveInput from '$lib/components/common/SensitiveInput.svelte';
- import Switch from '$lib/components/common/Switch.svelte';
- import Tooltip from '$lib/components/common/Tooltip.svelte';
- const dispatch = createEventDispatcher();
- const i18n = getContext('i18n');
- let loading = false;
- let config = null;
- let imageGenerationConfig = null;
- let models = null;
- let samplers = [
- 'DPM++ 2M',
- 'DPM++ SDE',
- 'DPM++ 2M SDE',
- 'DPM++ 2M SDE Heun',
- 'DPM++ 2S a',
- 'DPM++ 3M SDE',
- 'Euler a',
- 'Euler',
- 'LMS',
- 'Heun',
- 'DPM2',
- 'DPM2 a',
- 'DPM fast',
- 'DPM adaptive',
- 'Restart',
- 'DDIM',
- 'DDIM CFG++',
- 'PLMS',
- 'UniPC'
- ];
- let schedulers = [
- 'Automatic',
- 'Uniform',
- 'Karras',
- 'Exponential',
- 'Polyexponential',
- 'SGM Uniform',
- 'KL Optimal',
- 'Align Your Steps',
- 'Simple',
- 'Normal',
- 'DDIM',
- 'Beta'
- ];
- let requiredWorkflowNodes = [
- {
- type: 'prompt',
- key: 'text',
- node_ids: ''
- },
- {
- type: 'model',
- key: 'ckpt_name',
- node_ids: ''
- },
- {
- type: 'width',
- key: 'width',
- node_ids: ''
- },
- {
- type: 'height',
- key: 'height',
- node_ids: ''
- },
- {
- type: 'steps',
- key: 'steps',
- node_ids: ''
- },
- {
- type: 'seed',
- key: 'seed',
- node_ids: ''
- }
- ];
- const getModels = async () => {
- models = await getImageGenerationModels(localStorage.token).catch((error) => {
- toast.error(`${error}`);
- return null;
- });
- };
- const updateConfigHandler = async () => {
- const res = await updateConfig(localStorage.token, config)
- .catch((error) => {
- toast.error(`${error}`);
- return null;
- })
- .catch((error) => {
- toast.error(`${error}`);
- return null;
- });
- if (res) {
- config = res;
- }
- if (config.enabled) {
- backendConfig.set(await getBackendConfig());
- getModels();
- }
- };
- const validateJSON = (json) => {
- try {
- const obj = JSON.parse(json);
- if (obj && typeof obj === 'object') {
- return true;
- }
- } catch (e) {}
- return false;
- };
- const saveHandler = async () => {
- loading = true;
- if (config?.comfyui?.COMFYUI_WORKFLOW) {
- if (!validateJSON(config.comfyui.COMFYUI_WORKFLOW)) {
- toast.error('Invalid JSON format for ComfyUI Workflow.');
- loading = false;
- return;
- }
- }
- if (config?.comfyui?.COMFYUI_WORKFLOW) {
- config.comfyui.COMFYUI_WORKFLOW_NODES = requiredWorkflowNodes.map((node) => {
- return {
- type: node.type,
- key: node.key,
- node_ids:
- node.node_ids.trim() === '' ? [] : node.node_ids.split(',').map((id) => id.trim())
- };
- });
- }
- await updateConfig(localStorage.token, config).catch((error) => {
- toast.error(`${error}`);
- loading = false;
- return null;
- });
- await updateImageGenerationConfig(localStorage.token, imageGenerationConfig).catch((error) => {
- toast.error(`${error}`);
- loading = false;
- return null;
- });
- getModels();
- dispatch('save');
- loading = false;
- };
- onMount(async () => {
- if ($user.role === 'admin') {
- const res = await getConfig(localStorage.token).catch((error) => {
- toast.error(`${error}`);
- return null;
- });
- if (res) {
- config = res;
- }
- if (config.enabled) {
- getModels();
- }
- if (config.comfyui.COMFYUI_WORKFLOW) {
- config.comfyui.COMFYUI_WORKFLOW = JSON.stringify(
- JSON.parse(config.comfyui.COMFYUI_WORKFLOW),
- null,
- 2
- );
- }
- requiredWorkflowNodes = requiredWorkflowNodes.map((node) => {
- const n = config.comfyui.COMFYUI_WORKFLOW_NODES.find((n) => n.type === node.type) ?? node;
- console.log(n);
- return {
- type: n.type,
- key: n.key,
- node_ids: typeof n.node_ids === 'string' ? n.node_ids : n.node_ids.join(',')
- };
- });
- const imageConfigRes = await getImageGenerationConfig(localStorage.token).catch((error) => {
- toast.error(`${error}`);
- return null;
- });
- if (imageConfigRes) {
- imageGenerationConfig = imageConfigRes;
- }
- }
- });
- </script>
- <form
- class="flex flex-col h-full justify-between space-y-3 text-sm"
- on:submit|preventDefault={async () => {
- saveHandler();
- }}
- >
- <div class=" space-y-3 overflow-y-scroll scrollbar-hidden pr-2">
- {#if config && imageGenerationConfig}
- <div>
- <div class=" mb-1 text-sm font-medium">{$i18n.t('Image Settings')}</div>
- <div>
- <div class=" py-1 flex w-full justify-between">
- <div class=" self-center text-xs font-medium">
- {$i18n.t('Image Generation (Experimental)')}
- </div>
- <div class="px-1">
- <Switch
- bind:state={config.enabled}
- on:change={(e) => {
- const enabled = e.detail;
- if (enabled) {
- if (
- config.engine === 'automatic1111' &&
- config.automatic1111.AUTOMATIC1111_BASE_URL === ''
- ) {
- toast.error($i18n.t('AUTOMATIC1111 Base URL is required.'));
- config.enabled = false;
- } else if (
- config.engine === 'comfyui' &&
- config.comfyui.COMFYUI_BASE_URL === ''
- ) {
- toast.error($i18n.t('ComfyUI Base URL is required.'));
- config.enabled = false;
- } else if (config.engine === 'openai' && config.openai.OPENAI_API_KEY === '') {
- toast.error($i18n.t('OpenAI API Key is required.'));
- config.enabled = false;
- } else if (config.engine === 'gemini' && config.gemini.GEMINI_API_KEY === '') {
- toast.error($i18n.t('Gemini API Key is required.'));
- config.enabled = false;
- }
- }
- updateConfigHandler();
- }}
- />
- </div>
- </div>
- </div>
- {#if config.enabled}
- <div class=" py-1 flex w-full justify-between">
- <div class=" self-center text-xs font-medium">{$i18n.t('Image Prompt Generation')}</div>
- <div class="px-1">
- <Switch bind:state={config.prompt_generation} />
- </div>
- </div>
- {/if}
- <div class=" py-1 flex w-full justify-between">
- <div class=" self-center text-xs font-medium">{$i18n.t('Image Generation Engine')}</div>
- <div class="flex items-center relative">
- <select
- class=" dark:bg-gray-900 w-fit pr-8 cursor-pointer rounded-sm px-2 p-1 text-xs bg-transparent outline-hidden text-right"
- bind:value={config.engine}
- placeholder={$i18n.t('Select Engine')}
- on:change={async () => {
- updateConfigHandler();
- }}
- >
- <option value="openai">{$i18n.t('Default (Open AI)')}</option>
- <option value="comfyui">{$i18n.t('ComfyUI')}</option>
- <option value="automatic1111">{$i18n.t('Automatic1111')}</option>
- <option value="gemini">{$i18n.t('Gemini')}</option>
- </select>
- </div>
- </div>
- </div>
- <hr class=" border-gray-100 dark:border-gray-850" />
- <div class="flex flex-col gap-2">
- {#if (config?.engine ?? 'automatic1111') === 'automatic1111'}
- <div>
- <div class=" mb-2 text-sm font-medium">{$i18n.t('AUTOMATIC1111 Base URL')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <input
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- placeholder={$i18n.t('Enter URL (e.g. http://127.0.0.1:7860/)')}
- bind:value={config.automatic1111.AUTOMATIC1111_BASE_URL}
- />
- </div>
- <button
- class="px-2.5 bg-gray-50 hover:bg-gray-100 text-gray-800 dark:bg-gray-850 dark:hover:bg-gray-800 dark:text-gray-100 rounded-lg transition"
- type="button"
- on:click={async () => {
- await updateConfigHandler();
- const res = await verifyConfigUrl(localStorage.token).catch((error) => {
- toast.error(`${error}`);
- return null;
- });
- if (res) {
- toast.success($i18n.t('Server connection verified'));
- }
- }}
- >
- <svg
- xmlns="http://www.w3.org/2000/svg"
- viewBox="0 0 20 20"
- fill="currentColor"
- class="w-4 h-4"
- >
- <path
- fill-rule="evenodd"
- d="M15.312 11.424a5.5 5.5 0 01-9.201 2.466l-.312-.311h2.433a.75.75 0 000-1.5H3.989a.75.75 0 00-.75.75v4.242a.75.75 0 001.5 0v-2.43l.31.31a7 7 0 0011.712-3.138.75.75 0 00-1.449-.39zm1.23-3.723a.75.75 0 00.219-.53V2.929a.75.75 0 00-1.5 0V5.36l-.31-.31A7 7 0 003.239 8.188a.75.75 0 101.448.389A5.5 5.5 0 0113.89 6.11l.311.31h-2.432a.75.75 0 000 1.5h4.243a.75.75 0 00.53-.219z"
- clip-rule="evenodd"
- />
- </svg>
- </button>
- </div>
- <div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
- {$i18n.t('Include `--api` flag when running stable-diffusion-webui')}
- <a
- class=" text-gray-300 font-medium"
- href="https://github.com/AUTOMATIC1111/stable-diffusion-webui/discussions/3734"
- target="_blank"
- >
- {$i18n.t('(e.g. `sh webui.sh --api`)')}
- </a>
- </div>
- </div>
- <div>
- <div class=" mb-2 text-sm font-medium">
- {$i18n.t('AUTOMATIC1111 Api Auth String')}
- </div>
- <SensitiveInput
- placeholder={$i18n.t('Enter api auth string (e.g. username:password)')}
- bind:value={config.automatic1111.AUTOMATIC1111_API_AUTH}
- required={false}
- />
- <div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
- {$i18n.t('Include `--api-auth` flag when running stable-diffusion-webui')}
- <a
- class=" text-gray-300 font-medium"
- href="https://github.com/AUTOMATIC1111/stable-diffusion-webui/discussions/13993"
- target="_blank"
- >
- {$i18n
- .t('(e.g. `sh webui.sh --api --api-auth username_password`)')
- .replace('_', ':')}
- </a>
- </div>
- </div>
- <!---Sampler-->
- <div>
- <div class=" mb-2.5 text-sm font-medium">{$i18n.t('Set Sampler')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <Tooltip content={$i18n.t('Enter Sampler (e.g. Euler a)')} placement="top-start">
- <input
- list="sampler-list"
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- placeholder={$i18n.t('Enter Sampler (e.g. Euler a)')}
- bind:value={config.automatic1111.AUTOMATIC1111_SAMPLER}
- />
- <datalist id="sampler-list">
- {#each samplers ?? [] as sampler}
- <option value={sampler}>{sampler}</option>
- {/each}
- </datalist>
- </Tooltip>
- </div>
- </div>
- </div>
- <!---Scheduler-->
- <div>
- <div class=" mb-2.5 text-sm font-medium">{$i18n.t('Set Scheduler')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <Tooltip content={$i18n.t('Enter Scheduler (e.g. Karras)')} placement="top-start">
- <input
- list="scheduler-list"
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- placeholder={$i18n.t('Enter Scheduler (e.g. Karras)')}
- bind:value={config.automatic1111.AUTOMATIC1111_SCHEDULER}
- />
- <datalist id="scheduler-list">
- {#each schedulers ?? [] as scheduler}
- <option value={scheduler}>{scheduler}</option>
- {/each}
- </datalist>
- </Tooltip>
- </div>
- </div>
- </div>
- <!---CFG scale-->
- <div>
- <div class=" mb-2.5 text-sm font-medium">{$i18n.t('Set CFG Scale')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <Tooltip content={$i18n.t('Enter CFG Scale (e.g. 7.0)')} placement="top-start">
- <input
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- placeholder={$i18n.t('Enter CFG Scale (e.g. 7.0)')}
- bind:value={config.automatic1111.AUTOMATIC1111_CFG_SCALE}
- />
- </Tooltip>
- </div>
- </div>
- </div>
- {:else if config?.engine === 'comfyui'}
- <div class="">
- <div class=" mb-2 text-sm font-medium">{$i18n.t('ComfyUI Base URL')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <input
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- placeholder={$i18n.t('Enter URL (e.g. http://127.0.0.1:7860/)')}
- bind:value={config.comfyui.COMFYUI_BASE_URL}
- />
- </div>
- <button
- class="px-2.5 bg-gray-50 hover:bg-gray-100 text-gray-800 dark:bg-gray-850 dark:hover:bg-gray-800 dark:text-gray-100 rounded-lg transition"
- type="button"
- on:click={async () => {
- await updateConfigHandler();
- const res = await verifyConfigUrl(localStorage.token).catch((error) => {
- toast.error(`${error}`);
- return null;
- });
- if (res) {
- toast.success($i18n.t('Server connection verified'));
- }
- }}
- >
- <svg
- xmlns="http://www.w3.org/2000/svg"
- viewBox="0 0 20 20"
- fill="currentColor"
- class="w-4 h-4"
- >
- <path
- fill-rule="evenodd"
- d="M15.312 11.424a5.5 5.5 0 01-9.201 2.466l-.312-.311h2.433a.75.75 0 000-1.5H3.989a.75.75 0 00-.75.75v4.242a.75.75 0 001.5 0v-2.43l.31.31a7 7 0 0011.712-3.138.75.75 0 00-1.449-.39zm1.23-3.723a.75.75 0 00.219-.53V2.929a.75.75 0 00-1.5 0V5.36l-.31-.31A7 7 0 003.239 8.188a.75.75 0 101.448.389A5.5 5.5 0 0113.89 6.11l.311.31h-2.432a.75.75 0 000 1.5h4.243a.75.75 0 00.53-.219z"
- clip-rule="evenodd"
- />
- </svg>
- </button>
- </div>
- </div>
- <div class="">
- <div class=" mb-2 text-sm font-medium">{$i18n.t('ComfyUI API Key')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <SensitiveInput
- placeholder={$i18n.t('sk-1234')}
- bind:value={config.comfyui.COMFYUI_API_KEY}
- required={false}
- />
- </div>
- </div>
- </div>
- <div class="">
- <div class=" mb-2 text-sm font-medium">{$i18n.t('ComfyUI Workflow')}</div>
- {#if config.comfyui.COMFYUI_WORKFLOW}
- <textarea
- class="w-full rounded-lg mb-1 py-2 px-4 text-xs bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden disabled:text-gray-600 resize-none"
- rows="10"
- bind:value={config.comfyui.COMFYUI_WORKFLOW}
- required
- />
- {/if}
- <div class="flex w-full">
- <div class="flex-1">
- <input
- id="upload-comfyui-workflow-input"
- hidden
- type="file"
- accept=".json"
- on:change={(e) => {
- const file = e.target.files[0];
- const reader = new FileReader();
- reader.onload = (e) => {
- config.comfyui.COMFYUI_WORKFLOW = e.target.result;
- e.target.value = null;
- };
- reader.readAsText(file);
- }}
- />
- <button
- class="w-full text-sm font-medium py-2 bg-transparent hover:bg-gray-100 border border-dashed dark:border-gray-850 dark:hover:bg-gray-850 text-center rounded-xl"
- type="button"
- on:click={() => {
- document.getElementById('upload-comfyui-workflow-input')?.click();
- }}
- >
- {$i18n.t('Click here to upload a workflow.json file.')}
- </button>
- </div>
- </div>
- <div class="mt-2 text-xs text-gray-400 dark:text-gray-500">
- {$i18n.t('Make sure to export a workflow.json file as API format from ComfyUI.')}
- </div>
- </div>
- {#if config.comfyui.COMFYUI_WORKFLOW}
- <div class="">
- <div class=" mb-2 text-sm font-medium">{$i18n.t('ComfyUI Workflow Nodes')}</div>
- <div class="text-xs flex flex-col gap-1.5">
- {#each requiredWorkflowNodes as node}
- <div class="flex w-full items-center border dark:border-gray-850 rounded-lg">
- <div class="shrink-0">
- <div
- class=" capitalize line-clamp-1 font-medium px-3 py-1 w-20 text-center rounded-l-lg bg-green-500/10 text-green-700 dark:text-green-200"
- >
- {node.type}{node.type === 'prompt' ? '*' : ''}
- </div>
- </div>
- <div class="">
- <Tooltip content="Input Key (e.g. text, unet_name, steps)">
- <input
- class="py-1 px-3 w-24 text-xs text-center bg-transparent outline-hidden border-r dark:border-gray-850"
- placeholder="Key"
- bind:value={node.key}
- required
- />
- </Tooltip>
- </div>
- <div class="w-full">
- <Tooltip
- content="Comma separated Node Ids (e.g. 1 or 1,2)"
- placement="top-start"
- >
- <input
- class="w-full py-1 px-4 rounded-r-lg text-xs bg-transparent outline-hidden"
- placeholder="Node Ids"
- bind:value={node.node_ids}
- />
- </Tooltip>
- </div>
- </div>
- {/each}
- </div>
- <div class="mt-2 text-xs text-right text-gray-400 dark:text-gray-500">
- {$i18n.t('*Prompt node ID(s) are required for image generation')}
- </div>
- </div>
- {/if}
- {:else if config?.engine === 'openai'}
- <div>
- <div class=" mb-1.5 text-sm font-medium">{$i18n.t('OpenAI API Config')}</div>
- <div class="flex gap-2 mb-1">
- <input
- class="flex-1 w-full text-sm bg-transparent outline-hidden"
- placeholder={$i18n.t('API Base URL')}
- bind:value={config.openai.OPENAI_API_BASE_URL}
- required
- />
- <SensitiveInput
- placeholder={$i18n.t('API Key')}
- bind:value={config.openai.OPENAI_API_KEY}
- />
- </div>
- </div>
- {:else if config?.engine === 'gemini'}
- <div>
- <div class=" mb-1.5 text-sm font-medium">{$i18n.t('Gemini API Config')}</div>
- <div class="flex gap-2 mb-1">
- <input
- class="flex-1 w-full text-sm bg-transparent outline-none"
- placeholder={$i18n.t('API Base URL')}
- bind:value={config.gemini.GEMINI_API_BASE_URL}
- required
- />
- <SensitiveInput
- placeholder={$i18n.t('API Key')}
- bind:value={config.gemini.GEMINI_API_KEY}
- />
- </div>
- </div>
- {/if}
- </div>
- {#if config?.enabled}
- <hr class=" border-gray-100 dark:border-gray-850" />
- <div>
- <div class=" mb-2.5 text-sm font-medium">{$i18n.t('Set Default Model')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <div class="flex w-full">
- <div class="flex-1">
- <Tooltip content={$i18n.t('Enter Model ID')} placement="top-start">
- <input
- list="model-list"
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- bind:value={imageGenerationConfig.MODEL}
- placeholder="Select a model"
- required
- />
- <datalist id="model-list">
- {#each models ?? [] as model}
- <option value={model.id}>{model.name}</option>
- {/each}
- </datalist>
- </Tooltip>
- </div>
- </div>
- </div>
- </div>
- </div>
- <div>
- <div class=" mb-2.5 text-sm font-medium">{$i18n.t('Set Image Size')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <Tooltip content={$i18n.t('Enter Image Size (e.g. 512x512)')} placement="top-start">
- <input
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- placeholder={$i18n.t('Enter Image Size (e.g. 512x512)')}
- bind:value={imageGenerationConfig.IMAGE_SIZE}
- required
- />
- </Tooltip>
- </div>
- </div>
- </div>
- <div>
- <div class=" mb-2.5 text-sm font-medium">{$i18n.t('Set Steps')}</div>
- <div class="flex w-full">
- <div class="flex-1 mr-2">
- <Tooltip content={$i18n.t('Enter Number of Steps (e.g. 50)')} placement="top-start">
- <input
- class="w-full rounded-lg py-2 px-4 text-sm bg-gray-50 dark:text-gray-300 dark:bg-gray-850 outline-hidden"
- placeholder={$i18n.t('Enter Number of Steps (e.g. 50)')}
- bind:value={imageGenerationConfig.IMAGE_STEPS}
- required
- />
- </Tooltip>
- </div>
- </div>
- </div>
- {/if}
- {/if}
- </div>
- <div class="flex justify-end pt-3 text-sm font-medium">
- <button
- class="px-3.5 py-1.5 text-sm font-medium bg-black hover:bg-gray-900 text-white dark:bg-white dark:text-black dark:hover:bg-gray-100 transition rounded-full flex flex-row space-x-1 items-center {loading
- ? ' cursor-not-allowed'
- : ''}"
- type="submit"
- disabled={loading}
- >
- {$i18n.t('Save')}
- {#if loading}
- <div class="ml-2 self-center">
- <svg
- class=" w-4 h-4"
- viewBox="0 0 24 24"
- fill="currentColor"
- xmlns="http://www.w3.org/2000/svg"
- ><style>
- .spinner_ajPY {
- transform-origin: center;
- animation: spinner_AtaB 0.75s infinite linear;
- }
- @keyframes spinner_AtaB {
- 100% {
- transform: rotate(360deg);
- }
- }
- </style><path
- d="M12,1A11,11,0,1,0,23,12,11,11,0,0,0,12,1Zm0,19a8,8,0,1,1,8-8A8,8,0,0,1,12,20Z"
- opacity=".25"
- /><path
- d="M10.14,1.16a11,11,0,0,0-9,8.92A1.59,1.59,0,0,0,2.46,12,1.52,1.52,0,0,0,4.11,10.7a8,8,0,0,1,6.66-6.61A1.42,1.42,0,0,0,12,2.69h0A1.57,1.57,0,0,0,10.14,1.16Z"
- class="spinner_ajPY"
- /></svg
- >
- </div>
- {/if}
- </button>
- </div>
- </form>
|