1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
|
import type {
ActionFunctionArgs,
LoaderFunctionArgs,
MetaFunction,
} from "@remix-run/node";
import { Form, json, useActionData, useLocation } from "@remix-run/react";
import { ensureUserNotLoggedIn, login } from "../.server/auth";
import BorderedContainer from "../components/BorderedContainer";
import InputText from "../components/InputText";
import SubmitButton from "../components/SubmitButton";
export const meta: MetaFunction = () => [
{ title: "Login | iOSDC Japan 2024 Albatross.swift" },
];
export async function loader({ request }: LoaderFunctionArgs) {
return await ensureUserNotLoggedIn(request);
}
export async function action({ request }: ActionFunctionArgs) {
const formData = await request.clone().formData();
const username = String(formData.get("username"));
const password = String(formData.get("password"));
if (username === "" || password === "") {
return json(
{
message: "ユーザー名またはパスワードが誤っています",
errors: {
username:
username === "" ? "ユーザー名を入力してください" : undefined,
password:
password === "" ? "パスワードを入力してください" : undefined,
},
},
{ status: 400 },
);
}
if (username.includes("@")) {
return json(
{
message: "ユーザー名が誤っています",
errors: {
username: "メールアドレスではなくユーザー名を入力してください",
password: undefined,
},
},
{ status: 400 },
);
}
try {
await login(request);
} catch (error) {
if (error instanceof Error) {
return json(
{
message: error.message,
errors: {
username: undefined,
password: undefined,
},
},
{ status: 400 },
);
} else {
throw error;
}
}
return null;
}
export default function Login() {
const location = useLocation();
const searchParams = new URLSearchParams(location.search);
const registrationToken = searchParams.get("registration_token");
const loginErrors = useActionData<typeof action>();
return (
<div className="min-h-screen bg-gray-100 flex items-center justify-center">
<div className="mx-2">
<BorderedContainer>
<Form method="post" className="w-full max-w-sm p-2">
<h2 className="text-2xl mb-6 text-center">
fortee アカウントでログイン
</h2>
<p className="text-sm mb-4">
fortee
のアカウントをお持ちでない場合は、イベントスタッフにお声がけください。
</p>
{loginErrors?.message && (
<p className="text-red-500 text-sm mb-4">{loginErrors.message}</p>
)}
<div className="mb-4 flex flex-col gap-1">
<label
htmlFor="username"
className="block text-sm font-medium text-gray-700"
>
ユーザー名
</label>
<InputText type="text" name="username" id="username" required />
{loginErrors?.errors?.username && (
<p className="text-red-500 text-sm">
{loginErrors.errors.username}
</p>
)}
</div>
<div className="mb-6 flex flex-col gap-1">
<label
htmlFor="password"
className="block text-sm font-medium text-gray-700"
>
パスワード
</label>
<InputText
type="password"
name="password"
id="password"
autoComplete="current-password"
required
/>
{loginErrors?.errors?.password && (
<p className="text-red-500 text-sm">
{loginErrors.errors.password}
</p>
)}
</div>
<input
type="hidden"
name="registration_token"
value={registrationToken ?? ""}
/>
<div className="flex justify-center">
<SubmitButton type="submit">ログイン</SubmitButton>
</div>
</Form>
</BorderedContainer>
</div>
</div>
);
}
|