Build a Contact Form with React, AJAX, PHP and reCaptcha

Build a Contact Form with React, AJAX, PHP and reCaptcha

Table of Contents

Your new website is almost ready. The only thing you need is a contact form. Then, you are on the right place! This tutorial will show you how to build contact form with React, AJAX and PHP. What’s more, you will also learn how to implement reCaptcha and make your contact form secure. Now, let’s get to the work so you can ship your new website.

Add package.json and install dependencies

On one hand, you may already have a project with contact page and need just the contact form. On the other, you may not. Assuming the later is true, let’s start with creating a minimal package.json. And, to keep things simple, let’s add only necessary dependencies. These dependencies are jquery, react, react-dom, react-recaptcha and react-scripts.

The react, react-dom don’t need any explanation. react-recaptcha is a React library for Google reCaptcha. You will use it in your form to make sure people, and robots especially, will not spam you. Or, at least to reduce the amount of spam messages. react-scripts will provide you with scripts to run, build, test and eject this project.

Finally, jquery will make it easier to handle the AJAX XMLHttpRequest. You will use it to submit the form. If you stick only with this, the final package.json will probably look like the example below. However, you can use any configuration and add anything you want. This is your contact form, or page. When you are done, install all dependencies with npm i or yarn.

// contact-form/package.json

{
  "name": "contact-form",
  "version": "0.1.0",
  "private": true,
  "dependencies": {
    "jquery": "^3.3.1",
    "react": "^16.6.3",
    "react-dom": "^16.6.3",
    "react-recaptcha": "^2.3.10",
    "react-scripts": "2.1.1"
  },
  "scripts": {
    "start": "react-scripts start",
    "build": "react-scripts build",
    "test": "react-scripts test",
    "eject": "react-scripts eject"
  }
}

Create indexes

Next, if you don’t have a contact page for the contact form, you will need to create index.html and index.js. The index.html will contain a container div for the contact form you will build with React. Below is an example of a simple index.html. Notice that you will need link to reCaptcha API script in HEAD section. This is necessary to make reCaptcha work. The dependency itself is not enough.

<!-- contact-form/public/index.html -->

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="utf-8">
    <link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico">
    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
    <meta name="theme-color" content="#000000">
    <link rel="manifest" href="%PUBLIC_URL%/manifest.json">

    <title>Contact me</title>

    <!-- Load reCaptcha API -->
    <script src="https://www.google.com/recaptcha/api.js?onload=onloadCallback&render=explicit" async defer></script>
  </head>
  <body>
    <noscript>
      You need to enable JavaScript to run this app.
    </noscript>

    <!-- Container for React -->
    <div id="root"></div>
  </body>
</html>

In index.js you will import React and ReactDOM and also the contact page. After that, you will render the contact into HTML DOM, into the div container with id “root”.

// contact-form/src/index.js

// Import React and ReactDOM
import React from 'react'
import ReactDOM from 'react-dom'

// Import contact page
import ContactPage from './contact-page'

// Render contact page in DOM
ReactDOM.render(<ContactPage />, document.getElementById('root'))

Build your contact page

Now, when you have index.html and index.js ready, it is time to build the contact page. Or, at least to build the contact form. As usually, you will start with importing all necessary libraries. This means that you will need to import jquery, react, react-dom, react-recaptcha. And, you can also import stylesheet with some styles to make your contact form look better.

Then, you will create new React component called “ContactPage”. This component will start with state. However, we will not use class constructor. This is not necessary. When it comes to state, you will use it as a place to store all the values provided by page visitors through all inputs in the contact form.

There will also be some additional keys that will help with validation of the contact form. After that, you will add a simple method to handle text inputs. It can be called “handleInput”. You will use this method, along with onChange event, to find the correct key in contact page state and update its value.

You will do this by using the name attribute every input element in the contact form will have. Notice that this method will first test the length of the value and if the input is for email address. If it is email address, it will use regexp to validate the address provided by page visitors. Otherwise, it will skip this validation and just update the correct key in state.

