diff options
author | Masaya Tojo <masaya@tojo.tokyo> | 2024-09-02 00:42:51 +0900 |
---|---|---|
committer | Masaya Tojo <masaya@tojo.tokyo> | 2024-09-02 00:42:51 +0900 |
commit | edddd70996b05c4e091b6aa0af6c7f3ef7e690ee (patch) | |
tree | c4fffdd0b0633a337e7a1593d452e8b5c45fe291 /src | |
parent | 71abc48fd4525b9194df848a6915ab4dfc11c354 (diff) |
react-query で取得した値でセレクトボックスを出すところまで実装
Diffstat (limited to 'src')
-rw-r--r-- | src/SignUpConfirm.tsx | 19 | ||||
-rw-r--r-- | src/SignUpForm1.tsx | 6 | ||||
-rw-r--r-- | src/SignUpForm2.tsx | 18 | ||||
-rw-r--r-- | src/SignUpForm3.tsx | 104 | ||||
-rw-r--r-- | src/routes.tsx | 2 | ||||
-rw-r--r-- | src/signUpSchema.ts | 28 |
6 files changed, 150 insertions, 27 deletions
diff --git a/src/SignUpConfirm.tsx b/src/SignUpConfirm.tsx index 295bad6..63b9d10 100644 --- a/src/SignUpConfirm.tsx +++ b/src/SignUpConfirm.tsx @@ -1,17 +1,19 @@ import './SignUp.css' import { useLocation, Navigate } from 'react-router-dom'; -import { Form } from './signUpSchema' +import { Form1Data, Form2Data, Form3Data, FormData, displayTel } from './signUpSchema' export const SignUpConfirm = () => { const location = useLocation() - const form = location.state as Form | null; + const data = location.state as { form1: Form1Data, form2: Form2Data, form3: Form3Data } | null; - // form1 のデータがない場合は form1 にリダイレクトする - if (form === null) { + if (data === null) { return <Navigate replace to="/sign-up/form1" /> } + const { form1, form2, form3 } = data + const form: FormData = { ...form1, ...form2, ...form3 } + return ( <> <h1>会員登録 確認画面</h1> @@ -20,14 +22,21 @@ export const SignUpConfirm = () => { <div className="signUpForm"> <p>お名前: {form.name}</p> <p>名前カナ: {form.kana}</p> - <p>電話番号: {form.tel1}-{form.tel2}-{form.tel3}</p> + <p>電話番号: {displayTel(form)}</p> </div> <h2>ログイン関連の情報</h2> <div className="signUpForm"> <p>メールアドレス: {form.email}</p> <p>パスワード: セキュリティ上の理由のため非表示</p> </div> + <h2>プログラミング関連の情報</h2> + <div className="signUpForm"> + <p>GitHubのアカウントはある?: {form.hasGitHubRepo ? 'ある' : 'ない'}</p> + <p>GitHub の username: {form.gitHubUsername}</p> + <p>お気に入りの GitHub リポジトリ: {form.repoName}</p> + </div> </div> + <button>送信</button> </> ); } diff --git a/src/SignUpForm1.tsx b/src/SignUpForm1.tsx index 76c6020..4be29e9 100644 --- a/src/SignUpForm1.tsx +++ b/src/SignUpForm1.tsx @@ -2,15 +2,15 @@ import './SignUp.css' import { useForm } from 'react-hook-form' import { zodResolver } from "@hookform/resolvers/zod" import { useNavigate } from 'react-router-dom'; -import { Form1, form1Schema } from './signUpSchema' +import { Form1Data, form1Schema } from './signUpSchema' export const SignUpForm1 = () => { - const { register, handleSubmit, formState: { errors } } = useForm<Form1>({ + const { register, handleSubmit, formState: { errors } } = useForm<Form1Data>({ resolver: zodResolver(form1Schema), }); const navigate = useNavigate(); - const onsubmit = (state: Form1) => { + const onsubmit = (state: Form1Data) => { navigate('/sign-up/form2', { state }) } const onerror = (err: any) => console.log(err); diff --git a/src/SignUpForm2.tsx b/src/SignUpForm2.tsx index f355225..177373d 100644 --- a/src/SignUpForm2.tsx +++ b/src/SignUpForm2.tsx @@ -2,14 +2,14 @@ import './SignUp.css' import { useForm } from 'react-hook-form' import { zodResolver } from "@hookform/resolvers/zod" import { useLocation, Navigate, useNavigate } from 'react-router-dom'; -import { Form1, Form2, form2Schema } from './signUpSchema' +import { Form1Data, Form2Data, form2Schema } from './signUpSchema' export const SignUpForm2 = () => { - const { register, handleSubmit, formState: { errors } } = useForm<Form2>({ + const { register, handleSubmit, formState: { errors } } = useForm<Form2Data>({ resolver: zodResolver(form2Schema), }); const location = useLocation(); - const form1 = location.state as Form1 | null; + const form1 = location.state as Form1Data | null; const navigate = useNavigate(); // form1 のデータがない場合は form1 にリダイレクトする @@ -17,13 +17,11 @@ export const SignUpForm2 = () => { return <Navigate replace to="/sign-up/form1" /> } - const onsubmit = (form2: Form2) => { - const state = { - ...form1, - ...form2, - } - navigate('/sign-up/confirm', { state }) + const onsubmit = (form2: Form2Data) => { + const state = { form1, form2 } + navigate('/sign-up/form3', { state }) } + const onerror = (err: any) => console.log(err); return ( @@ -41,7 +39,7 @@ export const SignUpForm2 = () => { <div className="error">{errors.password?.message}</div> </div> <div> - <button type="submit">確認画面へ</button> + <button type="submit">次へ</button> </div> </form> </> diff --git a/src/SignUpForm3.tsx b/src/SignUpForm3.tsx new file mode 100644 index 0000000..bfd71d0 --- /dev/null +++ b/src/SignUpForm3.tsx @@ -0,0 +1,104 @@ +import './SignUp.css' +import { useForm } from 'react-hook-form' +import { zodResolver } from "@hookform/resolvers/zod" +import { useLocation, Navigate, useNavigate } from 'react-router-dom'; +import { Form1Data, Form2Data, Form3Data, form3Schema } from './signUpSchema' + +import { useQuery, QueryClient, QueryClientProvider } from 'react-query' + +export const SignUpForm3 = () => { + const queryClient = new QueryClient() + + return ( + <QueryClientProvider client={queryClient}> + <SignUpForm3Main /> + </QueryClientProvider> + ) +} + + +export const SignUpForm3Main = () => { + const { register, watch, setValue, handleSubmit, formState: { errors } } = useForm<Form3Data>({ + resolver: zodResolver(form3Schema), + + }); + const location = useLocation(); + const formData = location.state as { form1: Form1Data, form2: Form2Data } | null; + const navigate = useNavigate(); + const watchHasGitHubRepo = watch("hasGitHubRepo", false) + const gitHubUsername = watch("gitHubUsername", '') + + const repoQueryEnabled = watchHasGitHubRepo && gitHubUsername.length > 0; + + console.log(gitHubUsername) + console.log(repoQueryEnabled) + + const { data: reposData, isLoading: reposIsLoading, isError: reposIsError, error: reposError } = useQuery({ + queryKey: ['gitHubRepos', gitHubUsername], + queryFn: async () => { + const res = await fetch(`https://api.github.com/users/${gitHubUsername}/repos?sort=updated&direction=desc&per_page=100`) + if (res.ok) { return res.json(); } + throw new Error(res.statusText) + }, + enabled: repoQueryEnabled, + }) + + const repoNames: [string] = (repoQueryEnabled && !reposIsLoading && !reposIsError) ? + (reposData.map((json: any) => (json['name']))) : + [] + + // TODO: watchHasGitHubRepo が false の場合は gitHubUsername と repoName が '' になって欲しい + + // form1, form2 のデータがない場合は form1 にリダイレクトする + if (formData === null) { + return <Navigate replace to="/sign-up/form1" /> + } + + const { form1, form2 } = formData; + + const onsubmit = (form3: Form3Data) => { + const state = { form1, form2, form3 } + navigate('/sign-up/confirm', { state }) + } + const onerror = (err: any) => console.log(err); + + return ( + <> + <h1>会員登録 フェーズ 3</h1> + <form className="SignUpForm" onSubmit={handleSubmit(onsubmit, onerror)}> + <div> + <label htmlFor="hasGitHubRepo">GitHubのリポジトリある?: </label> + <input id="hasGitHubRepo" type="checkbox" {...register('hasGitHubRepo')}></input> + <div className="error">{errors.hasGitHubRepo?.message}</div> + </div> + { + (watchHasGitHubRepo && + <div> + <label htmlFor="gitHubUsername">github username: </label> + <input id="gitHubUsername" type="text" {...register('gitHubUsername')}></input> + </div> + ) + } + { + repoQueryEnabled && ( + ( + reposIsLoading ? 'Loading...' : + reposIsError ? `Error: ${reposError}` : + <div> + <label htmlFor="repoName">お気に入りのリポジトリを選んでください:</label> + <select id="repoName" {...register('repoName')}> + {repoNames.map((repoName) => ( + <option key={repoName} value={repoName}>{repoName}</option> + ))} + </select> + </div> + ) + ) + } + <div> + <button type="submit">確認画面へ</button> + </div> + </form > + </> + ); +} diff --git a/src/routes.tsx b/src/routes.tsx index 1a3daa1..c85de49 100644 --- a/src/routes.tsx +++ b/src/routes.tsx @@ -3,6 +3,7 @@ import { Top } from './Top'; import { SignUp } from './SignUp'; import { SignUpForm1 } from './SignUpForm1'; import { SignUpForm2 } from './SignUpForm2'; +import { SignUpForm3 } from './SignUpForm3'; import { SignUpConfirm } from './SignUpConfirm'; export const routes = createBrowserRouter([ @@ -10,5 +11,6 @@ export const routes = createBrowserRouter([ { path: '/sign-up', element: <SignUp /> }, { path: '/sign-up/form1', element: <SignUpForm1 /> }, { path: '/sign-up/form2', element: <SignUpForm2 /> }, + { path: '/sign-up/form3', element: <SignUpForm3 /> }, { path: '/sign-up/confirm', element: <SignUpConfirm /> }, ]); diff --git a/src/signUpSchema.ts b/src/signUpSchema.ts index 4d8b71f..3b75ba1 100644 --- a/src/signUpSchema.ts +++ b/src/signUpSchema.ts @@ -11,7 +11,7 @@ const form1SchemaWithoutRefine = z.object({ tel3: z.string(), }) -const addRefine = <Form1 extends z.ZodTypeAny>(schema: Form1) => ( +const addTelRefine = <Form1Data extends z.ZodTypeAny>(schema: Form1Data) => ( schema.refine( ({ tel1, tel2, tel3 }) => (tel1.length > 0 && tel2.length > 0 && tel3.length > 0), { @@ -30,9 +30,6 @@ const addRefine = <Form1 extends z.ZodTypeAny>(schema: Form1) => ( } )) - -export type Form1 = z.infer<typeof form1SchemaWithoutRefine> - export const form2Schema = z.object({ email: z.string().min(1, { message: '必須項目です' }).email({ message: 'メールアドレスを入力してください' }), password: z.string() @@ -41,11 +38,24 @@ export const form2Schema = z.object({ .max(128, { message: '128文字以下で入力してください' }) }) -export type Form2 = z.infer<typeof form2Schema> +const favLangSchema = z.object({ + name: z.string() +}) + +export const form3Schema = z.object({ + // favLangs: z.array(favLangSchema), + hasGitHubRepo: z.boolean(), + gitHubUsername: z.string(), + repoName: z.string(), +}) -const formSchemaWithoutRefine = form1SchemaWithoutRefine.merge(form2Schema) +const formSchemaWithoutRefine = form1SchemaWithoutRefine.merge(form2Schema).merge(form3Schema) +export const form1Schema = addTelRefine(form1SchemaWithoutRefine) +export const formSchema = addTelRefine(formSchemaWithoutRefine) -export type Form = z.infer<typeof formSchemaWithoutRefine> +export type Form1Data = z.infer<typeof form1SchemaWithoutRefine> +export type Form2Data = z.infer<typeof form2Schema> +export type Form3Data = z.infer<typeof form3Schema> +export type FormData = z.infer<typeof formSchemaWithoutRefine> -export const form1Schema = addRefine(form1SchemaWithoutRefine) -export const formSchema = addRefine(formSchemaWithoutRefine) +export const displayTel = ({ tel1, tel2, tel3 }: Form1Data) => (`${tel1}-${tel2}-${tel3}`) |