-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhookForm.ts
155 lines (145 loc) · 5.2 KB
/
hookForm.ts
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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
import type {
DeepPartial,
FieldErrors,
FieldValues,
Path,
UseFormReturn,
} from 'react-hook-form';
import { useMultiPageFormBase } from './base';
import type {
FormPage,
FormSequence,
MultiPageFormParams,
SequenceChild,
} from './types';
/**
* Represents a form page configured to work with React Hook Form.
*
* @template DataT - The type of the form data extending FieldValues.
* @template ComponentProps - (optional) The props passed to the component.
*/
export type HookFormPage<
DataT extends FieldValues,
ComponentProps = DefaultHookFormPageProps<DataT>,
> = FormPage<DataT, ComponentProps, FieldErrors<DataT>>;
/**
* Represents a form sequence configured to work with React Hook Form.
*
* @template DataT - The type of the form data extending FieldValues.
* @template ComponentProps - (optional) The props passed to the components.
*/
export type HookFormSequence<
DataT extends FieldValues,
ComponentProps = DefaultHookFormPageProps<DataT>,
> = FormSequence<DataT, ComponentProps, FieldErrors<DataT>>;
/**
* Represents a sequence child (either a page or a sequence) configured to work with React Hook Form.
*
* @template DataT - The type of the form data extending FieldValues.
* @template ComponentProps - (optional) The props passed to the components.
*/
export type HookFormSequenceChild<
DataT extends FieldValues,
ComponentProps = DefaultHookFormPageProps<DataT>,
> = SequenceChild<DataT, ComponentProps, FieldErrors<DataT>>;
/**
* Parameters for initializing and managing a multi-page form using React Hook Form.
*
* @template DataT - The type of the form data extending FieldValues.
* @template ComponentProps - (optional) The props passed to the form components.
*/
export type MultiPageReactHookFormParams<
DataT extends FieldValues,
ComponentProps,
> = DefaultHookFormPageProps<DataT> &
Omit<
MultiPageFormParams<DataT, ComponentProps, FieldErrors<DataT>>,
'getCurrentData'
>;
/**
* Default props for a form page using React Hook Form.
*
* @template DataT - The type of the form data extending FieldValues.
*/
export type DefaultHookFormPageProps<DataT extends FieldValues> = {
hookForm: UseFormReturn<DataT>;
};
/**
* Recursively retrieves all mounted field names from React Hook Form's internal fields.
*
* @template DataT - The type of the form data extending FieldValues.
* @param {UseFormReturn<DataT>['control']['_fields']} fields - The internal fields from React Hook Form's control object.
* @param {Path<DataT>[]} [mountedFields=[]] - An array to collect the mounted field paths.
* @param {string} [prefix=''] - The prefix for nested field paths.
* @returns {Path<DataT>[]} An array of mounted field paths.
*/
function getMountedFields<DataT extends FieldValues>(
fields: UseFormReturn<DataT>['control']['_fields'],
mountedFields: Path<DataT>[] = [],
prefix = '',
): Path<DataT>[] {
for (const field in fields) {
if (fields[field]?._f) {
if (fields[field]?._f.mount) {
mountedFields.push(`${prefix}${field}` as Path<DataT>);
}
} else {
getMountedFields(
fields[
field
] as unknown as UseFormReturn<DataT>['control']['_fields'],
mountedFields,
`${prefix}${field}.`,
);
}
}
return mountedFields;
}
/**
* A hook that integrates multi-page form logic with React Hook Form.
*
* This hook wraps the `useMultiPageFormBase` hook, providing integration with React Hook Form.
* It handles form validation, triggers, and manages form state transitions between pages.
*
* @template DataT - The type of the form data extending FieldValues.
* @template ComponentProps - (optional) The props passed to the form components.
*
* @param {MultiPageReactHookFormParams<DataT, ComponentProps>} params - The parameters for configuring the multi-page form.
* @returns {ReturnType<typeof useMultiPageFormBase>} An object containing the current page, navigation functions, and navigation state.
*/
export const useMultiPageHookForm = <
DataT extends FieldValues,
ComponentProps = DefaultHookFormPageProps<DataT>,
>({
hookForm,
onBeforePageChange,
...rest
}: MultiPageReactHookFormParams<DataT, ComponentProps>) => {
const { trigger, reset, control } = hookForm;
const multiPageForm = useMultiPageFormBase({
// @ts-ignore
getCurrentData: () => hookForm.getValues(),
onBeforePageChange: async (data, page) => {
if (onBeforePageChange) {
const result = await onBeforePageChange(data, page);
if (result !== true && result) {
return result;
}
}
const mountedFields = getMountedFields(
hookForm.control._fields,
) as Path<DataT>[];
const valid = await trigger(mountedFields);
if (valid) {
reset(undefined, { keepValues: true });
} else {
control._updateFormState({
isSubmitted: true,
});
}
return valid;
},
...rest,
});
return multiPageForm;
};