Author: Patrick Prunty.
formik-otp-input
is an enhancement to the formik library,
designed specifically for React applications. This extension introduces a specialized OTP (one-time-password) input
feature. It offers a customizable input field count for the password, along with user-defined props and options for
autoFocus
, autoSubmit
, borderColor
, highlightColor
, textColor
and backgroundColor
. The component is responsive,
meaning it is compatible with Android and iOS device browsers. Additionally, this
extension supports autofill suggestions on mobile devices, which may vary based on the user's mobile or email service
provider, as well as the format of the email body send to the user's device.
The inspiration for this project came in part from the smooth checkout process experienced with Stripe/Link payments. Its integration is versatile, making it suitable for a variety of applications, such as:
- Verification processes via email or mobile.
- Authentication workflows through email/SMS or passwordless systems.
- Two-factor authentication (2FA) for added security.
- Secure online payment and transaction confirmation.
- User registration and login procedures for web and mobile applications.
- Quick response actions in interactive customer service tools.
🖥️️ A live demo of this component is hosted on GitHub pages and can be previewed by following THIS LINK.
🧑🏼💻 The source code used for the demo can be previewed by following THIS LINK.
Install the package by running:
npm install formik-otp-input
yarn install formik-otp-input
Start by importing React, Formik, Yup, and the OtpInput component:
import React from 'react';
import { useFormik } from 'formik';
import * as Yup from 'yup';
import OtpInput from 'formik-otp-input';
Set the length of your OTP. This will be used in the OtpInput component.
const YOUR_OTP_LENGTH = 6; // Replace this with the length of your OTP
(Optional) Define CSS styles for your form elements. Adjust these styles according to your UI requirements.
const formStyle = { /* ... */ };
const fieldStyle = { /* ... */ };
const errorTextStyle = { /* ... */ };
const submitButtonStyle = { /* ... */ };
Create a functional component for your form. Within this component, you will use Formik's useFormik hook to handle form state and submission.
const OtpForm = () => {
// Formik hook
const formik = useFormik({
initialValues: {
otp: '',
// ... other form fields if you wish
},
onSubmit: (values) => {
// Handle form submission
},
});
// Return the form JSX
return (
<form style={formStyle} onSubmit={formik.handleSubmit}>
{/* OtpInput component and other form elements go here */}
</form>
);
};
Integrate the OtpInput component into your form. Pass relevant props to customize its behavior and appearance.
<OtpInput
length={YOUR_OTP_LENGTH}
value={formik.values.otp}
inputType={"numeric"} // Options: numeric, alphabetic, alphanumeric
autoFocus={true} // Auto-focus first digit
autoSubmit={true} // Auto-submit form on full OTP entry
onBlur={formik.handleBlur}
onChange={formik.handleChange}
onFullFill={formik.handleSubmit}
setFieldError={formik.setFieldError}
setFieldTouched={formik.setFieldTouched}
highlightColor={'#4caf50'} // optional
// ... other props and style overrides
// textColor={'#FFFFFF'}
// backgroundColor={'#FFFFFF'}
// borderColor={'#FFFFFF'}
// ... override any pre-existing styles if required
// style={{
// 'backgroundColor': '#ffc300'
// }}
/>
(Optional) Add a section to display form validation errors related to the OTP field.
{formik.errors.otp && formik.touched.otp && (
<div style={errorTextStyle}>{formik.errors.otp}</div>
)}
(Optional - if autoSubmit
is disabled) Include a submit button to allow users to submit the form.
<button type="submit" style={submitButtonStyle} >Submit</button>
Finally, export your OtpForm component.
export default OtpForm;
import React from 'react';
import { useFormik } from 'formik';
import * as Yup from 'yup';
import OtpInput from 'formik-otp-input';
const YOUR_OTP_LENGTH = 6; // Replace this with the length of your OTP
// CSS Styles, adjust according to your needs
const formStyle = {
display: 'flex',
flexDirection: 'column',
alignItems: 'center',
padding: '20px',
};
const fieldStyle = {
margin: '10px 0',
};
const errorTextStyle = {
marginTop: '15px',
fontSize: '14px',
color: '#ff6b6b',
marginBottom: '10px',
};
const submitButtonStyle = {
padding: '10px 20px',
backgroundColor: '#4caf50',
color: 'white',
border: 'none',
borderRadius: '5px',
cursor: 'pointer',
marginTop: '20px',
};
// Form component
const OtpForm = () => {
const formik = useFormik({
initialValues: {
otp: '',
// ... other form fields if you wish
},
onSubmit: (values) => {
console.log('Form data:', values);
window.alert(`Submitted otp value = ${values.otp}`);
// Perform submission actions
},
});
return (
<form style={formStyle} onSubmit={formik.handleSubmit}>
<OtpInput
length={YOUR_OTP_LENGTH}
value={formik.values.otp}
inputType={"numeric"} // Default is numeric. Options are numeric, alphabetic or alphanumeric
autoFocus={true} // Default is true. Will auto-focus first digit if true
autoSubmit={true} // Default is true. Will auto-submit form onFullFill
onBlur={formik.handleBlur} // Formik handler, used to handle onBlur events
onChange={formik.handleChange} // Formik handler, used to handle change events
onFullFill={formik.handleSubmit} // Formik handler, used to handle autoSubmit
setFieldError={formik.setFieldError} // Formik handler, used to handle error rendering
setFieldTouched={formik.setFieldTouched}
// ... other props you can pass
highlightColor={'#4caf50'}
// textColor={'#FFFFFF'}
// backgroundColor={'#FFFFFF'}
// borderColor={'#FFFFFF'}
// ... override any pre-existing styles if required
// style={{
// 'backgroundColor': '#ffc300'
// }}
/>
{formik.errors.otp && formik.touched.otp && (
<div style={errorTextStyle}>{formik.errors.otp}</div>
)}
<button type="submit" style={submitButtonStyle} >Submit</button>
</form>
);
};
export default OtpForm;
Typically, one-time-password flow is a two-step process. The first, involves providing an email or mobile number and making an API call to the backend to trigger the generation of the one-time-password. The second, involves providing the OTP input field for the user to input before making a second API call to the server to validate the OTP.
The following example details how to integration the Index component in such a two-step process:
todo: add example
This project is licensed under the MIT License.