Formik React Forms using Bootstrap with Validation Tutorial with Examples

In this React 15+ tutorial, we are going to discuss how to build robust and advanced forms with validation in the ReactJs and Native applications using Formik.

Creating forms in an application is always a painful and time-consuming process spacially in a composite application like React.

A Form not only takes input from the user and throw in the database but also includes various events and validation process which must be handled with care to consume only useful information.

Formik is a powerful tool which makes it very handy to construct simple to advanced level of forms with component support for multiple input control types, validation errors, and event handling.

Let’s get into it…

[lwptoc]

 

Create a React Application

First, we’ll create a new React application using npx create-react-app command

$ npx create-react-app react-formik-tutorial-app

Move inside the react app

$ cd react-formik-tutorial-app

Run application

$ npm start

 

Install formik Package

After creating the React application ready, install the formik package by running below npm command

$ npm install formik --save

 

Create Contact Us Form Component

To understand the implementation of Formik forms, let’s create a new component with a Contact Us form. Create a new file ~src/components/contactus-form.component.js

For creating a Formik form, import Formik, Form and Field components from 'formik'

Formik: It is used as a hook to initialize form values, add form Submit even handler, and Validation logic.

Form: It acts as an HTML form tag to wrap form controls.

Field: It is used to create form controls inside the form like input text, select box, textarea, checkboxes, etc.

 

Add CSS Style for Form Controls

As a quick trick, we’ll install the bootstrap package and use the bootstrap.css and its classes to style our Formik form.

Run following npm command to install bootstrap package.

$ npm install bootstrap

Now we just need to import the bootstrap CSS file in our component.

...
import 'bootstrap/dist/css/bootstrap.min.css';
...

 

Build Contact Us Form

In our Contact Us form, we’ll have Name, Email, Subject and Content fields to get data from the user.

Update the ‘~src/components/contactus-form.component.js‘ file

// components/contactus-form.component.js

import React from 'react';
import { Formik, Field, Form } from 'formik';
import 'bootstrap/dist/css/bootstrap.min.css';

const ContactUsForm = () => {
    return (
        <Formik
            initialValues={{ name: '', email: '', subject: '', content: '' }}
            onSubmit={(values, { setSubmitting }) => {
                setTimeout(() => {
                    alert(JSON.stringify(values, null, 2));
                    setSubmitting(false);
                }, 1000);
            }}
        >
            {({ isSubmitting }) => (
                <Form>
                    <div className="form-group">
                        <label htmlFor="name">Name</label>
                        <Field name="name" className="form-control" type="text" />
                    </div>

                    <div className="form-group">
                        <label htmlFor="email">Email Address</label>
                        <Field name="email" className="form-control" type="email" />
                    </div>

                    <div className="form-group">
                        <label htmlFor="subject">Subject</label>
                        <Field name="subject" className="form-control" type="text" />
                    </div>

                    <div className="form-group">
                        <label htmlFor="content">Content</label>
                        <Field name="content" className="form-control" as="textarea" />
                    </div>
                    <div className="form-group">
                        <button type="submit" className="btn btn-primary" disabled={isSubmitting}>{isSubmitting ? "Please wait..." : "Submit"}</button>
                    </div>

                </Form>
            )}
        </Formik>
    );
};

export default ContactUsForm;

 

The <Formik> component is having the initialValues prop to set form controls with initial values by their name.

The OnSubmit() function is triggered when the form is submitted. This can be used to POST form data on the server using HTTP calls.

The setSubmitting() controls the state to check if the form is in processing state. We can use the isSubmitting boolean flag on the submit button.

We have used bootstrap classes like form-group and form-control to style the input form controls.

 

Adding Form in App

Now open the App.js file with function component and import ContactUsForm to render in App function.

// App.js
import React from 'react';
import './App.css';
import ContactUsForm from './components/contactus-form.component';

function App() {
  return (
    <div className="container">
      <div className="form-wrapper">

        <h4>Contact Us</h4>

        <ContactUsForm />

      </div>
    </div>
  );
}

export default App;

Run React app by hitting $ npm start to see it working:

react-formik-form-demo-1

 

 

In the form, we used text inputs and textarea input. Other types of form controls can also be added discussed in the next section.

 

