"use client"; require("../polyfill"); import { useState, useEffect, useRef, useCallback, MouseEventHandler, } from "react"; import { IconButton } from "./button"; import styles from "./home.module.scss"; import SettingsIcon from "../icons/settings.svg"; import GithubIcon from "../icons/github.svg"; import ChatGptIcon from "../icons/chatgpt.svg"; import BotIcon from "../icons/bot.svg"; import AddIcon from "../icons/add.svg"; import LoadingIcon from "../icons/three-dots.svg"; import CloseIcon from "../icons/close.svg"; import { useChatStore } from "../store"; import { isMobileScreen } from "../utils"; import Locale from "../locales"; import { Chat } from "./chat"; import dynamic from "next/dynamic"; import { REPO_URL } from "../constant"; import { ErrorBoundary } from "./error"; import { useDebounce } from "use-debounce"; export function Loading(props: { noLogo?: boolean }) { return (
{!props.noLogo && }
); } const Settings = dynamic(async () => (await import("./settings")).Settings, { loading: () => , }); const ChatList = dynamic(async () => (await import("./chat-list")).ChatList, { loading: () => , }); function useSwitchTheme() { const config = useChatStore((state) => state.config); useEffect(() => { document.body.classList.remove("light"); document.body.classList.remove("dark"); 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])', ); 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]); } function useDragSideBar() { const limit = (x: number) => Math.min(500, Math.max(220, x)); const chatStore = useChatStore(); const startX = useRef(0); const startDragWidth = useRef(chatStore.config.sidebarWidth ?? 300); const lastUpdateTime = useRef(Date.now()); const handleMouseMove = useRef((e: MouseEvent) => { if (Date.now() < lastUpdateTime.current + 100) { return; } lastUpdateTime.current = Date.now(); const d = e.clientX - startX.current; const nextWidth = limit(startDragWidth.current + d); chatStore.updateConfig((config) => (config.sidebarWidth = nextWidth)); }); const handleMouseUp = useRef(() => { startDragWidth.current = chatStore.config.sidebarWidth ?? 300; window.removeEventListener("mousemove", handleMouseMove.current); window.removeEventListener("mouseup", handleMouseUp.current); }); const onDragMouseDown = (e: MouseEvent) => { startX.current = e.clientX; window.addEventListener("mousemove", handleMouseMove.current); window.addEventListener("mouseup", handleMouseUp.current); }; useEffect(() => { if (isMobileScreen()) { return; } document.documentElement.style.setProperty( "--sidebar-width", `${limit(chatStore.config.sidebarWidth ?? 300)}px`, ); }, [chatStore.config.sidebarWidth]); return { onDragMouseDown, }; } const useHasHydrated = () => { const [hasHydrated, setHasHydrated] = useState(false); useEffect(() => { setHasHydrated(true); }, []); return hasHydrated; }; function _Home() { const [createNewSession, currentIndex, removeSession] = useChatStore( (state) => [ state.newSession, state.currentSessionIndex, state.removeSession, ], ); const chatStore = useChatStore(); const loading = !useHasHydrated(); const [showSideBar, setShowSideBar] = useState(true); // setting const [openSettings, setOpenSettings] = useState(false); const config = useChatStore((state) => state.config); // drag side bar const { onDragMouseDown } = useDragSideBar(); useSwitchTheme(); if (loading) { return ; } return (
ChatGPT Next
Build your own AI assistant.
{ setOpenSettings(false); setShowSideBar(false); }} >
} onClick={chatStore.deleteSession} />
} onClick={() => { setOpenSettings(true); setShowSideBar(false); }} shadow />
} text={Locale.Home.NewChat} onClick={() => { createNewSession(); setShowSideBar(false); }} shadow />
onDragMouseDown(e as any)} >
{openSettings ? ( { setOpenSettings(false); setShowSideBar(true); }} /> ) : ( setShowSideBar(true)} sideBarShowing={showSideBar} /> )}
); } export function Home() { return ( <_Home> ); }