import { Button, Checkbox, Divider, Input } from "@mui/joy"; import { useEffect, useState } from "react"; import { toast } from "react-hot-toast"; import { Link } from "react-router-dom"; import AppearanceSelect from "@/components/AppearanceSelect"; import LocaleSelect from "@/components/LocaleSelect"; import * as api from "@/helpers/api"; import { absolutifyLink } from "@/helpers/utils"; import useLoading from "@/hooks/useLoading"; import useNavigateTo from "@/hooks/useNavigateTo"; import { useGlobalStore, useUserStore } from "@/store/module"; import { useUserV1Store } from "@/store/v1"; import { useTranslate } from "@/utils/i18n"; const SignIn = () => { const t = useTranslate(); const navigateTo = useNavigateTo(); const globalStore = useGlobalStore(); const userStore = useUserStore(); const userV1Store = useUserV1Store(); const actionBtnLoadingState = useLoading(false); const { appearance, locale, systemStatus } = globalStore.state; const mode = systemStatus.profile.mode; const [username, setUsername] = useState(""); const [password, setPassword] = useState(""); const [remember, setRemember] = useState(true); const disablePasswordLogin = systemStatus.disablePasswordLogin; const [identityProviderList, setIdentityProviderList] = useState([]); useEffect(() => { const fetchIdentityProviderList = async () => { const { data: identityProviderList } = await api.getIdentityProviderList(); setIdentityProviderList(identityProviderList); }; fetchIdentityProviderList(); }, []); useEffect(() => { if (mode === "demo") { setUsername("memos-demo"); setPassword("secret"); } }, [mode]); const handleUsernameInputChanged = (e: React.ChangeEvent) => { const text = e.target.value as string; setUsername(text); }; const handlePasswordInputChanged = (e: React.ChangeEvent) => { const text = e.target.value as string; setPassword(text); }; const handleLocaleSelectChange = (locale: Locale) => { globalStore.setLocale(locale); }; const handleAppearanceSelectChange = (appearance: Appearance) => { globalStore.setAppearance(appearance); }; const handleFormSubmit = (e: React.FormEvent) => { e.preventDefault(); handleSignInButtonClick(); }; const handleSignInButtonClick = async () => { if (username === "" || password === "") { return; } if (actionBtnLoadingState.isLoading) { return; } try { actionBtnLoadingState.setLoading(); const { data: user } = await api.signin(username, password, remember); if (user) { userStore.setCurrentUser(user); await userStore.fetchCurrentUser(); await userV1Store.getOrFetchUserByUsername(user.username); navigateTo("/"); } else { toast.error(t("message.login-failed")); } } catch (error: any) { console.error(error); toast.error(error.response.data.message || t("message.login-failed")); } actionBtnLoadingState.setFinish(); }; const handleSignInWithIdentityProvider = async (identityProvider: IdentityProvider) => { const stateQueryParameter = `auth.signin.${identityProvider.name}-${identityProvider.id}`; if (identityProvider.type === "OAUTH2") { const redirectUri = absolutifyLink("/auth/callback"); const oauth2Config = identityProvider.config.oauth2Config; const authUrl = `${oauth2Config.authUrl}?client_id=${ oauth2Config.clientId }&redirect_uri=${redirectUri}&state=${stateQueryParameter}&response_type=code&scope=${encodeURIComponent( oauth2Config.scopes.join(" ") )}`; window.location.href = authUrl; } }; return (

{systemStatus.customizedProfile.name}

{!disablePasswordLogin && ( <>
{t("common.username")}
{t("common.password")}
setRemember(e.target.checked)} />
{systemStatus.allowSignUp && (

{t("auth.sign-up-tip")} {t("common.sign-up")}

)} )} {identityProviderList.length > 0 && ( <> {!disablePasswordLogin && {t("common.or")}}
{identityProviderList.map((identityProvider) => ( ))}
)}
); }; export default SignIn;