Different Types of Formik Form Controls

Using the <Field /> component we can create multiple types of form controls. By default, it renders to create a text field.

Textarea Form Control

The text area is created by adding the as="textarea" property

<Field name="content" className="form-control" as="textarea" rows={3} cols={10} />

Rows and Cols can be defined as rows={3} cols={10}

 

 

Select Form Control

The select box is created by adding the as="select" property

<Field name="framework" as="select" multiple={true} className="form-control">
    <option value="react">React</option>
    <option value="ng">Angular</option>
    <option value="vue">Vue</option>
</Field>

The multiple property can be set to true or false for multiple file selection.

 

Checkbox Form Control

The <Field /> component will create a checkbox when type="checkbox" is added. In bootstrap we can create as shown below:

<div class="form-check">
     <Field name="terms" className="form-check-input" type="checkbox" id="gridCheck" />
     <label className="form-check-label" htmlFor="gridCheck">Accept Terms & Conditions</label>
</div>

 

Radio Form Control

The <Field /> component will create a checkbox when type="radio" is added with name and value. In bootstrap we can create as shown below:

<div className="form-group">
         <label>Gender</label>
         <div class="form-check">
                  <Field className="form-check-input" type="radio" name="gridRadios" id="gridRadios1" value="male" />
                  <label className="form-check-label" htmlFor="gridRadios1">Male</label>
          </div>
         <div class="form-check">
                   <Field className="form-check-input" type="radio" name="gridRadios" id="gridRadios2" value="female" />
                    <label className="form-check-label" htmlFor="gridRadios2">Female</label>
           </div>
</div>

 

File Browse Form Control

The File Browser controls are used to upload single or multiple files. The <Field/> component with bootstrap classes can be created as shown below

<div className="form-group">
    <label>Upload File</label>
    <div className="custom-file">
        <Field type="file" name="uploadedfile" className="custom-file-input" id="customFile" />
        <label className="custom-file-label" htmlFor="customFile">Choose file</label>
    </div>
</div>

 

Validation in Formik Form

Now we’ll add validation on Formik form we just created. The Formik component takes two properties validate and validationSchema. Any of these can be used to add validation on the form. Let’s discuss one by one.

 

Formik’s validate property

In the validate property we define an Object with validation rules for each form control. For making the Name, Email, Subject mandatory add following

// components/contactus-form.component.js

import React from 'react';
import { Formik, Field, Form } from 'formik';
import 'bootstrap/dist/css/bootstrap.min.css';

const ContactUsForm = () => {

    const validateForm = values => {
        const errors = {};
        if (!values.name) {
            errors.name = 'Name is required';
        } else if (values.name.length > 15) {
            errors.name = 'Must be 15 characters or less';
        }


        if (!values.email) {
            errors.email = 'Email is required';
        } else if (!/^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,4}$/i.test(values.email)) {
            errors.email = 'Invalid email address';
        }


        if (!values.subject) {
            errors.subject = 'Subject is required';
        }

        return errors;
    };


    return (
        <Formik
            initialValues={{ name: '', email: '', subject: '', content: '' }}
            onSubmit={(values, { setSubmitting }) => {
                setTimeout(() => {
                    alert(JSON.stringify(values, null, 2));
                    setSubmitting(false);
                }, 1000);
            }}
            validate={validateForm}
        >
            {(formik, isSubmitting) => (
                <Form>
                    <div className="form-group">
                        <label htmlFor="name">Name</label>
                        <Field name="name" className={(formik.touched.name && formik.errors.name) ? 'form-control is-invalid' : 'form-control'} type="text" />
                        
                        {formik.touched.name && formik.errors.name ? (
                            <div className="invalid-feedback">{formik.errors.name}</div>
                        ) : null}
                    </div>

                    <div className="form-group">
                        <label htmlFor="email">Email Address</label>
                        <Field name="email" className={(formik.touched.email && formik.errors.email) ? 'form-control is-invalid' : 'form-control'} type="email" />
                        
                        {formik.touched.email && formik.errors.email ? (
                            <div className="invalid-feedback">{formik.errors.email}</div>
                        ) : null}
                    </div>

                    <div className="form-group">
                        <label htmlFor="subject">Subject</label>
                        <Field name="subject" className={(formik.touched.subject && formik.errors.subject) ? 'form-control is-invalid' : 'form-control'} type="text" />
                        
                        {formik.touched.subject && formik.errors.subject ? (
                            <div className="invalid-feedback">{formik.errors.subject}</div>
                        ) : null}
                    </div>

                    <div className="form-group">
                        <label htmlFor="content">Content</label>
                        <Field name="content" className="form-control" as="textarea" rows={3} cols={10} />
                    </div>

                    <div className="form-group">
                        <button type="submit" className="btn btn-primary" disabled={isSubmitting}>{isSubmitting ? "Please wait..." : "Submit"}</button>
                    </div>

                </Form>
            )
            }
        </Formik >
    );
};

