Forms are already notoriously tough to customize and style — to the extent that we’re already starting to see new ideas for more flexible control. But what we don’t often discuss is designing good-form experiences beyond validation. That’s what Jima Victor discusses in this article, focusing specifically on creating multi-step forms that involve navigation between sections.
For a multistep form, planning involves structuring questions logically across steps, grouping similar questions, and minimizing the number of steps and the amount of required information for each step. Whatever makes each step focused and manageable is what should be aimed for.
In this tutorial, we will create a multistep form for a job application. Here are the details we are going to be requesting from the applicant at each step:
- Personal Information
Collects applicant’s name, email, and phone number. - Work Experience
Collects the applicant’s most recent company, job title, and years of experience. - Skills & Qualifications
The applicant lists their skills and selects their highest degree. - Review & Submit
This step is not going to collect any information. Instead, it provides an opportunity for the applicant to go back and review the information entered in the previous steps of the form before submitting it.
You can think of structuring these questions as a digital way of getting to know somebody. You can’t meet someone for the first time and ask them about their work experience without first asking for their name.
Based on the steps we have above, this is what the body of our HTML with our form should look like. First, the main <form>
element:
Step 1 is for filling in personal information, like the applicant’s name, email address, and phone number:
Once the applicant completes the first step, we’ll navigate them to Step 2,
focusing on their work experience so that we can collect information
like their most recent company, job title, and years of experience.
We’ll tack on a new <fieldset>
with those inputs:
Step 3 is all about the applicant listing their skills and qualifications for the job they’re applying for:
And, finally, we’ll allow the applicant to review their information before submitting it:
hidden
attribute to every fieldset
element but the first one. This ensures that the user sees only the
first step. Once they are done with the first step, they can proceed to
fill out their work experience on the second step by clicking a
navigational button. We’ll add this button later on.Adding Styles #
To keep things focused, we’re not going to be emphasizing the styles in this tutorial. What we’ll do to keep things simple is leverage the Simple.css style framework to get the form in good shape for the rest of the tutorial.
If you’re following along, we can include Simple’s styles in the document <head>
:
And from there, go ahead and create a style.css
file with the following styles that I’ve folded up.
Form Navigation And Validation
An easy way to ruin the user experience for a multi-step form is to wait until the user gets to the last step in the form before letting them know of any error they made along the way. Each step of the form should be validated for errors before moving on to the next step, and descriptive error messages should be displayed to enable users to understand what is wrong and how to fix it.
Now, the only part of our form that is visible is the first step. To complete the form, users need to be able to navigate to the other steps. We are going to use several buttons to pull this off. The first step is going to have a Next button. The second and third steps are going to have both a Previous and a Next button, and the fourth step is going to have a Previous and a Submit button.
Notice: We’ve added onclick
attributes to the Previous and Next buttons to link them to their respective JavaScript functions: previousStep()
and nextStep()
.
The “Next” Button
The nextStep()
function is linked to the Next button. Whenever the user clicks the Next button, the nextStep()
function will first check to ensure that all the fields for whatever
step the user is on have been filled out correctly before moving on to
the next step. If the fields haven’t been filled correctly, it displays
some error messages, letting the user know that they’ve done something
wrong and informing them what to do to make the errors go away.
Before we go into the implementation of the nextStep
function, there are certain variables we need to define because they
will be needed in the function. First, we need the input fields from the
DOM so we can run checks on them to make sure they are valid.
Then, we’re going to need an array to store our error messages.
Also,
we would need an element in the DOM where we can insert those error
messages after they’ve been generated. This element should be placed in
the HTML just below the last fieldset
closing tag:
Add the above div
to the JavaScript code using the following line:
And finally, we need a variable to keep track of the current step.
let currentStep = 1;
Now that we have all our variables in place, here’s the implementation of the nextstep()
function:
The moment the Next
button is pressed, our code first checks which step the user is
currently on, and based on this information, it validates the data for
that specific step by calling the addValidationErrors()
function. If there are errors, we display them. Then, the form calls the validateStep()
function to verify that there are no errors before moving on to the
next step. If there are errors, it prevents the user from going on to
the next step.
Whenever the nextStep()
function runs,
the error messages are cleared first to avoid appending errors from a
different step to existing errors or re-adding existing error messages
when the addValidationErrors
function runs. The addValidationErrors
function is called for each step using the fields for that step as arguments.
Here’s how the addValidationErrors
function is implemented:
This is how the validateStep()
function is defined:
The validateStep()
function checks for errors. If there are none, it proceeds to the next step with the help of the showStep()
function.
The showStep()
function requires the four fieldsets in the DOM. Add the following line
to the top of the JavaScript code to make the fieldsets available:
What the showStep()
function does is to go through all the fieldsets
in our form and hide whatever fieldset
is not equal to the one we’re navigating to. Then, it updates the currentStep
variable to be equal to the step we’re navigating to.
The “Previous” Button #
The previousStep()
function is linked to the Previous button. Whenever the previous button is clicked, similarly to the nextStep
function, the error messages are also cleared from the page, and navigation is also handled by the showStep
function.
Whenever the showStep()
function is called with “currentStep - 1
” as an argument (as in this case), we go back to the previous step, while moving to the next step happens by calling the showStep()
function with “currentStep + 1
” as an argument (as in the case of the validateStep()
function).
Improving User Experience With Visual Cues #
One other way of improving the user experience for a multi-step form, is by integrating visual cues, things that will give users feedback on the process they are on. These things can include a progress indicator or a stepper to help the user know the exact step they are on.
Integrating A Stepper #
To integrate a stepper into our form (sort of like this one from Material Design), the first thing we need to do is add it to the HTML just below the opening <form>
tag.
Next,
we need to query the part of the stepper that will represent the
current step. This is the span tag with the class name of currentStep
.
Now, we need to update the stepper value whenever the previous or next buttons are clicked. To do this, we need to update the showStep()
function by appending the following line to it:
This line is added to the showStep()
function because the showStep()
function is responsible for navigating between steps and updating the currentStep
variable. So, whenever the currentStep
variable is updated, the currentStepDiv should also be updated to reflect that change.
Storing And Retrieving User Data #
One major way we can improve the form’s user experience is by storing user data in the browser. Multistep forms are usually long and require users to enter a lot of information about themselves. Imagine a user filling out 95% of a form, then accidentally hitting the F5 button on their keyboard and losing all their progress. That would be a really bad experience for the user.
Using localStorage
,
we can store user information as soon as it is entered and retrieve it
as soon as the DOM content is loaded, so users can always continue
filling out their forms from wherever they left off. To add this feature
to our forms, we can begin by saving the user’s information as soon as
it is typed. This can be achieved using the input
event.
Before adding the input
event listener, get the form element from the DOM:
Now we can add the input
event listener:
Next, we need to add some code to help us retrieve the user data once the DOM content is loaded.
Lastly, it is good practice to remove data from localStorage
as soon as it is no longer needed:
Adding The Current Step Value To localStorage
If
the user accidentally closes their browser, they should be able to
return to wherever they left off. This means that the current step value
also has to be saved in localStorage
.
To save this value, append the following line to the showStep()
function:
Now
we can retrieve the current step value and return users to wherever
they left off whenever the DOM content loads. Add the following code to
the DOMContentLoaded
handler to do so:
Also, do not forget to clear the current step value from localStorage
when the form is submitted.
The above line should be added to the submit handler.
Wrapping Up #
Creating multi-step forms can help improve user experience for complex data entry. By carefully planning out steps, implementing form validation at each step, and temporarily storing user data in the browser, you make it easier for users to complete long forms.
For the full implementation of this multi-step form, you can access the complete code on GitHub.
No comments:
Post a Comment