diff --git a/packages/ui/src/containers/EmailForm/EmailForm.test.tsx b/packages/ui/src/containers/EmailForm/EmailForm.test.tsx index fa26bc460..6f19a56b1 100644 --- a/packages/ui/src/containers/EmailForm/EmailForm.test.tsx +++ b/packages/ui/src/containers/EmailForm/EmailForm.test.tsx @@ -70,7 +70,7 @@ describe('', () => { }); test('should display and clear the form error message as expected', () => { - const { queryByText, container, getByText } = renderWithPageContext( + const { queryByText, container } = renderWithPageContext( { - // eslint-disable-next-line @silverhand/fp/no-mutation - errorMessageRef.current = errorMessage; - }, [errorMessage]); - - useEffect(() => { - if (errorMessageRef.current) { - clearErrorMessage?.(); - } - }, [clearErrorMessage, errorMessageRef, fieldValue.email]); - const onSubmitHandler = useCallback( async (event?: React.FormEvent) => { event?.preventDefault(); @@ -75,6 +61,8 @@ const EmailForm = ({ [validateForm, hasTerms, termsValidation, onSubmit, fieldValue.email] ); + const { onChange, ...rest } = register('email', emailValidation); + return (
{ + onChange(event); + clearErrorMessage?.(); + }} + {...rest} onClear={() => { setFieldValue((state) => ({ ...state, email: '' })); }} diff --git a/packages/ui/src/containers/PhoneForm/PhoneForm.test.tsx b/packages/ui/src/containers/PhoneForm/PhoneForm.test.tsx new file mode 100644 index 000000000..59c74ba31 --- /dev/null +++ b/packages/ui/src/containers/PhoneForm/PhoneForm.test.tsx @@ -0,0 +1,178 @@ +import { fireEvent, waitFor, act } from '@testing-library/react'; +import { MemoryRouter } from 'react-router-dom'; + +import renderWithPageContext from '@/__mocks__/RenderWithPageContext'; +import SettingsProvider from '@/__mocks__/RenderWithPageContext/SettingsProvider'; +import { getDefaultCountryCallingCode } from '@/utils/country-code'; + +import PhoneForm from './PhoneForm'; + +const onSubmit = jest.fn(); +const clearErrorMessage = jest.fn(); + +jest.mock('i18next', () => ({ + language: 'en', +})); + +describe('', () => { + afterEach(() => { + jest.clearAllMocks(); + }); + + const phoneNumber = '8573333333'; + const defaultCountryCallingCode = getDefaultCountryCallingCode(); + + test('render', () => { + const { queryByText, container } = renderWithPageContext( + + + + ); + expect(container.querySelector('input[name="phone"]')).not.toBeNull(); + expect(queryByText('action.continue')).not.toBeNull(); + }); + + test('render with terms settings', () => { + const { queryByText } = renderWithPageContext( + + + + + + ); + expect(queryByText('description.terms_of_use')).not.toBeNull(); + }); + + test('render with terms settings but hasTerms param set to false', () => { + const { queryByText } = renderWithPageContext( + + + + + + ); + expect(queryByText('description.terms_of_use')).toBeNull(); + }); + + test('required phone with error message', () => { + const { queryByText, container, getByText } = renderWithPageContext( + + + + ); + const submitButton = getByText('action.continue'); + + fireEvent.click(submitButton); + expect(queryByText('invalid_phone')).not.toBeNull(); + expect(onSubmit).not.toBeCalled(); + + const phoneInput = container.querySelector('input[name="phone"]'); + + if (phoneInput) { + fireEvent.change(phoneInput, { target: { value: '1113' } }); + expect(queryByText('invalid_phone')).not.toBeNull(); + + fireEvent.change(phoneInput, { target: { value: phoneNumber } }); + expect(queryByText('invalid_phone')).toBeNull(); + } + }); + + test('should display and clear the form error message as expected', () => { + const { queryByText, container } = renderWithPageContext( + + + + ); + + expect(queryByText('form error')).not.toBeNull(); + + const phoneInput = container.querySelector('input[name="phone"]'); + + if (phoneInput) { + fireEvent.change(phoneInput, { target: { value: phoneNumber } }); + expect(clearErrorMessage).toBeCalled(); + } + }); + + test('should blocked by terms validation with terms settings enabled', async () => { + const { container, getByText } = renderWithPageContext( + + + + + + ); + const phoneInput = container.querySelector('input[name="phone"]'); + + if (phoneInput) { + fireEvent.change(phoneInput, { target: { value: phoneNumber } }); + } + + const submitButton = getByText('action.continue'); + + act(() => { + fireEvent.click(submitButton); + }); + + await waitFor(() => { + expect(onSubmit).not.toBeCalled(); + }); + }); + + test('should call submit method properly with terms settings enabled but hasTerms param set to false', async () => { + const { container, getByText } = renderWithPageContext( + + + + + + ); + const phoneInput = container.querySelector('input[name="phone"]'); + + if (phoneInput) { + fireEvent.change(phoneInput, { target: { value: phoneNumber } }); + } + + const submitButton = getByText('action.continue'); + + act(() => { + fireEvent.click(submitButton); + }); + + await waitFor(() => { + expect(onSubmit).toBeCalledWith(`${defaultCountryCallingCode}${phoneNumber}`); + }); + }); + + test('should call submit method properly with terms settings enabled and checked', async () => { + const { container, getByText } = renderWithPageContext( + + + + + + ); + const phoneInput = container.querySelector('input[name="phone"]'); + + if (phoneInput) { + fireEvent.change(phoneInput, { target: { value: phoneNumber } }); + } + + const termsButton = getByText('description.agree_with_terms'); + fireEvent.click(termsButton); + + const submitButton = getByText('action.continue'); + + act(() => { + fireEvent.click(submitButton); + }); + + await waitFor(() => { + expect(onSubmit).toBeCalledWith(`${defaultCountryCallingCode}${phoneNumber}`); + }); + }); +}); diff --git a/packages/ui/src/containers/PhoneForm/PhoneForm.tsx b/packages/ui/src/containers/PhoneForm/PhoneForm.tsx new file mode 100644 index 000000000..b9e240ce5 --- /dev/null +++ b/packages/ui/src/containers/PhoneForm/PhoneForm.tsx @@ -0,0 +1,115 @@ +import classNames from 'classnames'; +import { useCallback, useEffect } from 'react'; +import { useTranslation } from 'react-i18next'; + +import Button from '@/components/Button'; +import ErrorMessage from '@/components/ErrorMessage'; +import { PhoneInput } from '@/components/Input'; +import PasswordlessSwitch from '@/containers/PasswordlessSwitch'; +import TermsOfUse from '@/containers/TermsOfUse'; +import useForm from '@/hooks/use-form'; +import usePhoneNumber from '@/hooks/use-phone-number'; +import useTerms from '@/hooks/use-terms'; + +import * as styles from './index.module.scss'; + +type Props = { + className?: string; + // eslint-disable-next-line react/boolean-prop-naming + autoFocus?: boolean; + hasTerms?: boolean; + hasSwitch?: boolean; + errorMessage?: string; + clearErrorMessage?: () => void; + onSubmit: (phone: string) => Promise; +}; + +type FieldState = { + phone: string; +}; + +const defaultState: FieldState = { phone: '' }; + +const PhoneForm = ({ + autoFocus, + hasTerms = true, + hasSwitch = false, + className, + errorMessage, + clearErrorMessage, + onSubmit, +}: Props) => { + const { t } = useTranslation(); + + const { termsValidation } = useTerms(); + const { countryList, phoneNumber, setPhoneNumber, isValidPhoneNumber } = usePhoneNumber(); + + const { fieldValue, setFieldValue, validateForm, register } = useForm(defaultState); + + // Validate phoneNumber with given country code + const phoneNumberValidation = useCallback( + (phoneNumber: string) => { + if (!isValidPhoneNumber(phoneNumber)) { + return 'invalid_phone'; + } + }, + [isValidPhoneNumber] + ); + + // Sync phoneNumber + useEffect(() => { + setFieldValue((previous) => ({ + ...previous, + phone: `${phoneNumber.countryCallingCode}${phoneNumber.nationalNumber}`, + })); + }, [phoneNumber, setFieldValue]); + + const onSubmitHandler = useCallback( + async (event?: React.FormEvent) => { + event?.preventDefault(); + + if (!validateForm()) { + return; + } + + if (hasTerms && !(await termsValidation())) { + return; + } + + await onSubmit(fieldValue.phone); + }, + [validateForm, hasTerms, termsValidation, onSubmit, fieldValue.phone] + ); + + return ( + + { + setPhoneNumber((previous) => ({ ...previous, ...data })); + clearErrorMessage?.(); + }} + /> + + {errorMessage && {errorMessage}} + + {hasSwitch && } + + {hasTerms && } + +