Skip to content

Commit

Permalink
Add example for HTML template rendering
Browse files Browse the repository at this point in the history
  • Loading branch information
shuhanmirza committed Jan 12, 2024
1 parent 543bf20 commit 2d2e1cf
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 3 deletions.
Binary file added Doc/example-html.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
67 changes: 64 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -31,12 +31,12 @@ See our example LaTex template in [`Doc/example.tex`](https://github.com/shuhanm
You can see that, the variables are written between two `%` s.
https://github.com/shuhanmirza/SpringBooTex/blob/49782f4cdd94a3e620f92739f7b37f381c3d77c0/Doc/example.tex#L12-L21

And the pdf has to embed an image named `universe.jpg`
And the pdf has to embed an image file named `universe.jpg`
https://github.com/shuhanmirza/SpringBooTex/blob/49782f4cdd94a3e620f92739f7b37f381c3d77c0/Doc/example.tex#L37-L38

Let's prepare the JSON payload,
Let's prepare the JSON payload.

Now, encode the latex template to base64 by running:
At first, encode the latex template to base64 by running:
```bash
cat example.tex | base64
```
Expand Down Expand Up @@ -79,6 +79,67 @@ The JSON payload should look like this,
And the generated PDF is
![example-latex.pdf](Doc/example-latex.png)

At first, SpringBooTex downloads all the files mentioned in the `fileUrlMap`. Then, it puts the string and list variables mentioned in `stringMap` and `listMap`. Finally, it compiles the PDF using `pdfLatex`.


### Example: Generating a PDF from HTML template

See our example HTML template in [`Doc/example.html`](https://github.com/shuhanmirza/SpringBooTex/blob/main/Doc/example.html)
You can see that, the thymeleaf templating has been used in the template.

https://github.com/shuhanmirza/SpringBooTex/blob/543bf20ccad01250303531e9491080804bc48aa6/Doc/example.html#L230-L241

Like in the earlier example, you have to encode the template to a base64 string and prepare JSON payload.
```JSON
{
"templateType": "HTML",
"templateSourceType": "BASE64",
"templateSource": "<!-- thanks https://htmlpdfapi.com/blog/free_html5_invoice_templates-->
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="utf-8" />
    <title>Example HTML</title>
    <style>
        @font-face {
            font-family: Junge;
            src: url(https://s3-eu-west-1.amazonaws.com/htmlpdfapi.production/free_html5_invoice_templates/example3/Junge-Regular.ttf);
        }

        .clearfix:after {
            content: "";
            display: table;
            clear: both;
        }

        a {
            color: #001028;
            text-decoration: none;
        }

        @page {
            size: A4 portrait;
        }

        body {
            font-family: Junge;
            position: relative;
            margin: 0 auto;
            color: #001028;
            background: #FFFFFF;
            font-size: 14px;
        }

        .arrow {
            margin-bottom: 4px;
        }

        .arrow.back {
            text-align: right;
        }

        .inner-arrow {
            padding-right: 10px;
            height: 30px;
            display: inline-block;
            background-color: rgb(233, 125, 49);
            text-align: center;

            line-height: 30px;
            vertical-align: middle;
        }

        .arrow.back .inner-arrow {
            background-color: rgb(233, 217, 49);
            padding-right: 0;
            padding-left: 10px;
        }

        .arrow:before,
        .arrow:after {
            content: '';
            display: inline-block;
            width: 0;
            height: 0;
            border: 15px solid transparent;
            vertical-align: middle;
        }

        .arrow:before {
            border-top-color: rgb(233, 125, 49);
            border-bottom-color: rgb(233, 125, 49);
            border-right-color: rgb(233, 125, 49);
        }

        .arrow.back:before {
            border-top-color: transparent;
            border-bottom-color: transparent;
            border-right-color: rgb(233, 217, 49);
            border-left-color: transparent;
        }

        .arrow:after {
            border-left-color: rgb(233, 125, 49);
        }

        .arrow.back:after {
            border-left-color: rgb(233, 217, 49);
            border-top-color: rgb(233, 217, 49);
            border-bottom-color: rgb(233, 217, 49);
            border-right-color: transparent;
        }

        .arrow span {
            display: inline-block;
            width: 80px;
            margin-right: 20px;
            text-align: right;
        }

        .arrow.back span {
            margin-right: 0;
            margin-left: 20px;
            text-align: left;
        }

        h1 {
            color: #5D6975;
            font-family: Junge;
            font-size: 2.4em;
            line-height: 1.4em;
            font-weight: normal;
            text-align: center;
            border-top: 1px solid #5D6975;
            border-bottom: 1px solid #5D6975;
            margin: 0 0 2em 0;
        }

        h1 small {
            font-size: 0.45em;
            line-height: 1.5em;
            float: left;
        }

        h1 small:last-child {
            float: right;
        }

        #project {
            float: left;
        }

        #company {
            float: right;
        }

        table {
            width: 100%;
            border-collapse: collapse;
            border-spacing: 0;
            margin-bottom: 30px;
        }

        table th,
        table td {
            text-align: center;
        }

        table th {
            padding: 5px 20px;
            color: #5D6975;
            border-bottom: 1px solid #C1CED9;
            white-space: nowrap;
            font-weight: normal;
        }

        table .service,
        table .desc {
            text-align: left;
        }

        table td {
            padding: 20px;
            text-align: right;
        }

        table td.service,
        table td.desc {
            vertical-align: top;
        }

        table td.unit,
        table td.qty,
        table td.total {
            font-size: 1.2em;
        }

        table td.sub {
            border-top: 1px solid #C1CED9;
        }

        table td.grand {
            border-top: 1px solid #5D6975;
        }

        table tr:nth-child(2n-1) td {
            background: #EEEEEE;
        }

        table tr:last-child td {
            background: #DDDDDD;
        }

        #details {
            margin-bottom: 30px;
        }

        footer {
            color: #5D6975;
            width: 100%;
            height: 30px;
            position: absolute;
            bottom: 0;
            border-top: 1px solid #C1CED9;
            padding: 8px 0;
            text-align: center;
        }
    </style>
</head>

<body>
    <main>
        <h1 class="clearfix"><small><span>DATE</span><br /><span th:text="${date}"></span></small> <span
                th:text="${title}"></span> <small><span>DUE
                    DATE</span><br /><span th:text="${dueDate}"></span></small></h1>
        <table>
            <thead>
                <tr>
                    <th class="service">SERVICE</th>
                    <th class="desc">DESCRIPTION</th>
                    <th>PRICE</th>
                    <th>QTY</th>
                    <th>TOTAL</th>
                </tr>
            </thead>
            <tbody>
                <tr th:each="service, serviceStat : ${serviceList}">
                    <td class="service" th:text="${serviceList.get(serviceStat.index)}"></td>
                    <td class="desc" th:text="${descriptionList.get(serviceStat.index)}"></td>
                    <td class="unit" th:text="${unitList.get(serviceStat.index)}"></td>
                    <td class="qty" th:text="${quantityList.get(serviceStat.index)}"></td>
                    <td class="total" th:text="${totalAmountList.get(serviceStat.index)}">$1</td>
                </tr>

                <tr>
                    <td colspan="4" class="sub">SUBTOTAL</td>
                    <td class="sub total" th:text="${subtotal}"></td>
                </tr>
                <tr>
                    <td colspan="4" th:text="${taxTitle}"></td>
                    <td class="total" th:text="${taxAmount}"></td>
                </tr>
                <tr>
                    <td colspan="4" class="grand total">GRAND TOTAL</td>
                    <td class="grand total" th:text="${grandTotal}"></td>
                </tr>
            </tbody>
        </table>
        <div id="details" class="clearfix">
            <div id="project">
                <div class="arrow">
                    <div class="inner-arrow"><span>PROJECT</span> Website development</div>
                </div>
                <div class="arrow">
                    <div class="inner-arrow"><span>CLIENT</span> John Doe</div>
                </div>
                <div class="arrow">
                    <div class="inner-arrow"><span>ADDRESS</span> 796 Silver Harbour, TX 79273, US</div>
                </div>
                <div class="arrow">
                    <div class="inner-arrow"><span>EMAIL</span> <a href="mailto:john@example.com">john@example.com</a>
                    </div>
                </div>
            </div>
            <div id="company">
                <div class="arrow back">
                    <div class="inner-arrow">Company Name <span>COMPANY</span></div>
                </div>
                <div class="arrow back">
                    <div class="inner-arrow">455 Foggy Heights, AZ 85004, US <span>ADDRESS</span></div>
                </div>
                <div class="arrow back">
                    <div class="inner-arrow">(602) 519-0450 <span>PHONE</span></div>
                </div>
                <div class="arrow back">
                    <div class="inner-arrow"><a href="mailto:company@example.com">company@example.com</a>
                        <span>EMAIL</span></div>
                </div>
            </div>
        </div>
        <div id="notices">
            <div>NOTICE:</div>
            <div class="notice">A finance charge of 1.5% will be made on unpaid balances after 30 days.</div>
        </div>
    </main>
    <footer>
        Invoice was created on a computer and is valid without the signature and seal.
    </footer>
</body>

</html>",
"stringMap": {
"title": "Invoice #100",
"date": "January 1st, 2024",
"dueDate": "January 31st, 2024",
"subtotal": "$5,040",
"taxTitle": "TAX 15%",
"taxAmount": "$756",
"grandTotal": "$5,796"
},
"listMap": {
"serviceList": [
"Design",
"Development",
"SEO"
],
"descriptionList": [
"Creating a recognizable design solution based on the company's existing visual identity",
"Developing a Content Management System-based Website",
"Optimize the site for search engines (SEO)"
],
"unitList": [
"$40.00",
"$40.00",
"$40.00"
],
"quantityList": [
"26",
"80",
"20"
],
"totalAmountList": [
"$1,040.00",
"$3,200.00",
"$800.00"
]
},
"responseType": "JSON"
}
```

And the generated PDF is
![example-html.pdf](Doc/example-html.png)

At first, SpringBooTex prepares the html using Thymeleaf template engine. Then it leverages `openhtmltopdf` to render PDF from html. If you do not want to provide fonts through HTML, you can provide it by adding font urls in the `fileUrlMap`. SpringBootex will download and add the fonts during PDF rendering.


## Contributing
Please!
Expand Down

0 comments on commit 2d2e1cf

Please sign in to comment.