From 29ed0983076c11bedfe3b59b989b9a91c48cc45c Mon Sep 17 00:00:00 2001
From: Adam Iskounen <44589599+iskounen@users.noreply.github.com>
Date: Wed, 16 Oct 2024 06:17:30 -0400
Subject: [PATCH] fix(DIA-930): tidy-up login with OTP step [WIP] (#10954)
* adds textContentType for password managers
* prevented users from switching code types for on_demand
* add title to otp step
* started writing tests
* change sentence case to title case
---
.../Auth2/scenes/ForgotPasswordStep.tsx | 2 +-
.../Onboarding/Auth2/scenes/LoginOTPStep.tsx | 193 +++++++++---------
.../Auth2/scenes/LoginWelcomeStep.tsx | 2 +-
.../scenes/__tests__/LoginOTPStep.test.tsx | 5 -
.../scenes/__tests__/LoginOTPStep.tests.tsx | 46 +++++
5 files changed, 148 insertions(+), 100 deletions(-)
delete mode 100644 src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.test.tsx
create mode 100644 src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.tests.tsx
diff --git a/src/app/Scenes/Onboarding/Auth2/scenes/ForgotPasswordStep.tsx b/src/app/Scenes/Onboarding/Auth2/scenes/ForgotPasswordStep.tsx
index bd711f92e1a..b8bf9b37eb2 100644
--- a/src/app/Scenes/Onboarding/Auth2/scenes/ForgotPasswordStep.tsx
+++ b/src/app/Scenes/Onboarding/Auth2/scenes/ForgotPasswordStep.tsx
@@ -146,7 +146,7 @@ const ForgotPasswordStepForm: React.FC = () => {
haptic="impactMedium"
testID="returnToLoginButton"
>
- Return to login
+ Return to Login
diff --git a/src/app/Scenes/Onboarding/Auth2/scenes/LoginOTPStep.tsx b/src/app/Scenes/Onboarding/Auth2/scenes/LoginOTPStep.tsx
index 4832a2a8af7..4c9d0fc6093 100644
--- a/src/app/Scenes/Onboarding/Auth2/scenes/LoginOTPStep.tsx
+++ b/src/app/Scenes/Onboarding/Auth2/scenes/LoginOTPStep.tsx
@@ -15,7 +15,7 @@ import {
} from "app/Scenes/Onboarding/Auth2/hooks/useAuthNavigation"
import { useInputAutofocus } from "app/Scenes/Onboarding/Auth2/hooks/useInputAutofocus"
import { GlobalStore } from "app/store/GlobalStore"
-import { Formik, useFormikContext } from "formik"
+import { Formik } from "formik"
import { useRef, useState } from "react"
import * as Yup from "yup"
@@ -24,13 +24,22 @@ interface LoginOTPStepFormValues {
}
export const LoginOTPStep: React.FC = () => {
+ const [codeType, setCodeType] = useState<"authentication" | "recovery">("authentication")
+
+ const navigation = useAuthNavigation()
const screen = useAuthScreen()
+ const otpRef = useRef(null)
+
+ const { color } = useTheme()
+
+ useInputAutofocus({
+ screenName: "LoginOTPStep",
+ inputRef: otpRef,
+ })
return (
initialValues={{ otp: "" }}
- validateOnChange={true}
- validateOnMount={false}
validationSchema={Yup.object().shape({
otp: Yup.string().required("This field is required"),
})}
@@ -54,101 +63,99 @@ export const LoginOTPStep: React.FC = () => {
}
}}
>
-
-
- )
-}
-
-const LoginOTPStepForm: React.FC = () => {
- const [recoveryCodeMode, setRecoveryCodeMode] = useState(false)
-
- const {
- errors,
- handleChange,
- handleSubmit,
- isSubmitting,
- isValid,
- setErrors,
- validateForm,
- values,
- resetForm,
- } = useFormikContext()
+ {({
+ errors,
+ handleChange,
+ handleSubmit,
+ isSubmitting,
+ isValid,
+ validateForm,
+ values,
+ resetForm,
+ }) => (
+
+ {
+ navigation.goBack()
+ resetForm()
+ setCodeType("authentication")
+ }}
+ />
+
+
+
+ Authentication Code
+
+ {
+ handleChange("otp")(text)
+ }}
+ onBlur={() => validateForm()}
+ />
+
+
+
+ {screen.params?.otpMode === "on_demand" && (
+ <>
+
+
+
+ Your safety and security are important to us. Please check your email for a one-time
+ authentication code to complete your login.
+
+ >
+ )}
- const navigation = useAuthNavigation()
- const screen = useAuthScreen()
- const otpRef = useRef(null)
-
- const { color } = useTheme()
-
- useInputAutofocus({
- screenName: "LoginOTPStep",
- inputRef: otpRef,
- })
+
- const handleBackButtonPress = () => {
- resetForm()
- navigation.goBack()
- setRecoveryCodeMode(false)
- }
+
- return (
-
-
-
- {
- // Hide error when the user starts to type again
- if (errors.otp) {
- setErrors({ otp: undefined })
- validateForm()
- }
- handleChange("otp")(text)
- }}
- onBlur={() => validateForm()}
- />
-
-
-
- {screen.params?.otpMode === "on_demand" && (
- <>
-
- Your safety and security are important to us. Please check your email for a one-time
- authentication code to complete your login.
-
- >
+ {screen.params?.otpMode === "standard" && (
+ {
+ if (codeType === "authentication") {
+ setCodeType("recovery")
+ } else {
+ setCodeType("authentication")
+ }
+ }}
+ >
+
+ {codeType === "authentication"
+ ? "Enter recovery code instead"
+ : "Enter authentication code"}
+
+
+ )}
+
)}
-
-
-
-
-
-
-
- {
- setRecoveryCodeMode((mode) => !mode)
- }}
- >
-
- {recoveryCodeMode ? "Enter authentication code" : "Enter recovery code instead"}
-
-
-
+
)
}
diff --git a/src/app/Scenes/Onboarding/Auth2/scenes/LoginWelcomeStep.tsx b/src/app/Scenes/Onboarding/Auth2/scenes/LoginWelcomeStep.tsx
index 2a093ce4023..1798a6d5373 100644
--- a/src/app/Scenes/Onboarding/Auth2/scenes/LoginWelcomeStep.tsx
+++ b/src/app/Scenes/Onboarding/Auth2/scenes/LoginWelcomeStep.tsx
@@ -128,7 +128,7 @@ const LoginWelcomeStepForm: React.FC = () => {
ref={emailRef}
spellCheck={false}
keyboardType="email-address"
- textContentType="none"
+ textContentType="username"
returnKeyType="next"
title="Email"
value={values.email}
diff --git a/src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.test.tsx b/src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.test.tsx
deleted file mode 100644
index 3c49297b976..00000000000
--- a/src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.test.tsx
+++ /dev/null
@@ -1,5 +0,0 @@
-describe("LoginOTPStep", () => {
- it("renders correctly", () => {
- // TODO
- })
-})
diff --git a/src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.tests.tsx b/src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.tests.tsx
new file mode 100644
index 00000000000..6af50e62e27
--- /dev/null
+++ b/src/app/Scenes/Onboarding/Auth2/scenes/__tests__/LoginOTPStep.tests.tsx
@@ -0,0 +1,46 @@
+import { fireEvent, screen } from "@testing-library/react-native"
+import { LoginOTPStep } from "app/Scenes/Onboarding/Auth2/scenes/LoginOTPStep"
+import { renderWithWrappers } from "app/utils/tests/renderWithWrappers"
+
+const mockUseAuthScreen = jest.fn()
+
+jest.mock("app/Scenes/Onboarding/Auth2/hooks/useAuthNavigation", () => ({
+ useAuthNavigation: jest.fn(),
+ useAuthScreen: () => mockUseAuthScreen(),
+}))
+
+describe("LoginOTPStep", () => {
+ beforeEach(() => {
+ mockUseAuthScreen.mockReturnValue({ params: { otpMode: "standard" } })
+ })
+
+ it("renders correctly", () => {
+ renderWithWrappers()
+
+ expect(screen.getByText("Authentication Code")).toBeDefined()
+ expect(screen.getByText("Authentication code")).toBeDefined()
+ })
+
+ it("allows the user to switch between authentication and recovery codes", () => {
+ renderWithWrappers()
+
+ const recoveryCodeButton = screen.getByText("Enter recovery code instead")
+ expect(recoveryCodeButton).toBeDefined()
+
+ fireEvent.press(recoveryCodeButton)
+
+ expect(screen.getByText("Recovery code")).toBeDefined()
+ })
+
+ it("renders instructions when OTP mode is 'on_demand'", () => {
+ mockUseAuthScreen.mockReturnValue({ params: { otpMode: "on_demand" } })
+
+ renderWithWrappers()
+
+ expect(
+ screen.getByText(
+ "Your safety and security are important to us. Please check your email for a one-time authentication code to complete your login."
+ )
+ ).toBeDefined()
+ })
+})