How to convert HTML(CSS) To PDF Using JavaScript

1 comment
JavaScript was a language used  for basic things like form validations. Now the javascript can do so many thing which was only possible by server side technologies such as php and ASP.net.
With New HTML5 API’s Canvas, svg, video, audio and etc… we can do things lot easier now than before. Ok now why am I telling all this now. Because in this tutorial we going to see how to convert an html page to a pdf document using JavaScript.

To do it I’m going to use https://github.com/MrRio/jsPDF andhttps://github.com/niklasvh/html2canvas
We will html2canvas to take a screenshot (It will parse the DOM, Calculate the styles applied). Html2canvas returns the canvas object. Once we have the canvas object we will use it to create image using build in toDataURL() function. toDataURL() function will return a base64 encoded image as string. Finally we will use the jsPDF to create PDF document.
Enough explanation let’s dive in to the actual code now.
First of let’s create html from which we want to generate our pdf document. For better understanding I would like to create html form with not less than 10 controls in it. To create UI we will use semantic UI.

index.html(boilerplate)

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>HTML to PDF - techumber</title>
  6. <link rel="stylesheet" type="text/css" href="//cdnjs.cloudflare.com/ajax/libs/semantic-ui/1.12.0/semantic.min.css">
  7. </head>
  8. <body>
  9. <div class="ui page grid">
  10. <div class="wide column">
  11. <h1 class="ui header aligned center">HTML to PDF</h1>
  12. <div class="ui divider hidden"></div>
  13. </div>
  14. </div>
  15. <!-- scripts -->
  16. <script src="//ajax.googleapis.com/ajax/libs/jquery/1.11.0/jquery.min.js"></script>
  17. <script type="text/javascript" src="//cdn.rawgit.com/MrRio/jsPDF/master/dist/jspdf.min.js"></script> 
  1. <script type="text/javascript" src="//cdn.rawgit.com/niklasvh/html2canvas/0.5.0-alpha2/dist/html2canvas.min.js"></script>
  2. <script type="text/javascript" src="app.js"></script>
  3. </body>
  4. </html>
Added semantic ui cdn link for our styles. html2canvas and jsPdf added thorught rawgit cdn.

index.html

  1. <div class="ui segment">
  2. <div class="ui button aligned center teal" id="create_pdf">Create PDF</div>
  3. <div class="ui divider"></div>
  4. <form class="ui form">
  5. <h4 class="ui dividing header">Personal Information</h4>
  6. <div class="two fields">
  7. <div class="field">
  8. <label for="">First Name</label>
  9. <input type="text" name="first-name" placeholder="First Name">
  10. </div>
  11. <div class="field">
  12. <label for="">Last Name</label>
  13. <input type="text" name="last-name" placeholder="First Name">
  14. </div>
  15. </div>
  16. <div class="field">
  17. <label>Biography</label>
  18. <textarea></textarea>
  19. </div>
  20. <h4 class="ui dividing header">Account Info</h4>
  21. <div class="two fields">
  22. <div class="required field">
  23. <label>Username</label>
  24. <div class="ui icon input">
  25. <input type="text" placeholder="Username">
  26. <i class="user icon"></i>
  27. </div>
  28. </div>
  29. <div class="required field">
  30. <label>Password</label>
  31. <div class="ui icon input">
  32. <input type="password">
  33. <i class="lock icon"></i>
  34. </div>
  35. </div>
  36. </div>
  37. <h4 class="ui top attached header">Import Settings</h4>
  38. <div class="ui bottom attached segment">
  39. <div class="grouped fields">
  40. <label for="alone">Would you like us to import your current settings?</label>
  41. <div class="field">
  42. <div class="ui checkbox">
  43. <input type="radio" checked="" name="import">
  44. <label>Yes</label>
  45. </div>
  46. </div>
  47. <div class="field">
  48. <div class="ui checkbox">
  49. <input type="radio" name="import">
  50. <label>No</label>
  51. </div>
  52. </div>
  53. </div>
  54. </div>
  55. <h4 class="ui dividing header">Settings</h4>
  56. <h5 class="ui header">Privacy</h5>
  57. <div class="field">
  58. <div class="ui checkbox">
  59. <input type="radio" name="privacy">
  60. <label>Allow <b>anyone</b> to see my account</label>
  61. </div>
  62. </div>
  63. <div class="field">
  64. <div class="ui checkbox">
  65. <input type="radio" name="privacy">
  66. <label>Allow <b>only friends</b> to see my account</label>
  67. </div>
  68. </div>
  69. <h5 class="ui header">Newsletter Subscriptions</h5>
  70. <div class="field">
  71. <div class="ui checkbox">
  72. <input type="checkbox" name="top-posts">
  73. <label>Top Posts This Week</label>
  74. </div>
  75. </div>
  76. <div class="field">
  77. <div class="ui checkbox">
  78. <input type="checkbox" name="hot-deals">
  79. <label>Hot Deals</label>
  80. </div>
  81. </div>
  82. <div class="ui hidden divider"></div>
  83. <div class="field">
  84. <div class="ui checkbox">
  85. <input type="checkbox" name="hot-deals">
  86. <label>I agree to the <a href="#">Terms of Service</a>.</label>
  87. </div>
  88. </div>
  89. <div class="ui error message">
  90. <div class="header">We noticed some issues</div>
  91. </div>
  92. <div class="ui submit button blue">Register</div>
  93. </form>
  94. </div>
I created a simple html form using semantic ui.

app.js

  1. (function(){
  2. var
  3. form = $('.form'),
  4. cache_width = form.width(),
  5. a4 =[ 595.28, 841.89]; // for a4 size paper width and height
  6.  
  7. $('#create_pdf').on('click',function(){
  8. $('body').scrollTop(0);
  9. createPDF();
  10. });
  11. //create pdf
  12. function createPDF(){
  13. getCanvas().then(function(canvas){
  14. var
  15. img = canvas.toDataURL("image/png"),
  16. doc = new jsPDF({
  17. unit:'px',
  18. format:'a4'
  19. });
  20. doc.addImage(img, 'JPEG', 20, 20);
  21. doc.save('techumber-html-to-pdf.pdf');
  22. form.width(cache_width);
  23. });
  24. }
  25.  
  26. // create canvas object
  27. function getCanvas(){
  28. form.width((a4[0]*1.33333) -80).css('max-width','none');
  29. return html2canvas(form,{
  30. imageTimeout:2000,
  31. removeContainer:true
  32. });
  33. }
  34.  
  35. }());
Now, this is important. The app.js will contain main JavaScript code that we going to use for our app to work. getCanvas function will return promised object later we can use it in other places. In create PDF we converting the canvas to image and adding that image to jsPDF function addImage.

Here, you need to have few tricks. First jsPDF support many document formats. Since a4 is most popular we will use the same there. We are changing the form width to a4 size default width so that we can get exact screenshot otherwise the image will overflow when we add it to PDF document.

We always have to make sure to our page scroll in top position otherwise we won't get exact screenshot. To do so we are having $('body').scrollTop(0) function.