arrow left
Back to Developer Education

Building a Webscript to Generate Hand-Signed PDFs using JavaScript Libraries

Building a Webscript to Generate Hand-Signed PDFs using JavaScript Libraries

In this tutorial, we will learn how to convert a webpage into a PDF document using open-source JavaScript libraries. Hand signatures are a good way to authenticate official documents. <!--more--> What if the parties involved are far from each other? We have to think of a way to enable them to sign the documents online.

Good for us, we can do that in our web applications without much effort using an open-source JavaScript library called Signature Pad.

We will build a simple invoice with an embedded section where one's signature will appear after signing. Later on, we will convert the webpage to a PDF document using another open-source library called jsPDF.

Our final output should look like this:

Generated pdf

Prerequisites

As a prerequisite, the reader must have the following:

  • An understanding of HTML and CSS.
  • Knowledge of JavaScript and Document Object Manipulation (DOM) manipulation.

Goals

We will learn to:

  • Create a signature feature for a web app.
  • Generate PDFs using the jsPDF library.
  • Convert the signature generated to an image.

An overview of the two libraries

Let's have a quick walk-through of the Signature Pad and jsPDF libraries.

Signature Pad

Signature Pad allows us to draw smooth curves on a canvas element. This capability makes it suitable to create a signing pad for the end-users.

Signature Pad provides features to undo, clear, change the curve colors, convert the signatures to different image formats like .png, .jpeg, .svg, and so on.

In our article, we will convert the signatures into .png images.

jsPDF

jsPDF is used to generate a PDF document from a web page. It provides options to set the font type, font size, line width, text color, etc.

However, we won't go deep into its uses. We will simply generate a PDF using it. I will leave the other functionalities for you to carry your research on them through the links at the end.

With that, let's first design a layout of our page.

Implementation

Page layout

In this tutorial, we will use the Bootstrap library to style the web pages.

The main bootstrap styling classes that we will use are:

  • col-md-* grid partitioning for a page. This styling breaks a page into 12 columns. For example, if you want a certain element to cover a third of the page width, you use the col-md-4 class.
  • d-flex- This allows us to make an element conform to the CSS's flexbox properties where we will use justify-content-* positioning to arrange items on a page. If we want the elements of a div to be in the center, we will use the d-flex justify-content-center class.

Let's briefly look at the components.

The <head> sets the character code to UTF-8, the title of the page, and the link to Bootstrap's styling.

<head>
  <meta charset="utf-8" />
  <title>Signature pad HTML2PDF</title>
  <link
    rel="stylesheet"
    href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/css/bootstrap.min.css"
    integrity="sha384-Gn5384xqQ1aoWXA+058RXPxPg6fy4IWvTNh0E263XmFcJlSAwiGgFAW/dAiS6JXm"
    crossorigin="anonymous"
  />
</head>

We head on to the <body> section where we have several <div> elements in which we will only look at the important ones.

We have the onselectstart attribute set to return false, which allows us to only make selections on our canvas.

  • The second <div> contains the area where the table and the signature appear.
<div id="toPrint" class="col-md-12">
  <table class="table">
    <thead class="thead-dark">
      <tr>
        <th scope="col"></th>
        <th scope="col">Item</th>
        <th scope="col">Price</th>
        <th scope="col">Quantity</th>
        <th scope="col">Price</th>
      </tr>
    </thead>
    <tbody>
      <tr>
        <th scope="row">1</th>
        <td>Toothbrush</td>
        <td>KES. 180</td>
        <td>3</td>
        <td>KES. 540</td>
      </tr>
      <tr>
        <th scope="row">2</th>
        <td>Toothpaste</td>
        <td>KES. 560</td>
        <td>1</td>
        <td>KES. 560</td>
      </tr>
      <tr>
        <th scope="row">3</th>
        <td>Cookies</td>
        <td>KES. 120</td>
        <td>2</td>
        <td>KES. 240</td>
      </tr>
      <tr>
        <th scope="row"></th>
        <td></td>
        <td></td>
        <th class="text-danger">GRAND TOTAL</th>
        <td>KES. 1340</td>
      </tr>
    </tbody>
  </table>
  <div class="col-md-12 d-flex justify-content-end">
    <div class="col-md-3 d-flex justify-content-center">
      <span class=" mb-4">Your Signature</span>
    </div>
  </div>
  <div class="col-md-12 d-flex justify-content-end">
    <div class="col-md-3 d-flex justify-content-center">
      <img
        id="signature"
        style="border: 1px solid black;"
        width="250px"
        height="100px"
      />
    </div>
  </div>
</div>
  • The next one div contains the canvas element where we allow the user to draw the signature.
<div class="col-md-12 d-flex justify-content-center ">
  <span class="col-md-3 mb-4">Sign here</span>
</div>
<div class="col-md-12 d-flex justify-content-center ">
  <canvas
    id="canvas"
    height="500"
    width="800"
    style="border: 1px solid black"
  ></canvas>
