How to easily generate word documents (.docx) in javascript using Docxtemplater

How to easily generate word documents (.docx) in javascript using Docxtemplater

Learn how to generate word documents in javascript by building a simple resume generator

In my latest project, I learnt how to generate a word document from a template using a javascript package called Docxtemplater. It's easier to use and setup compared to the docx package, plus once setup, your clients can edit the templates to their desired style without having to touch the software's code

In this article, i will show you how to use Docxtemplater to generate a simple resume document from a word template. But first...

docxtemplater homepage.png Docxtemplater is an open sourced javascript package that can generate MS Office documents (.docx, .ppt, .xlsx files) from a preexisting template.

The package can create a new document out of a word template which contain tags (like {}) that can be replaced with data in javascript. That means if my document contains {first_name}, it resolves the value of first_name and replaces the tag with that value

Like in their demo below

docxtemplater demo.png

This package abstracts above the other popular method of generating word documents which is the docx package, but it is harder to create and maintain documents as you will be building the needed document programmatically.

And that's where Docxtemplater shines

It goes through a word template and makes a copy out of it while filling it with data. That way you can for instance generate an entire employee database as documents by using a template without creating them manually

Alright, let's build our project

We'll kick off by cloning a starter repo i created for you to get up and running quickly. Clone the repo here

You will also need to download and install nodejs if you haven't already. Download it here

After downloading the starter repo locally, let's install the dependencies needed to build our application. Enter this command into the terminal

npm install

Once your dependencies has installed completely, fire up the server with this command

npm run dev

Your browser should open on localhost:8080

I've provided a word template in the starter repo we will use for this project. Here's what the template looks like (Notice the tags? Docxtemplater replaces those with data)

how template looks.png

Since this is a simple tutorial, i will limit our interface to a simple button that triggers the document generation process.

In the src/index.html file, let's create a button that will generate our document when clicked

<!-- src/index.html -->
<body>
    <button>Generate Resume</button>
</body>

Make the button pretty by styling it in src/index.css

/* src/index.css */
body {
    margin: 0;
    padding: 0;
    height: 100vh;
    display: flex;
    align-items: center;
    justify-content: center;
}
button {
    padding: 8px 12px;
    background-color: rgb(0, 102, 255);
    border: none;
    border-radius: 5px;
    color: white;
    font-size: 16px;
}
button:hover {
    background-color: rgb(0, 80, 199);
    cursor: pointer;
}

Now that we have our interface down, let's build the core part of the application. I've broken this section into 4 steps so you can follow along:

In src/index.js, import:

  • Docxtemplater for document generation
  • Pizzip to create a zip file from our template
  • saveAs function from file-saverto prompt the user to save the generated document
  • Our template's path as template from src/simple.docx

If you're wondering how we can import a docx file into a js application here's the gist. I setup webpack in the starter repo which is a bundling tool for frontend development. Webpack processes the template simple.docx and resolves its path to template

// in src/index.js
import Docxtemplater from 'docxtemplater';
import PizZip from 'pizzip';
import { saveAs } from 'file-saver';
import template from './simple.docx';

Now let's define our resume data as an object. This will be the data that will be used to generate our document from the template.

You can even change the values of the keys in the object, just don't change the keys themselves

// in src/index.js
let resume = {
    name: {
        first: 'John',
        last: 'Doe'
    },
    personalSummary: "Hi i'm John, a software engineer passionate about building well...software",
    jobTitle: 'Software Engineer',
    contact: {
        address: "Lagos, Nigeria",
        phone: '08123456789',
        email: 'johndoe@gmail.com'
    },
    meta_details: {
        dateOfBirth: '24th June, 1995',
        stateOfOrigin: 'Enugu',
        lga: 'Oji-River',
        gender: 'Male',
        maritalStatus: 'Single',
        religion: 'Christian'
    },
    workExperience: [
        {
            nameOfOrg: 'Acme Inc.',
            position: 'Software Developer',
            from: 'July, 2022',
            to: 'Present'
        }
    ],
    education: [
        {
            name: 'Creation Academy',
            location: 'Earth',
            type: 'Primary',
            qualificationObtained: 'Elementary School Certificate',
            started: '18th Feb, 2017',
            finished: '6th July, 2022'
        }
    ],
    referees: [
        {
            name: "Big man",
            nameOfOrg: 'Big man Inc',
            position: 'Big man position',
            contact: 'bigman@verybig.com'
        }
    ]
}