Next, you will add another method to handle visitor’s interaction with checkboxes. It will work in similar way as the one for input. It will use name attribute, check if checkbox is checked and update the state. Let’s call this method “handleInput”. Then, you will add another two small and simple methods for reCaptcha.

The first one, called “onCaptchaLoad”, will be used by reCaptcha when the plugin is loaded. It can just log a message in console. The second one will be used when reCaptcha successfully validates the visitor. When this happens, it will update the value of isCaptchaValid key in state. Finally, you will create the method for submitting the contact form.

This method will assume that fields for name, email and message are required. In other words, it will check the length of values of inputEmail, inputName and inputMessage keys stored in state. It will also check the value of isCaptchaValid key, to make sure reCaptcha validated the visitor.

If any of these checks fails, it will set the value of state key isErrorShown to true. This will trigger render of error message below the contact form. Otherwise, it will proceed. Meaning, it will set the isErrorShow key in state to false to make sure no error message is visible. Then, it will set the value of isFormValid key to true.

After doing this maintenance, it will proceed to making the AJAX XMLHttpRequest with jQuery. This will be simple. it will send the content of state via data and set the type of the request to POST. Next, it will specify the name of the PHP file that will provide the API for your contact form. Then, it will add methods to log success and error.

When this is done, the handleFormSubmit will do the last thing to finish the work. It will reset all values stored in state. If you use defaultValue attribute on inputs, and set to correct keys in state, this will trigger resetting the contact form and clearing all inputs. Below is example of how the contact form may look like.

Side note: Google reCaptcha plugin requires unique sitekey key in order to run. If you don’t have this key, you can get it on Google reCaptcha website. Then, pass this key to sitekey attribute on Recaptcha component in your contact form. I marked this area in the code with {/* !! */} to make it more visible.

// contact-form/src/contact-page.js

// Import React and ReactDOM
import React, { Component } from 'react'

// Import jQuery
import $ from 'jquery'

// Import reCaptcha
import Recaptcha from 'react-recaptcha'

// Import some simple styles for contact form
import './styles/styles.css'

export default class ContactPage extends Component {
  state = {
    inputEmail: '',
    inputCheckBoth: false,
    inputCheckDesign: false,
    inputCheckDev: false,
    inputMessage: '',
    inputName: '',
    isCaptchaValid: false,
    isErrorShown: false,
    isFormValid: false
  }

  // Handle visitor's interaction with inputs
  handleInput = event => {
    // Test for input and length of the value
    if (event.target.value.length > 0 && event.target.name !== 'inputEmail') {
      this.setState({
        [event.target.name]: event.target.value
      })
    }

    // If input is for email address validate it with regexp
    if (event.target.name === 'inputEmail') {
      // eslint-disable-next-line
      const reg = /^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/

      if (reg.test(String(event.target.value).toLowerCase())) {
        this.setState({
          [event.target.name]: event.target.value
        })
      }
    }
  }

  // Handle visitor's interaction with checkboxes
  handleCheckbox = event => {
    this.setState({
      [event.target.name]: event.target.checked
    })
  }

  // Show message in console when reCaptcha plugin is loaded
  onCaptchaLoad = () => {
    console.log('Captcha loaded')
  }

  // Update state after reCaptcha validates visitor
  onCaptchaVerify = (response) => {
    this.setState({
      isCaptchaValid: true
    })
  }

  handleFormSubmit = event => {
    event.preventDefault()

    // Test
    if (this.state.inputEmail.length > 0 && this.state.inputName.length > 0 && this.state.inputMessage.length > 0 && this.state.isCaptchaValid) {
      this.setState({
        isErrorShown: false,
        isFormValid: true
      })

      // Send the form with AJAX
      $.ajax({
        data: this.state,
        type: 'POST',
        url: '/mailer.php',
        success: function(data) {
          console.info(data)
        },
        error: function(xhr, status, err) {
          console.error(status, err.toString())
        }
      })

      // Reset state after sending the form
      this.setState({
        inputEmail: '',
        inputCheckBoth: false,
        inputCheckDesign: false,
        inputCheckDev: false,
        inputMessage: '',
        inputName: '',
        isCaptchaValid: false,
        isErrorShown: false,
        isFormValid: false
      })
    } else {
      // Show error message
      this.setState({
        isErrorShown: true
      })
    }
  }

