Thomas G. Lopes
IndexedDb & Structured output (#82)
1778c9e unverified
raw
history blame
4.71 kB
<script lang="ts">
import type { ConversationClass } from "$lib/state/conversations.svelte.js";
import { isNumber } from "$lib/utils/is.js";
import { watch } from "runed";
import IconX from "~icons/carbon/close";
import { GENERATION_CONFIG_KEYS, GENERATION_CONFIG_SETTINGS } from "./generation-config-settings.js";
import StructuredOutputModal from "./structured-output-modal.svelte";
import { maxAllowedTokens } from "./utils.svelte.js";
interface Props {
conversation: ConversationClass;
classNames?: string;
}
const { conversation, classNames = "" }: Props = $props();
const maxTokens = $derived(maxAllowedTokens(conversation));
watch(
() => maxTokens,
() => {
const curr = conversation.data.config.max_tokens;
if (!curr || curr <= maxTokens) return;
conversation.update({
config: {
...conversation.data.config,
max_tokens: maxTokens,
},
});
}
);
type Config = (typeof conversation)["data"]["config"];
function updateConfigKey<K extends keyof Config>(k: K, v: Config[K]) {
conversation.update({
...conversation.data,
config: {
...conversation.data.config,
[k]: v,
},
});
}
let editingStructuredOutput = $state(false);
</script>
<div class="flex flex-col gap-y-7 {classNames}">
{#each GENERATION_CONFIG_KEYS as key}
{@const { label, min, step } = GENERATION_CONFIG_SETTINGS[key]}
{@const isMaxTokens = key === "max_tokens"}
{@const max = isMaxTokens ? maxTokens : GENERATION_CONFIG_SETTINGS[key].max}
<div>
<div class="flex items-center justify-between">
<label for={key} class="mb-0.5 block text-sm font-medium text-gray-900 dark:text-white">
{label}
</label>
<div class="flex items-center gap-2">
{#if !isMaxTokens || isNumber(conversation.data.config[key])}
<input
type="number"
class="w-20 rounded-sm border bg-transparent px-1 py-0.5 text-right text-sm dark:border-gray-700"
{min}
{max}
{step}
bind:value={() => conversation.data.config[key], v => updateConfigKey(key, v)}
/>
{/if}
{#if isMaxTokens && isNumber(conversation.data.config[key])}
<button class="btn-mini" onclick={() => updateConfigKey(key, undefined)}> <IconX /> </button>
{:else if isMaxTokens}
<button class="btn-mini" onclick={() => updateConfigKey(key, maxTokens / 2)}> set </button>
{/if}
</div>
</div>
{#if !isMaxTokens || isNumber(conversation.data.config[key])}
<input
id={key}
type="range"
{min}
{max}
{step}
bind:value={() => conversation.data.config[key], v => updateConfigKey(key, v)}
class="h-2 w-full cursor-pointer appearance-none rounded-lg bg-gray-200 accent-black dark:bg-gray-700 dark:accent-blue-500"
/>
{/if}
</div>
{/each}
<label class="mt-2 flex cursor-pointer items-center justify-between">
<input
type="checkbox"
bind:checked={() => conversation.data.streaming, v => conversation.update({ streaming: v })}
class="peer sr-only"
/>
<span class="text-sm font-medium text-gray-900 dark:text-gray-300">Streaming</span>
<div
class="peer relative h-5 w-9 rounded-full bg-gray-200 peer-checked:bg-black peer-focus:outline-hidden after:absolute after:start-[2px] after:top-[2px] after:h-4 after:w-4 after:rounded-full after:border after:border-gray-300 after:bg-white after:transition-all after:content-[''] peer-checked:after:translate-x-full peer-checked:after:border-white dark:border-gray-600 dark:bg-gray-700 dark:peer-checked:bg-blue-600"
></div>
</label>
<label class="mt-2 flex cursor-pointer items-center justify-between" for="structured-output">
<span class="text-sm font-medium text-gray-900 dark:text-gray-300">Structured Output</span>
<div class="flex items-center gap-2">
<input
type="checkbox"
bind:checked={
() => conversation.data.structuredOutput?.enabled,
v => conversation.update({ structuredOutput: { ...conversation.data.structuredOutput, enabled: v ?? false } })
}
class="peer sr-only"
id="structured-output"
/>
<button class="btn-mini" type="button" onclick={() => (editingStructuredOutput = true)}> edit </button>
<div
class="peer relative h-5 w-9 rounded-full bg-gray-200 peer-checked:bg-black peer-focus:outline-hidden after:absolute after:start-[2px] after:top-[2px] after:h-4 after:w-4 after:rounded-full after:border after:border-gray-300 after:bg-white after:transition-all after:content-[''] peer-checked:after:translate-x-full peer-checked:after:border-white dark:border-gray-600 dark:bg-gray-700 dark:peer-checked:bg-blue-600"
></div>
</div>
</label>
</div>
<StructuredOutputModal {conversation} bind:open={editingStructuredOutput} />