export default ContactUsForm;

The validateForm object keeps the rules for each control validation. Then the formik object is used to errors for each control and display messages. Errors will only be shown it form is touched.

We used Bootstrap error messages with is-invalid class.

 

Formik’s validationSchema property

Formik loves Yup, it is ready to use library for validation purposes. It lets you define inbuilt validation rules instead of writing them all.

Let’s apply validation using Yup. Install it by running below command

$ npm install yup --save

Then define the validationSchema property as shown

// components/contactus-form.component.js

import React from 'react';
import { Formik, Field, Form } from 'formik';
import 'bootstrap/dist/css/bootstrap.min.css';

import * as Yup from 'yup';

const ContactUsForm = () => {

    return (
        <Formik
            initialValues={{ name: '', email: '', subject: '', content: '' }}
            onSubmit={(values, { setSubmitting }) => {
                setTimeout(() => {
                    alert(JSON.stringify(values, null, 2));
                    setSubmitting(false);
                }, 1000);
            }}
            validationSchema={Yup.object({
                name: Yup.string()
                    .max(15, 'Must be 15 characters or less')
                    .required('Name is required'),
                subject: Yup.string()
                    .required('Subject is required'),
                email: Yup.string()
                    .email('Invalid email address')
                    .required('Email is required'),
            })}
        >
            {(formik, isSubmitting) => (
                <Form>
                    <div className="form-group">
                        <label htmlFor="name">Name</label>
                        <Field name="name" className={(formik.touched.name && formik.errors.name) ? 'form-control is-invalid' : 'form-control'} type="text" />
                        {formik.touched.name && formik.errors.name ? (
                            <div className="invalid-feedback">{formik.errors.name}</div>
                        ) : null}
                    </div>

                    <div className="form-group">
                        <label htmlFor="email">Email Address</label>
                        <Field name="email" className={(formik.touched.email && formik.errors.email) ? 'form-control is-invalid' : 'form-control'} type="email" />
                        {formik.touched.email && formik.errors.email ? (
                            <div className="invalid-feedback">{formik.errors.email}</div>
                        ) : null}
                    </div>

                    <div className="form-group">
                        <label htmlFor="subject">Subject</label>
                        <Field name="subject" className={(formik.touched.subject && formik.errors.subject) ? 'form-control is-invalid' : 'form-control'} type="text" />
                        {formik.touched.subject && formik.errors.subject ? (
                            <div className="invalid-feedback">{formik.errors.subject}</div>
                        ) : null}
                    </div>

                    <div className="form-group">
                        <label htmlFor="content">Content</label>
                        <Field name="content" className="form-control" as="textarea" rows={3} cols={10} />
                    </div>

                    <div className="form-group">
                        <button type="submit" className="btn btn-primary" disabled={isSubmitting}>{isSubmitting ? "Please wait..." : "Submit"}</button>
                    </div>

                </Form>
            )
            }
        </Formik >
    );
};

export default ContactUsForm;

 

Conclusion

Using Formik makes it easy to create complex forms. We can add any type of Form controls and add validation with customized login or use Yup with the special supporting property.

Here we used Bootstrap to create React form with validation messages. Using Bootstrap classes make development even faster with a responsive approach.

You can check more UI Bootstrap components tutorial here using the react-bootstrap package.

Do share your comments and suggestions below.

Stay Safe!

 

Leave a Comment

Your email address will not be published. Required fields are marked *