  render() {
    return (
      <div className="contact-page">
        <h1>Let's get in touch!</h1>

        <p>Feel free to get in touch with me. I am always open to discussing new projects, creative ideas or opportunities to be part of your visions.</p>

        <form action="">
          <fieldset>
            <label htmlFor="inputName">Name</label>

            <input onChange={this.handleInput} type="text" name="inputName" id="inputName" required={true} />
          </fieldset>

          <fieldset>
            <label htmlFor="inputEmail">Email</label>

            <input onChange={this.handleInput} type="email" name="inputEmail" id="inputEmail" required={true} />
          </fieldset>

          <div className="form__row">
            <div className="form__col">
              <fieldset>
                <label htmlFor="inputCheckDesign">
                  <input onClick={this.handleCheckbox} type="checkbox" name="inputCheckDesign" id="inputCheckDesign" defaultChecked={false} />

                  <span>Design</span>
                </label>
              </fieldset>
            </div>

            <div className="form__col">
              <fieldset>
                <label htmlFor="inputCheckDev">
                  <input onClick={this.handleCheckbox} type="checkbox" name="inputCheckDev" id="inputCheckDev" defaultChecked={false} />

                  <span>Development</span>
                </label>
              </fieldset>
            </div>

            <div className="form__col">
              <fieldset>
                <label htmlFor="inputCheckBoth">
                  <input onClick={this.handleCheckbox} type="checkbox" name="inputCheckBoth" id="inputCheckBoth" defaultChecked={false} />

                  <span>Design &amp development</span>
                </label>
              </fieldset>
            </div>
          </div>

          <fieldset>
            <label>message</label>

            <textarea onChange={this.handleInput} name="inputMessage" id="inputMessage" required={true} />
          </fieldset>

          {/* !! */}
          {/* Make sure to use your 'sitekey' for Google reCaptcha API! */}
          {/* !! */}
          <fieldset>
            <Recaptcha
              onloadCallback={this.onCaptchaLoad}
              sitekey="xxxxxxxxxxxxxxx"
              render="explicit"
              verifyCallback={this.onCaptchaVerify}
            />
          </fieldset>

          {this.state.isFormSubmitted && (
            <fieldset>
              <p>Thank you for contacting me. I will reply in four days.</p>
            </fieldset>
          )}

          {this.state.isErrorShown && (
            <fieldset>
              <p>Please, make sure to fill all fields.</p>
            </fieldset>
          )}

          <fieldset>
            <button onClick={this.handleFormSubmit} className="btn">
              Send
            </button>
          </fieldset>
        </form>
      </div>
    )
  }
}

Add the PHP

You’ve just arrived to the final part. Now, you will write the PHP code and create API for your contact form so you can then send it. You may want to have the code for this contact form, and whole project, backed up in some repository. And, you may not want to tell everyone what your email address is. You can save your email in a file and then read it.

This is exactly the approach you will see in the PHP code for this contact form. First, you will save your email in some file. It doesn’t have to be text file. It can be without any file extension. For example, it can be .credentials. Just make sure to include only your email address. No other text. Next, you will use fopen() function to open it and store it in $credentialsFile variable.

After that, you will use fgets() function on that $credentialsFile variable to read the first line and store the result in $myEmail variable. Next, you will extract data from the contact form you sent via AJAX. As you remember, you passed the whole state as the value for data in the AJAX request. And, you sent it as POST.

This means that all this data is now content of global $_POST variability. This variable is an associative array and you can access all data using specific name. This name are the same as the keys in state. In other words, inputName, inputEmail, inputMessage, inputCheckBoth, inputCheckDesign, inputCheckDev and so on.

