ChatGPT-Next-Web/app/components/home.tsx

209 lines
5.5 KiB
TypeScript
Raw Normal View History

2023-03-09 17:01:40 +00:00
"use client";
require("../polyfill");
import { useState, useEffect } from "react";
2023-03-11 17:14:07 +00:00
2023-03-09 17:01:40 +00:00
import { IconButton } from "./button";
2023-03-12 19:06:21 +00:00
import styles from "./home.module.scss";
2023-03-09 17:01:40 +00:00
import SettingsIcon from "../icons/settings.svg";
import GithubIcon from "../icons/github.svg";
import ChatGptIcon from "../icons/chatgpt.svg";
2023-04-02 15:05:54 +00:00
2023-03-09 17:01:40 +00:00
import BotIcon from "../icons/bot.svg";
import AddIcon from "../icons/add.svg";
2023-03-10 18:25:33 +00:00
import LoadingIcon from "../icons/three-dots.svg";
2023-03-14 17:44:42 +00:00
import CloseIcon from "../icons/close.svg";
2023-03-10 18:25:33 +00:00
import { useChatStore } from "../store";
import { isMobileScreen } from "../utils";
2023-03-21 14:56:27 +00:00
import Locale from "../locales";
2023-04-02 15:05:54 +00:00
import { Chat } from "./chat";
2023-03-13 16:25:07 +00:00
import dynamic from "next/dynamic";
2023-03-23 16:01:00 +00:00
import { REPO_URL } from "../constant";
import { ErrorBoundary } from "./error";
2023-04-01 18:40:00 +00:00
import calcTextareaHeight from "../calcTextareaHeight";
2023-03-21 14:56:27 +00:00
export function Loading(props: { noLogo?: boolean }) {
return (
<div className={styles["loading-content"]}>
{!props.noLogo && <BotIcon />}
<LoadingIcon />
</div>
);
}
2023-03-21 14:56:27 +00:00
const Settings = dynamic(async () => (await import("./settings")).Settings, {
loading: () => <Loading noLogo />,
});
const ChatList = dynamic(async () => (await import("./chat-list")).ChatList, {
loading: () => <Loading noLogo />,
2023-03-21 14:56:27 +00:00
});
2023-03-11 12:54:24 +00:00
2023-03-12 19:06:21 +00:00
function useSwitchTheme() {
const config = useChatStore((state) => state.config);
useEffect(() => {
document.body.classList.remove("light");
document.body.classList.remove("dark");
2023-03-12 19:06:21 +00:00
if (config.theme === "dark") {
document.body.classList.add("dark");
} else if (config.theme === "light") {
document.body.classList.add("light");
}
const metaDescriptionDark = document.querySelector(
'meta[name="theme-color"][media]',
);
const metaDescriptionLight = document.querySelector(
'meta[name="theme-color"]:not([media])',
);
2023-03-15 17:24:03 +00:00
if (config.theme === "auto") {
metaDescriptionDark?.setAttribute("content", "#151515");
metaDescriptionLight?.setAttribute("content", "#fafafa");
} else {
const themeColor = getComputedStyle(document.body)
.getPropertyValue("--theme-color")
.trim();
metaDescriptionDark?.setAttribute("content", themeColor);
metaDescriptionLight?.setAttribute("content", themeColor);
}
}, [config.theme]);
2023-03-19 15:13:10 +00:00
}
2023-03-27 08:58:53 +00:00
const useHasHydrated = () => {
const [hasHydrated, setHasHydrated] = useState<boolean>(false);
useEffect(() => {
setHasHydrated(true);
}, []);
return hasHydrated;
};
function _Home() {
2023-03-21 14:56:27 +00:00
const [createNewSession, currentIndex, removeSession] = useChatStore(
(state) => [
state.newSession,
state.currentSessionIndex,
2023-04-01 18:56:45 +00:00
state.removeSession,
],
2023-03-21 14:56:27 +00:00
);
2023-04-06 16:14:27 +00:00
const chatStore = useChatStore();
2023-03-27 08:58:53 +00:00
const loading = !useHasHydrated();
2023-03-14 17:44:42 +00:00
const [showSideBar, setShowSideBar] = useState(true);
2023-03-10 18:25:33 +00:00
2023-03-16 16:08:16 +00:00
// setting
2023-03-11 17:14:07 +00:00
const [openSettings, setOpenSettings] = useState(false);
2023-03-12 19:21:48 +00:00
const config = useChatStore((state) => state.config);
2023-03-11 17:14:07 +00:00
2023-03-12 19:06:21 +00:00
useSwitchTheme();
2023-03-13 16:25:07 +00:00
if (loading) {
return <Loading />;
2023-03-13 16:25:07 +00:00
}
2023-03-09 17:01:40 +00:00
return (
2023-03-12 19:21:48 +00:00
<div
2023-03-21 14:56:27 +00:00
className={`${
2023-03-30 16:20:47 +00:00
config.tightBorder && !isMobileScreen()
? styles["tight-container"]
: styles.container
2023-03-21 14:56:27 +00:00
}`}
2023-03-12 19:21:48 +00:00
>
2023-03-14 17:44:42 +00:00
<div
className={styles.sidebar + ` ${showSideBar && styles["sidebar-show"]}`}
>
2023-03-09 17:01:40 +00:00
<div className={styles["sidebar-header"]}>
<div className={styles["sidebar-title"]}>ChatGPT Next</div>
<div className={styles["sidebar-sub-title"]}>
Build your own AI assistant.
</div>
<div className={styles["sidebar-logo"]}>
<ChatGptIcon />
</div>
</div>
2023-03-12 19:06:21 +00:00
<div
className={styles["sidebar-body"]}
2023-03-20 16:51:20 +00:00
onClick={() => {
2023-03-21 14:56:27 +00:00
setOpenSettings(false);
setShowSideBar(false);
2023-03-20 16:51:20 +00:00
}}
2023-03-12 19:06:21 +00:00
>
2023-03-09 17:01:40 +00:00
<ChatList />
</div>
<div className={styles["sidebar-tail"]}>
<div className={styles["sidebar-actions"]}>
2023-03-14 17:44:42 +00:00
<div className={styles["sidebar-action"] + " " + styles.mobile}>
<IconButton
icon={<CloseIcon />}
2023-04-06 16:14:27 +00:00
onClick={chatStore.deleteSession}
2023-03-14 17:44:42 +00:00
/>
</div>
2023-03-09 17:01:40 +00:00
<div className={styles["sidebar-action"]}>
2023-03-11 17:14:07 +00:00
<IconButton
icon={<SettingsIcon />}
onClick={() => {
2023-03-21 14:56:27 +00:00
setOpenSettings(true);
setShowSideBar(false);
}}
shadow
2023-03-11 17:14:07 +00:00
/>
2023-03-09 17:01:40 +00:00
</div>
<div className={styles["sidebar-action"]}>
2023-03-23 16:01:00 +00:00
<a href={REPO_URL} target="_blank">
<IconButton icon={<GithubIcon />} shadow />
2023-03-10 18:25:33 +00:00
</a>
2023-03-09 17:01:40 +00:00
</div>
</div>
<div>
2023-03-10 18:25:33 +00:00
<IconButton
icon={<AddIcon />}
2023-03-20 16:17:45 +00:00
text={Locale.Home.NewChat}
2023-03-28 17:30:11 +00:00
onClick={() => {
createNewSession();
setShowSideBar(false);
}}
shadow
2023-03-10 18:25:33 +00:00
/>
2023-03-09 17:01:40 +00:00
</div>
</div>
</div>
2023-03-11 17:14:07 +00:00
<div className={styles["window-content"]}>
2023-03-14 17:44:42 +00:00
{openSettings ? (
2023-03-21 14:56:27 +00:00
<Settings
closeSettings={() => {
setOpenSettings(false);
setShowSideBar(true);
}}
/>
2023-03-14 17:44:42 +00:00
) : (
<Chat
key="chat"
showSideBar={() => setShowSideBar(true)}
sideBarShowing={showSideBar}
2023-04-01 18:40:00 +00:00
autoSize={{ minRows: 2, maxRows: 6 }}
/>
2023-03-14 17:44:42 +00:00
)}
2023-03-11 17:14:07 +00:00
</div>
2023-03-09 17:01:40 +00:00
</div>
);
}
export function Home() {
return (
<ErrorBoundary>
<_Home></_Home>
</ErrorBoundary>
);
}