概要
ログインフォームと処理をServer ComponentsとReact Hook Formを使用して実装しました。
あとで見返すようとして、記事に残しておきます
(もっと良い方法やリファクタできる点があればぜひ教えてください...🙋♂️)
Server Actions
src/actions/login.ts
'use server';
import { apolloClient, LoginDocument } from '@/app/graphql';
import { cookies } from 'next/headers';
import { ApolloError } from '@apollo/client';
import { setFlash } from '@/actions/flash';
type LoginResponse = {
success: boolean;
redirectUrl?: string;
error?: string;
}
const loginProcessingError = 'ログイン処理中にエラーが発生しました。再度お試しください。'
export async function login(formData: FormData): Promise<LoginResponse> {
const email = formData.get('email') as string;
const password = formData.get('password') as string;
try {
const cookieHeader = await cookies();
const cookie = cookieHeader.toString();
const { data } = await apolloClient.mutate({
mutation: LoginDocument,
variables: { email, password },
context: {
headers: {
Cookie: cookie,
},
},
});
if (data?.login?.token) {
const cookieStore = await cookies();
cookieStore.set('ss_sid', data?.login?.token || '', {
expires: new Date(Date.now() + 365 * 24 * 60 * 60 * 1000),
httpOnly: true,
secure: true,
sameSite: 'lax'
});
await setFlash({
type: 'success',
message: 'ログインしました'
});
return { success: true, redirectUrl: '/account' };
} else {
throw new Error(loginProcessingError);
}
} catch (error) {
let errorMessage;
if (error instanceof ApolloError) {
errorMessage = error.message;
} else {
errorMessage = loginProcessingError
}
await setFlash({
type: 'error',
message: errorMessage
});
return {
success: false,
error: errorMessage
};
}
}
Login Form(React Hook Form使用)
src/components/ui/LoginForm/index.tsx
"use client";
import { useState } from "react";
import { useRouter } from "next/navigation";
import { z } from "zod";
import { useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import styles from "./LoginForm.module.css";
import Button from "@/components/ui/Button";
import Card from "@/components/ui/Card";
import Link from "next/link";
import { login } from "@/actions/login";
type LoginState = {
success: boolean;
error?: string | null;
redirectUrl?: string;
};
const loginSchema = z.object({
email: z.string().email("有効なメールアドレスを入力してください"),
password: z.string().min(6, "パスワードは6文字以上で入力してください"),
});
type LoginFormData = z.infer<typeof loginSchema>;
export default function LoginForm() {
const router = useRouter();
const [isLoading, setIsLoading] = useState(false);
const [error, setError] = useState<string | null>(null);
const {
register,
handleSubmit,
formState: { errors },
} = useForm<LoginFormData>({
resolver: zodResolver(loginSchema),
mode: "onBlur"
});
const onSubmit = async (data: LoginFormData) => {
setIsLoading(true);
setError(null);
try {
const formData = new FormData();
formData.append("email", data.email);
formData.append("password", data.password);
const result = await login(formData) as LoginState;
if (result.success) {
if (result.redirectUrl) {
router.push(result.redirectUrl);
}
} else {
setError(result.error || "ログインに失敗しました");
}
} catch {
setError("原因不明のエラーが発生しました。再度お試しください。");
} finally {
setIsLoading(false);
}
};
return (
<div className={styles.formContainer}>
<div className={styles.formWrapper}>
<Card title="アカウントにログインする">
{error && (
<div className={styles.errorAlert}>{error}</div>
)}
<form className={styles.form} onSubmit={handleSubmit(onSubmit)}>
<div className={styles.formFields}>
<div className={styles.formField}>
<label htmlFor="email" className={styles.fieldLabel}>
メールアドレス
</label>
<input
id="email"
type="email"
autoComplete="email"
{...register("email")}
className={`${styles.fieldInput} ${errors.email ? styles.fieldInputError : ""}`}
placeholder="例)example@example.com"
/>
{errors.email && (
<p className={styles.errorMessage}>{errors.email.message}</p>
)}
</div>
<div className={styles.formField}>
<label htmlFor="password" className={styles.fieldLabel}>
パスワード
</label>
<input
id="password"
type="password"
autoComplete="current-password"
{...register("password")}
className={`${styles.fieldInput} ${errors.password ? styles.fieldInputError : ""}`}
placeholder="例)password123"
/>
{errors.password ? (
<p className={styles.errorMessage}>{errors.password.message}</p>
) : (
<p className={styles.helpText}>6文字以上の文字列</p>
)}
</div>
</div>
<div>
<Button
type="primary"
size="large"
buttonType="submit"
isSolid
isFull
isDisabled={isLoading}
>
{isLoading ? "ログイン中..." : "ログイン"}
</Button>
</div>
<div className={styles.registerLink}>
<p>初めてご利用ですか? <Link href="/register">新規登録はこちら</Link></p>
</div>
</form>
</Card>
</div>
</div>
);
}