blind_chat / src /routes /+layout.svelte
lauro1's picture
updates
d54ea4b
raw
history blame
8.33 kB
<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;
// async function isLogged() {
// try {
// const response = await fetch("https://cloud.mithrilsecurity.io/api/auth/getUserInfo", {
// method: "GET",
// credentials: "include",
// headers: {
// "Content-Type": "application/json",
// },
// });
// dataLayer.push({'event': 'login'});
// if (response.ok) {
// 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);
// }
// else {
// // Handle errors here
// console.error("User is not logged in");
// }
// } catch (err) {
// // Handle network errors here
// console.error("Network error", err);
// }
// is_magic_writable.set(true);
// }
// isLogged();
</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"
/>
<!-- Icon Support for iOS Bookmark Home Screen -->
<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>