<script lang="ts"> import { createEventDispatcher } from "svelte"; import Modal from "$lib/components/Modal.svelte"; import CarbonClose from "~icons/carbon/close"; import CarbonCheckmark from "~icons/carbon/checkmark-filled"; import ModelCardMetadata from "./ModelCardMetadata.svelte"; import type { Model } from "$lib/types/Model"; import type { LayoutData } from "../../routes/$types"; import { enhance } from "$app/forms"; import { base } from "$app/paths"; import CarbonEdit from "~icons/carbon/edit"; import CarbonSave from "~icons/carbon/save"; import CarbonRestart from "~icons/carbon/restart"; import { curr_model_writable } from "../../routes/LayoutWritable"; export let settings: LayoutData["settings"]; export let models: Array<Model>; let selectedModelId = ""; let selectedNum = 0; curr_model_writable.subscribe((val) => { selectedModelId = models[val].name; selectedNum = val; }); const dispatch = createEventDispatcher<{ close: void; closeAndSave }>(); let expanded = false; function onToggle() { if (expanded) { settings.customPrompts[selectedModelId] = value; } expanded = !expanded; } function onApply() { curr_model_writable.set(selectedNum); dispatch("close"); } let value = ""; function onModelChange() { value = settings.customPrompts[selectedModelId] ?? models.filter((el) => el.id === selectedModelId)[0].preprompt ?? ""; selectedNum = models.findIndex((el) => el.id == selectedModelId); } $: selectedModelId, onModelChange(); </script> <Modal width="max-w-lg" on:close> <form on:submit={() => { if (expanded) { onToggle(); } }} class="flex w-full flex-col gap-5 p-6" > {#each Object.entries(settings).filter(([k]) => !(k == "activeModel" || k === "customPrompts")) as [key, val]} <input type="hidden" name={key} value={val} /> {/each} <input type="hidden" name="customPrompts" value={JSON.stringify(settings.customPrompts)} /> <div class="flex items-start justify-between text-xl font-semibold text-gray-800"> <h2>Models</h2> <button type="button" class="group" on:click={() => dispatch("close")}> <CarbonClose class="text-gray-900 group-hover:text-gray-500" /> </button> </div> <div class="space-y-4"> {#each models as model} {@const active = model.id === selectedModelId} <div class="rounded-xl border border-gray-100 {active ? 'bg-gradient-to-r from-primary-200/40 via-primary-500/10' : ''}" > <label class="group flex cursor-pointer p-3" on:change aria-label={model.displayName}> <input type="radio" class="sr-only" name="activeModel" value={model.id} bind:group={selectedModelId} /> <span> <span class="text-md block font-semibold leading-tight text-gray-800" >{model.displayName}</span > {#if model.description} <span class="text-xs text-[#9FA8B5]">{model.description}</span> {/if} </span> <CarbonCheckmark class="-mr-1 -mt-1 ml-auto shrink-0 text-xl {active ? 'text-primary-400' : 'text-transparent group-hover:text-gray-200'}" /> </label> {#if active} <div class=" overflow-hidden rounded-xl px-3 pb-2"> <div class="flex flex-row flex-nowrap gap-2 pb-1"> <div class="text-xs font-semibold text-gray-500">System Prompt</div> {#if expanded} <button class="text-gray-500 hover:text-gray-900" on:click|preventDefault={onToggle} > <CarbonSave class="text-sm " /> </button> <button class="text-gray-500 hover:text-gray-900" on:click|preventDefault={() => { value = model.preprompt ?? ""; }} > <CarbonRestart class="text-sm " /> </button> {:else} <button class=" text-gray-500 hover:text-gray-900" on:click|preventDefault={onToggle} > <CarbonEdit class="text-sm " /> </button> {/if} </div> <textarea enterkeyhint="send" tabindex="0" rows="1" class="h-20 w-full resize-none scroll-p-3 overflow-x-hidden overflow-y-scroll rounded-md border border-gray-300 bg-transparent p-1 text-xs outline-none focus:ring-0 focus-visible:ring-0" bind:value hidden={!expanded} /> </div> {/if} <ModelCardMetadata {model} /> </div> {/each} </div> <button type="button" class="mt-2 rounded-full bg-black px-5 py-2 text-lg font-semibold text-gray-100 ring-gray-400 ring-offset-1 transition-colors hover:ring" on:click={() => dispatch("closeAndSave", { id: selectedNum, })} > Apply </button> </form> </Modal>