|
<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"; |
|
|
|
export let settings: LayoutData["settings"]; |
|
export let models: Array<Model>; |
|
|
|
let selectedModelId = settings.activeModel; |
|
|
|
const dispatch = createEventDispatcher<{ close: void }>(); |
|
|
|
let expanded = false; |
|
|
|
function onToggle() { |
|
if (expanded) { |
|
settings.customPrompts[selectedModelId] = value; |
|
} |
|
expanded = !expanded; |
|
} |
|
|
|
let value = ""; |
|
|
|
function onModelChange() { |
|
value = |
|
settings.customPrompts[selectedModelId] ?? |
|
models.filter((el) => el.id === selectedModelId)[0].preprompt ?? |
|
""; |
|
} |
|
|
|
$: selectedModelId, onModelChange(); |
|
</script> |
|
|
|
<Modal width="max-w-lg" on:close> |
|
<form |
|
action="{base}/settings" |
|
method="post" |
|
on:submit={() => { |
|
if (expanded) { |
|
onToggle(); |
|
} |
|
}} |
|
use:enhance={() => { |
|
dispatch("close"); |
|
}} |
|
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="relative 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 flex-col p-3" |
|
on:change |
|
aria-label={model.displayName} |
|
> |
|
<input |
|
type="radio" |
|
class="sr-only" |
|
name="activeModel" |
|
value={model.id} |
|
bind:group={selectedModelId} |
|
/> |
|
<div |
|
class="mb-1.5 block pr-8 text-sm font-semibold leading-tight text-gray-800 sm:text-base" |
|
> |
|
{model.displayName} |
|
</div> |
|
{#if model.description} |
|
<div class="text-xs text-gray-500 sm:text-sm">{model.description}</div> |
|
{/if} |
|
<CarbonCheckmark |
|
class="absolute right-2 top-2 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="submit" |
|
class="sticky bottom-6 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" |
|
> |
|
Apply |
|
</button> |
|
</form> |
|
</Modal> |
|
|