mirror of
https://github.com/documenso/documenso.git
synced 2026-06-22 04:12:06 +10:00
122 lines
4.0 KiB
TypeScript
122 lines
4.0 KiB
TypeScript
import { getOptionalSession } from '@documenso/auth/server/lib/utils/get-session';
|
|
import {
|
|
IS_GOOGLE_SSO_ENABLED,
|
|
IS_MICROSOFT_SSO_ENABLED,
|
|
IS_OIDC_SSO_ENABLED,
|
|
isSignupEnabledForProvider,
|
|
OIDC_PROVIDER_LABEL,
|
|
} from '@documenso/lib/constants/auth';
|
|
import { isValidReturnTo, normalizeReturnTo } from '@documenso/lib/utils/is-valid-return-to';
|
|
import { Alert, AlertDescription } from '@documenso/ui/primitives/alert';
|
|
import { msg } from '@lingui/core/macro';
|
|
import { useLingui } from '@lingui/react';
|
|
import { Trans } from '@lingui/react/macro';
|
|
import { useEffect, useState } from 'react';
|
|
import { Link, redirect, useSearchParams } from 'react-router';
|
|
|
|
import { SignInForm } from '~/components/forms/signin';
|
|
import { SIGNUP_ERROR_MESSAGES } from '~/components/forms/signup';
|
|
import { appMetaTags } from '~/utils/meta';
|
|
|
|
import type { Route } from './+types/signin';
|
|
|
|
export function meta() {
|
|
return appMetaTags(msg`Sign In`);
|
|
}
|
|
|
|
export async function loader({ request }: Route.LoaderArgs) {
|
|
const { isAuthenticated } = await getOptionalSession(request);
|
|
|
|
// SSR env variables.
|
|
const isGoogleSSOEnabled = IS_GOOGLE_SSO_ENABLED;
|
|
const isMicrosoftSSOEnabled = IS_MICROSOFT_SSO_ENABLED;
|
|
const isOIDCSSOEnabled = IS_OIDC_SSO_ENABLED;
|
|
const oidcProviderLabel = OIDC_PROVIDER_LABEL;
|
|
const isSignupEnabled =
|
|
isSignupEnabledForProvider('email') ||
|
|
(IS_GOOGLE_SSO_ENABLED && isSignupEnabledForProvider('google')) ||
|
|
(IS_MICROSOFT_SSO_ENABLED && isSignupEnabledForProvider('microsoft')) ||
|
|
(IS_OIDC_SSO_ENABLED && isSignupEnabledForProvider('oidc'));
|
|
|
|
let returnTo = new URL(request.url).searchParams.get('returnTo') ?? undefined;
|
|
|
|
returnTo = isValidReturnTo(returnTo) ? normalizeReturnTo(returnTo) : undefined;
|
|
|
|
if (isAuthenticated) {
|
|
throw redirect(returnTo || '/');
|
|
}
|
|
|
|
return {
|
|
isGoogleSSOEnabled,
|
|
isMicrosoftSSOEnabled,
|
|
isOIDCSSOEnabled,
|
|
isSignupEnabled,
|
|
oidcProviderLabel,
|
|
returnTo,
|
|
};
|
|
}
|
|
|
|
export default function SignIn({ loaderData }: Route.ComponentProps) {
|
|
const { isGoogleSSOEnabled, isMicrosoftSSOEnabled, isOIDCSSOEnabled, isSignupEnabled, oidcProviderLabel, returnTo } =
|
|
loaderData;
|
|
|
|
const { _ } = useLingui();
|
|
|
|
const [searchParams] = useSearchParams();
|
|
const [isEmbeddedRedirect, setIsEmbeddedRedirect] = useState(false);
|
|
|
|
const errorParam = searchParams.get('error');
|
|
const signupError = errorParam ? SIGNUP_ERROR_MESSAGES[errorParam] : undefined;
|
|
|
|
useEffect(() => {
|
|
const hash = window.location.hash.slice(1);
|
|
|
|
const params = new URLSearchParams(hash);
|
|
|
|
setIsEmbeddedRedirect(params.get('embedded') === 'true');
|
|
}, []);
|
|
|
|
return (
|
|
<div className="w-screen max-w-lg px-4">
|
|
<div className="z-10 rounded-xl border border-border bg-neutral-100 p-6 dark:bg-background">
|
|
{signupError && (
|
|
<Alert variant="destructive" className="mb-4">
|
|
<AlertDescription>{_(signupError)}</AlertDescription>
|
|
</Alert>
|
|
)}
|
|
|
|
<h1 className="font-semibold text-2xl">
|
|
<Trans>Sign in to your account</Trans>
|
|
</h1>
|
|
|
|
<p className="mt-2 text-muted-foreground text-sm">
|
|
<Trans>Welcome back, we are lucky to have you.</Trans>
|
|
</p>
|
|
<hr className="-mx-6 my-4" />
|
|
|
|
<SignInForm
|
|
isGoogleSSOEnabled={isGoogleSSOEnabled}
|
|
isMicrosoftSSOEnabled={isMicrosoftSSOEnabled}
|
|
isOIDCSSOEnabled={isOIDCSSOEnabled}
|
|
oidcProviderLabel={oidcProviderLabel}
|
|
returnTo={returnTo}
|
|
/>
|
|
|
|
{!isEmbeddedRedirect && isSignupEnabled && (
|
|
<p className="mt-6 text-center text-muted-foreground text-sm">
|
|
<Trans>
|
|
Don't have an account?{' '}
|
|
<Link
|
|
to={returnTo ? `/signup?returnTo=${encodeURIComponent(returnTo)}` : '/signup'}
|
|
className="text-documenso-700 duration-200 hover:opacity-70"
|
|
>
|
|
Sign up
|
|
</Link>
|
|
</Trans>
|
|
</p>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|