chat-ui / src /routes /+page.svelte
coyotte508
A new start
fc69895
<script lang="ts">
import { goto, replaceState } from "$app/navigation";
import { base } from "$app/paths";
import { page } from "$app/state";
import { usePublicConfig } from "$lib/utils/PublicConfig.svelte";
const publicConfig = usePublicConfig();
import ChatWindow from "$lib/components/chat/ChatWindow.svelte";
import { ERROR_MESSAGES, error } from "$lib/stores/errors";
import { pendingMessage } from "$lib/stores/pendingMessage";
import { useSettingsStore } from "$lib/stores/settings.js";
import { findCurrentModel } from "$lib/utils/models";
import { sanitizeUrlParam } from "$lib/utils/urlParams";
import { onMount, tick } from "svelte";
import { loading } from "$lib/stores/loading.js";
import { loadAttachmentsFromUrls } from "$lib/utils/loadAttachmentsFromUrls";
let { data } = $props();
let hasModels = $derived(Boolean(data.models?.length));
let files: File[] = $state([]);
let draft = $state("");
const settings = useSettingsStore();
async function createConversation(message: string) {
try {
$loading = true;
// check if $settings.activeModel is a valid model
// else check if it's an assistant, and use that model
// else use the first model
const validModels = data.models.map((model) => model.id);
let model;
if (validModels.includes($settings.activeModel)) {
model = $settings.activeModel;
} else {
model = data.models[0].id;
}
const res = await fetch(`${base}/conversation`, {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
model,
preprompt: $settings.customPrompts[$settings.activeModel],
}),
});
if (!res.ok) {
const errorMessage = (await res.json()).message || ERROR_MESSAGES.default;
error.set(errorMessage);
console.error("Error while creating conversation: ", errorMessage);
return;
}
const { conversationId } = await res.json();
// Ugly hack to use a store as temp storage, feel free to improve ^^
pendingMessage.set({
content: message,
files,
});
// invalidateAll to update list of conversations
await goto(`${base}/conversation/${conversationId}`, { invalidateAll: true });
} catch (err) {
error.set((err as Error).message || ERROR_MESSAGES.default);
console.error(err);
} finally {
$loading = false;
}
}
onMount(async () => {
try {
// Handle attachments parameter first
if (page.url.searchParams.has("attachments")) {
const result = await loadAttachmentsFromUrls(page.url.searchParams);
files = result.files;
// Show errors if any
if (result.errors.length > 0) {
console.error("Failed to load some attachments:", result.errors);
error.set(
`Failed to load ${result.errors.length} attachment(s). Check console for details.`
);
}
// Clean up URL
const url = new URL(page.url);
url.searchParams.delete("attachments");
history.replaceState({}, "", url);
}
const query = sanitizeUrlParam(page.url.searchParams.get("q"));
if (query) {
void createConversation(query);
const url = new URL(page.url);
url.searchParams.delete("q");
tick().then(() => {
replaceState(url, page.state);
});
return;
}
const promptQuery = sanitizeUrlParam(page.url.searchParams.get("prompt"));
if (promptQuery && !draft) {
draft = promptQuery;
const url = new URL(page.url);
url.searchParams.delete("prompt");
tick().then(() => {
replaceState(url, page.state);
});
}
} catch (err) {
console.error("Failed to process URL parameters:", err);
}
});
let currentModel = $derived(findCurrentModel(data.models, data.oldModels, $settings.activeModel));
</script>
<svelte:head>
<title>{publicConfig.PUBLIC_APP_NAME}</title>
</svelte:head>
{#if hasModels}
<ChatWindow
onmessage={(message) => createConversation(message)}
loading={$loading}
{currentModel}
models={data.models}
bind:files
bind:draft
/>
{:else}
<div class="mx-auto my-20 max-w-xl rounded-xl border p-6 text-center dark:border-gray-700">
<h2 class="mb-2 text-xl font-semibold">No models available</h2>
<p class="text-gray-600 dark:text-gray-300">
No chat models are configured. Set `OPENAI_BASE_URL` and ensure the server can reach the
endpoint, then reload. If unset, the app defaults to the Hugging Face router.
</p>
</div>
{/if}