|
<script lang="ts"> |
|
import { onDestroy, onMount } from "svelte"; |
|
import { goto, invalidate } from "$app/navigation"; |
|
import { page } from "$app/stores"; |
|
import { browser } from "$app/environment"; |
|
import "../styles/main.css"; |
|
import { base } from "$app/paths"; |
|
import { |
|
PUBLIC_ORIGIN, |
|
PUBLIC_APP_DISCLAIMER, |
|
PUBLIC_SHOW_LOCAL_MODELS_WARNING, |
|
} from "$env/static/public"; |
|
|
|
import { shareConversation } from "$lib/shareConversation"; |
|
import { UrlDependency } from "$lib/types/UrlDependency"; |
|
import { error } from "$lib/stores/errors"; |
|
|
|
import MobileNav from "$lib/components/MobileNav.svelte"; |
|
import NavMenu from "$lib/components/NavMenu.svelte"; |
|
import Toast from "$lib/components/Toast.svelte"; |
|
import ConfirmModal from "$lib/components/ConfirmModal.svelte"; |
|
import ShouldLoginModal from "$lib/components/ShouldLoginModal.svelte"; |
|
import SettingsModal from "$lib/components/SettingsModal2.svelte"; |
|
import LoadingModal from "$lib/components/LoadingModal.svelte"; |
|
import LoginModal from "$lib/components/LoginModal.svelte"; |
|
import { PUBLIC_APP_ASSETS, PUBLIC_APP_NAME } from "$env/static/public"; |
|
import { |
|
isloading_writable, |
|
is_init_writable, |
|
refresh_chats_writable, |
|
refresh_chats_writable_empty, |
|
api_key_writable, |
|
is_logged_writable, |
|
showLoggedPopup_writable, |
|
is_magic_writable, |
|
email_addr_writable |
|
} from "./LayoutWritable"; |
|
import { |
|
deleteAllChats, |
|
deleteChat, |
|
getChats, |
|
getMessages, |
|
modifyTitle, |
|
} from "../routes/LocalDB"; |
|
import { env } from "$env/dynamic/public"; |
|
import InitModelModal from "$lib/components/InitModelModal.svelte"; |
|
import { getApiKey } from "./tools"; |
|
|
|
export let data; |
|
let isloading = false; |
|
let isInit = false; |
|
let showWarning = PUBLIC_SHOW_LOCAL_MODELS_WARNING === "true" ? true : false; |
|
let shouldLogin = false; |
|
|
|
let go_to_main = false; |
|
|
|
let conversations_list = []; |
|
|
|
showLoggedPopup_writable.subscribe((value) => { |
|
shouldLogin = value; |
|
}); |
|
|
|
is_init_writable.subscribe((value) => { |
|
isInit = value; |
|
}); |
|
|
|
isloading_writable.subscribe((value) => { |
|
isloading = value; |
|
}); |
|
|
|
let isNavOpen = false; |
|
let isSettingsOpen = false; |
|
let errorToastTimeout: ReturnType<typeof setTimeout>; |
|
let currentError: string | null; |
|
|
|
refresh_chats_writable.subscribe(async (value) => { |
|
if (value.length > 0) { |
|
conversations_list = value; |
|
refresh_chats_writable.set([]); |
|
} |
|
}); |
|
|
|
refresh_chats_writable_empty.subscribe(async (value) => { |
|
conversations_list = []; |
|
refresh_chats_writable.set(conversations_list); |
|
}); |
|
|
|
export function getProgress(progress: number) {} |
|
|
|
async function onError() { |
|
// If a new different error comes, wait for the current error to hide first |
|
if ($error && currentError && $error !== currentError) { |
|
clearTimeout(errorToastTimeout); |
|
currentError = null; |
|
await new Promise((resolve) => setTimeout(resolve, 300)); |
|
} |
|
|
|
currentError = $error; |
|
|
|
errorToastTimeout = setTimeout(() => { |
|
$error = null; |
|
currentError = null; |
|
}, 3000); |
|
} |
|
|
|
async function deleteConversation(id: string) { |
|
await deleteChat(id); |
|
|
|
if ($page.params.id !== id) { |
|
await invalidate(UrlDependency.ConversationList); |
|
} else { |
|
await goto(`${base}/`, { invalidateAll: true }); |
|
} |
|
} |
|
|
|
async function deleteAllConversations(id: string) { |
|
await deleteAllChats(); |
|
|
|
if ($page.params.id !== id) { |
|
await invalidate(UrlDependency.ConversationList); |
|
} else { |
|
await goto(`${base}/`, { invalidateAll: true }); |
|
} |
|
} |
|
|
|
async function editConversationTitle(id: string, title: string) { |
|
await modifyTitle(id, title); |
|
} |
|
|
|
onMount(async () => { |
|
localStorage.theme = "dark"; |
|
await refreshChats(); |
|
}); |
|
|
|
onDestroy(() => { |
|
clearTimeout(errorToastTimeout); |
|
}); |
|
|
|
$: if ($error) onError(); |
|
|
|
data.requiresLogin = true; |
|
const requiresLogin = |
|
!$page.error && |
|
!$page.route.id?.startsWith("/r/") && |
|
(data.requiresLogin |
|
? !data.user |
|
: !data.settings.ethicsModalAcceptedAt && !!PUBLIC_APP_DISCLAIMER); |
|
|
|
let loginModalVisible = false; |
|
|
|
async function refreshChats() { |
|
let ret = await getChats(); |
|
data.conversations = ret; |
|
conversations_list = ret; |
|
} |
|
$: title = env.PUBLIC_APP_NAME; |
|
let loggedIn = false; |
|
{ |
|
// try { |
|
// const response = await fetch("https://cloud.mithrilsecurity.io/api/auth/getUserInfo", { |
|
// method: "GET", |
|
// credentials: "include", |
|
// headers: { |
|
// "Content-Type": "application/json", |
|
// }, |
|
|
|
{'event': 'login'}); |
|
{ |
|
// const res = await response.text() |
|
// const json: JSON = JSON.parse(res) |
|
// email_addr_writable.set(json.email) |
|
// console.log(res) |
|
// // Handle a successful response here |
|
// console.log("User is logged in successfully"); |
|
// var apiKey = await getApiKey(); |
|
// loggedIn = true; |
|
// is_logged_writable.set(loggedIn); |
|
// api_key_writable.set(apiKey); |
|
// } |
|
{ |
|
// // Handle errors here |
|
// console.error("User is not logged in"); |
|
// } |
|
{ |
|
// // Handle network errors here |
|
// console.error("Network error", err); |
|
|
|
// } |
|
|
|
|
|
|
|
</script> |
|
|
|
<svelte:head> |
|
<title>{PUBLIC_APP_NAME}</title> |
|
<meta name="og:description" content="Private Conversational AI" /> |
|
<meta property="og:title" content={PUBLIC_APP_NAME} /> |
|
<meta property="og:type" content="website" /> |
|
<meta property="og:url" content="{PUBLIC_ORIGIN || $page.url.origin}{base}" /> |
|
<meta |
|
property="og:image" |
|
content="{PUBLIC_ORIGIN || $page.url.origin}{base}/{PUBLIC_APP_ASSETS}/thumbnail.jpg" |
|
/> |
|
<link |
|
rel="icon" |
|
href="{PUBLIC_ORIGIN || $page.url.origin}{base}/{PUBLIC_APP_ASSETS}/favicon.png" |
|
type="image/png" |
|
/> |
|
|
|
<link |
|
rel="apple-touch-icon" |
|
href="{PUBLIC_ORIGIN || $page.url.origin}{base}/{PUBLIC_APP_ASSETS}/touch-icon-ipad-retina.png" |
|
sizes="167x167" |
|
type="image/png" |
|
/> |
|
<link |
|
rel="apple-touch-icon" |
|
href="{PUBLIC_ORIGIN || $page.url.origin}{base}/{PUBLIC_APP_ASSETS}/touch-icon-ipad.png" |
|
sizes="152x152" |
|
type="image/png" |
|
/> |
|
<link |
|
rel="apple-touch-icon" |
|
href="{PUBLIC_ORIGIN || |
|
$page.url.origin}{base}/{PUBLIC_APP_ASSETS}/touch-icon-iphone-retina.png" |
|
sizes="180x180" |
|
type="image/png" |
|
/> |
|
</svelte:head> |
|
|
|
<div |
|
class="grid h-full w-screen grid-cols-1 grid-rows-[auto,1fr] overflow-hidden text-smd dark:text-gray-300 md:grid-cols-[280px,1fr] md:grid-rows-[1fr]" |
|
> |
|
<MobileNav |
|
isOpen={isNavOpen} |
|
on:toggle={(ev) => (isNavOpen = ev.detail)} |
|
title={conversations_list.find((conv) => conv.id === $page.params.id)?.title} |
|
> |
|
<NavMenu |
|
conversations={conversations_list} |
|
user={data.user} |
|
canLogin={data.user === undefined && data.requiresLogin} |
|
signedIn={loggedIn} |
|
bind:loginModalVisible |
|
on:shareConversation={(ev) => shareConversation(ev.detail.id, ev.detail.title)} |
|
on:deleteConversation={(ev) => deleteConversation(ev.detail)} |
|
on:clickSettings={() => (isSettingsOpen = true)} |
|
on:editConversationTitle={(ev) => editConversationTitle(ev.detail.id, ev.detail.title)} |
|
/> |
|
</MobileNav> |
|
<nav class="grid max-h-screen grid-cols-1 grid-rows-[auto,1fr,auto] max-md:hidden"> |
|
<NavMenu |
|
conversations={conversations_list} |
|
user={data.user} |
|
canLogin={data.user === undefined && data.requiresLogin} |
|
signedIn={loggedIn} |
|
bind:loginModalVisible |
|
on:shareConversation={(ev) => shareConversation(ev.detail.id, ev.detail.title)} |
|
on:deleteConversation={(ev) => deleteConversation(ev.detail)} |
|
on:clickSettings={() => (isSettingsOpen = true)} |
|
on:editConversationTitle={(ev) => editConversationTitle(ev.detail.id, ev.detail.title)} |
|
/> |
|
</nav> |
|
{#if currentError} |
|
<Toast message={currentError} /> |
|
{/if} |
|
{#if shouldLogin} |
|
<ShouldLoginModal on:close={() => showLoggedPopup_writable.set(false)} /> |
|
{/if} |
|
{#if showWarning} |
|
<ConfirmModal on:close={() => (showWarning = false)} /> |
|
{/if} |
|
{#if isInit} |
|
<InitModelModal /> |
|
{/if} |
|
{#if isloading} |
|
<LoadingModal /> |
|
{/if} |
|
{#if isSettingsOpen} |
|
<SettingsModal |
|
on:close={() => (isSettingsOpen = false)} |
|
on:deleteAllConversations={() => ((isSettingsOpen = false), deleteAllChats())} |
|
settings={data.settings} |
|
models={data.models} |
|
/> |
|
{/if} |
|
<!-- {#if (requiresLogin && data.messagesBeforeLogin === 0) || loginModalVisible} --> |
|
<slot /> |
|
</div> |
|
|