Let's add a click event listener to our button that calls a generating function we will define soon. We'll pass the resume object and template to generateDocument()

// in src/index.js
let trigger = document.querySelector('button');
trigger.addEventListener('click', (e) => {
    e.preventDefault();

    return generateDocument(resume, template);
});

First off, let's define a function generateDocument() that takes 2 arguments; a resume object and templatePath which is the path to our template file.

In that function, load the template as an arrayBuffer using fetch() and create a zip file from it using Pizzip

The function is asynchronous so we need to use async...await to define the function

// in src/index.js
async function generateDocument(resume, templatePath) {
    try {
        let response = await fetch(templatePath);
        let data = await response.arrayBuffer();
        let zip = PizZip(data);
    } catch (error) {
    console.log('Error: ' + error);
    }
}

Now let's create a Docxtemplater instance with the zip file and pass our resume object to it's render() method

// inside the try-catch block
let templateDoc = new Docxtemplater(zip, {
    paragraphLoop: true,
    linebreaks: true
})

templateDoc.render(resume);

Finally generate our word document filled up with the resume object and prompt the user to save the document with a unique name using saveAs

// inside the try-catch block
let generatedDoc = templateDoc.getZip().generate({
    type: "blob",
    mimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
    compression: "DEFLATE"
})

saveAs(generatedDoc, `${resume.name.first}'s resume.docx`);

Your src/index.js file should now look like this

// in src/index.js
import './index.css';

import Docxtemplater from 'docxtemplater';
import PizZip from 'pizzip';
import { saveAs } from 'file-saver';

import template from './simple.docx';

let resume = {
    name: {
        first: 'John',
        last: 'Doe'
    },
    personalSummary: "Hi i'm John, a software engineer passionate about building well...software",
    jobTitle: 'Software Engineer',
    contact: {
        address: "Lagos, Nigeria",
        phone: '08123456789',
        email: 'johndoe@gmail.com'
    },
    meta_details: {
        dateOfBirth: '24th June, 1995',
        stateOfOrigin: 'Enugu',
        lga: 'Oji-River',
        gender: 'Male',
        maritalStatus: 'Single',
        religion: 'Christian'
    },
    workExperience: [
        {
            nameOfOrg: 'Acme Inc.',
            position: 'Software Developer',
            from: 'July, 2022',
            to: 'Present'
        }
    ],
    education: [
        {
            name: 'Creation Academy',
            location: 'Earth',
            type: 'Primary',
            qualificationObtained: 'Elementary School Certificate',
            started: '18th Feb, 2017',
            finished: '6th July, 2022'
        }
    ],
    referees: [
        {
            name: "Big man",
            nameOfOrg: 'Big man Inc',
            position: 'Big man position',
            contact: 'bigman@verybig.com'
        }
    ]

}

let trigger = document.querySelector('button');

trigger.addEventListener('click', (e) => {
    e.preventDefault();

    return generateDocument(resume, template);
});

async function generateDocument(resume, templatePath) {
    // load the document template into docxtemplater
    try {
        let response = await fetch(templatePath);
        let data = await response.arrayBuffer();

        let zip = PizZip(data);

        let templateDoc = new Docxtemplater(zip, {
            paragraphLoop: true,
            linebreaks: true
        })

        templateDoc.render(resume);

        let generatedDoc = templateDoc.getZip().generate({
            type: "blob",
            mimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
            compression: "DEFLATE"
        })

        saveAs(generatedDoc, `${resume.name.first}'s resume.docx`);
    } catch (error) {
        console.log('Error: ' + error);
    }
}

Now hit the "Generate Resume" button in the browser and you should see a save as dialog box prompting you to save the document.

Screenshot 2022-09-15 122450.png If you have any issues, reference what we did above or comment below

And that's it, you've learnt how to generate word documents in the browser. You can integrate docxtemplater into your projects to generate stuff like invoices, office memos and more.

You can even take it up a notch and build a usable resume generator saas like i did or some other document generator (an invoicing application for instance). Really it's up to you

Thanks for reading.

Jeffrey

You might be wondering why i didn't convert our document into a pdf and the honest answer is i tried but i kept having issues setting that up.

So if you have any experience generating PDFs on the browser i would love to pick your brain so i can polish up the actual project that inspired this article.

You can leave a comment below and i'll reach out. Thanks