</div>
  • The last div contains four action buttons to process the signature (Save), clear the signature (Clear), undo it (Undo), and generate the PDF document (Get PDF), on click.
<div class="col-md-12 d-flex justify-content-around ">
  <button
    type="button"
    style="margin-top: 2%; margin-bottom: 3%"
    class="btn btn-dark"
    data-action="action-save"
  >
    Save
  </button>
  <button
    type="button"
    style="margin-top: 2%; margin-bottom: 3%"
    class="btn btn-dark"
    data-action="action-undo"
  >
    Undo
  </button>
  <button
    type="button"
    style="margin-top: 2%; margin-bottom: 3%"
    class="btn btn-dark"
    data-action="action-clear"
  >
    Clear
  </button>
  <button
    type="button"
    style="margin-top: 2%; margin-bottom: 3%"
    class="btn btn-dark"
    data-action="action-pdf"
  >
    Get PDF
  </button>
</div>

You can find the full HTML code snippet here.

Note the script and styling tags where we will reference the JavaScript and Bootstrap styling files.

In the next section, we will look at how we perform the actions after the button clicks using JavaScript.

JavaScript code

You can create a file and name it sign-handler.js or any other name as long as you update the HTML code script reference.

We will start by initializing variables for accessing the DOM elements.

var canvas = document.getElementById("canvas");
var saveSignBtn = document.querySelector("[data-action=action-save]");
var undoBtn = document.querySelector("[data-action=action-undo]");
var clearBtn = document.querySelector("[data-action=action-clear]");
var createPDFBtn = document.querySelector("[data-action=action-pdf]");
  • Next, we initialize our SignaturePad class while we pass in the background color as white and set the canvas with 2D context.
var ourPad = new SignaturePad(canvas, {
  backgroundColor: "rgb(255, 255, 255)",
});

canvas.getContext("2d");
  • After that, we create a function to create a blob. We use the blob to get an image path, where we set the image's src path.

You can learn more about the canvas toBlob() function in the official MDN documentation.

function processImage() {
  //creating a blob and displaying the image in the image element
  canvas.toBlob(function (blob) {
    var targetImg = document.querySelector("img"),
      url = URL.createObjectURL(blob);
    targetImg.src = url;
  });
}
  • The next three functions are straightforward. They are to add click event listeners to the three buttons.
    • saveSignBtn
    • clearBtn
    • undoBtn
  • For the undo button, we delete the last dot or curve drawn to the canvas using the pop() function and then update the canvas using the fromData() method.
saveSignBtn.addEventListener("click", function (event) {
  if (ourPad.isEmpty()) {
    alert("Please sign first.");
  } else {
    processImage();
  }
});

//clearing the signature drawing
clearBtn.addEventListener("click", function (event) {
  if (ourPad.isEmpty()) {
    alert("Please sign first.");
  } else {
    ourPad.clear();
  }
});

undoBtn.addEventListener("click", function (event) {
  var signMark = ourPad.toData();
  if (ourPad.isEmpty()) {
    alert("Please sign first.");
  } else {
    if (signMark) {
      signMark.pop(); // deletinging the last marked dot or drawn line
      ourPad.fromData(signMark);
    }
  }
});
  • Finally, we have the button click listener function to generate a PDF document.
//creating a pdf using the jsPDF library
createPDFBtn.addEventListener("click", function (event) {
  window.html2canvas = html2canvas;
  window.jsPDF = window.jspdf.jsPDF;
  var doc = new jsPDF("l", "pt", "a4");

  doc.html(document.getElementById("toPrint"), {
    callback: function (doc) {
      doc.save();
    },
    x: 10,
    y: 10,
  });
});

The above code does the following:

  • We initialize the html2canvas and jsPDF library.
  • In case you are wondering why we are initializing the html2canvas, the jsPDF library uses it to capture the page screenshots.
  • The jsPDF constructor is passed in the orientation (l for landscape), units of measurement (pt for points), and A4 sizing. We chose the landscape portrait for our page to fit in the PDF document.
  • The html() function is called where we pass in the element to be printed and an options object.
  • The object contains a save() callback function to save the PDF with the X and Y positioning of the document. There are a bunch of other options which you can add to this object.

You can view the full source code here.

Conclusion

To conclude, we looked at an overview of the two libraries, created an HTML layout, and looked at the JavaScript code.

You can further enhance this signature feature and use it for biometric authentication. The signature pad library is can be very useful in your React-Native apps, web frameworks such as Express, and even desktop apps.

Happy coding!

Further reading

There are a few other PDF generator libraries that you can have a look at. They include:


Peer Review Contributions by: Srishilesh P S

Published on: Mar 10, 2022
Updated on: Jul 15, 2024
CTA

Cloudzilla is FREE for React and Node.js projects

Deploy GitHub projects across every major cloud in under 3 minutes. No credit card required.
Get Started for Free