| <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> |
|
|