You will use trim to remove any potential white space at the beginning or end of the input values. Next, you will once again check if name, email and message variables contain text. If not, the server will return 400 response code. Meaning, there is some problem. Otherwise, it will use the variables with contact form data, put together the content of the email, and send it.

When all this is done, server will return 200 response code. This means that the message as sent. Otherwise, if there is some issue, server will return either 500 or 403 error response code, which one depends on the type of the issue.

Side note: You may hear about using isset() to check if checkboxes are checked. This is not necessary. In a fact, it would not work. As long as there are some checkboxes you would always get “1”. Instead, you can just load the value just as you did with other inputs. The result will be the content of the state. You will get either true or false.

It is up to you to decide whether to use these values in the email template. Otherwise, you can create some simple if statement with custom text when checkbox is checked (true) and when it is not (false).

// contact-form/src/mailer.php

<?php
    // Only process POST requests.
    if ($_SERVER["REQUEST_METHOD"] == "POST") {
      // Get email address from '.credentials' file in the root
      $credentialsFile = fopen(".credentials","r");
      $myEmail = fgets($credentialsFile);

      // Get the form fields and remove any potential whitespace.
      $name = strip_tags(trim($_POST["inputName"]));
      $name = str_replace(array("\r","\n"),array(" "," "),$name);
      $email = filter_var(trim($_POST["inputEmail"]), FILTER_SANITIZE_EMAIL);
      $message = trim($_POST["inputMessage"]);
      $checkBoth = trim($_POST["inputCheckBoth"]);
      $checkDesign = trim($_POST["inputCheckDesign"]);
      $checkDev = trim($_POST["inputCheckDev"]);

      // Check that data was sent to the mailer.
     if ( empty($name) OR empty($message) OR !filter_var($email, FILTER_VALIDATE_EMAIL)) {
        // Set a 400 (bad request) response code and exit.
        //http_response_code(400);
        echo "Oops! There was a problem with your submission. Please complete the form and try again.";
        exit;
      }

      // Set the recipient email address.
      $recipient = "$myEmail";

      // Set the email subject.
      $subject = "New contact from $name";

      // Build the email content.
      $email_content = "Name: $name\n";
      $email_content .= "Email: $email\n\n";
      $email_content .= "Subject: New contact\n\n";
      $email_content .= "Message:\n$message\n\n";
      $email_content .= "Wants design: $checkDesign\n\n";
      $email_content .= "Wants dev: $checkDev\n\n";
      $email_content .= "Wants both: $checkBoth\n\n";

      // Build the email headers.
      $email_headers = "From: $name <$email>";

      // Send the email.
      if (mail($recipient, $subject, $email_content, $email_headers)) {
        // Set a 200 (okay) response code.
        //http_response_code(200);
        echo "Thank You! Your message has been sent.";
      } else {
        // Set a 500 (internal server error) response code.
        //http_response_code(500);
        echo "Oops! Something went wrong and we couldn\"t send your message.";
      }

    } else {
      // Not a POST request, set a 403 (forbidden) response code.
      //http_response_code(403);
      echo "There was a problem with your submission, please try again.";
    }
?>

Epilogue: Build a Contact Form with React, AJAX, PHP and reCaptcha

Congratulations! You’ve just finished this short tutorial and created your own contact form. I hope you liked it and learn something new, something you can use. As you can see, you don’t have to limit yourself to just one framework, technology or language. You can combine them as you want to get the job done. This is exactly what you did in this tutorial.

You created a contact form combining JavaScript in the form of React and AJAX with PHP. So, ignore those people saying that you have to choose one language or framework. The truth is that you don’t have to. You can choose any languages and frameworks or their combination that you like. What matters is that it will get the job done.

If you liked this article, please subscribe so you don't miss any future post.

If you'd like to support me and this blog, you can become a patron, or you can buy me a coffee 🙂

By Alex Devero

I'm Founder/CEO of DEVERO Corporation. Entrepreneur, designer, developer. My mission and MTP is to accelerate the development of humankind through technology.

Leave a Reply

This site uses Akismet to reduce spam. Learn how your comment data is processed.