PDF asset
  • 18 Apr 2025
  • 3 Minutes to read
  • Contributors
  • Dark
    Light

PDF asset

  • Dark
    Light

Article summary

A PDF asset can be used to build PDFs based on HTML Liquid templates. Liquid is an open-source template language created by Shopify.

How it works

You define the PDF structure with the template code, which is HTML + Liquid.
Liquid makes it possible to access data from the context. The context is represented by a JSON object and can contain data from a configuration.

The template code and the context together result in the final HTML.
The final HTML will be converted to a PDF by the Combeenation service (Puppeteer is used under the hood).

image.png

Images & fonts in header & footer

Resource URLs (e.g. images, fonts, ...) in header & footer won't be resolved by puppeteer, they need to be embedded via base64 encoding. For more information, see data_uri filter.

Encoding

Textual data which is injected via the context is HTML encoded by default. This means if you want to inject HTML or SVG into the PDF asset, then you have to disable this encoding.
For this, simply use the raw filter (e.g. {{ svg | raw }} or use a {% raw %} ... {% endraw %} section as described here.

How to create a PDF asset

  1. Got to the asset manager

  2. Click on the (+) button to create a new PDF asset
    new-pdf-asset.png

  3. Select a name, folder, template and click on Create asset.

Template language: Liquid

If you want to inject flexible data into your HTML sections, then simply use the context in combination with several Liquid statements. The most common ones are described in this documentation, for more details visit the official liquid documentation.

Objects

For example, if you define the following Context with a company object and a name property, then you can inject this data into your HTML section as described here.
image.png

Context

{
  "company": {
    "name": "Acme Solutions Ltd."
  }
}

Input

{{ company.name }}

Output

Acme Solutions Ltd.

For-loops

In addition, it's possible to iterate over a list of objects using for-loops.

Context

{
  "lineItems": [
    {
      "position": 1,
      "description": "Service A description",
      "price": "40,00 €"
    },
    {
      "position": 2,
      "description": "Service B description",
      "price": "25,00 €"
    }
  ]
}

Input

<table class="main-table">
  <thead>
    <tr>
      <th class="pos-column">Pos.</th>
      <th>Description</th>
      <th class="price-column">Price</th>
    </tr>
  </thead>
  <tbody>
    {% for item in lineItems %}
    <tr>
      <td>{{ item.position }}</td>
      <td>{{ item.description }}</td>
      <td class="price-column">{{ item.price }}</td>
    </tr>
    {% endfor %}
  </tbody>
</table>

Output

Pos.DescriptionPrice
1Service A description40,00 €
2Service B description25,00 €

Filters

We extended the liquid template language with custom filters which can be used in combination with Assets. The following filters are available:

  • url ... to retrieve the URL of an image asset
  • data_uri to retrieve the base64 encoded data of an image asset, to directly embed an image into a header or footer.

Asset access

Two recommended ways to access the image asset in your template code:

  • Direct access (blue arrow): Use the full asset path as a string (e.g. folderName.assetName.) before the above mentioned filter
  • Context-based injection (green arrow): Pass the full asset path via the context. This is preferred when the asset is dynamic and selected during the configuration process

image.png

If the given asset is not valid or is missing, then a corresponding asset error is shown.

Asset
There must be one asset named logo.

Context

{
  "company": {
    "logo": "MyFolder.Logo"
  }
}

Input

<img src="{{ company.logo | url }}" style="height: 2cm" />
<img src="{{ company.logo | data_uri }}" style="height: 2cm" />

HTML Output

<img src="https://cbnpeuwstassets.blob.core.windows.net/public/COMBEENATION/0Dxez2x2M09.png" style="height: 2cm" />
<img src="" style="height: 2cm" />

Output

Page numbers and similar

Only in header & footer

These only work in the header & footer! They have no effect inside the body.

Specific css classes can be used to inject auto-generated values into the PDF.
To do so, simply create a html-tag with one of the css classes stated below.

Possible values (css classes)

  • date - formatted print date
  • title - document title
  • url - document location
  • pageNumber - current page number
  • totalPages - total pages in the document

Example usage

<span class="pageNumber"></span> of <span  class="totalPages"></span> 

See also Puppeteer PDFOptions.headerTemplate

How to use a pdf asset

A PDF asset can finally be used to create a PDF based on any data during a configuration. The required data can be dynamically passed via the Context object. As a result, it's possible to create a PDF using Custom Code.

Basically, the PDF is created via the PDF asset and a dynamically passed Context object. This generated PDF is saved as a "configuration file" and be used as described here.

Example - Create PDF from asset

const context = {
    "company": {
        "logo": "MyFolder.MyFinalLogo"
    }
};
const resultUrl = await CfgnFiles.createPdfFromAsset(
    'MyBundle',
    'MyFolder.MyPdfAsset',
    context,
    'MyCfgnFileName',
    'MyCfgnFileDisplayName'}
);
console.log('url is: ', resultUrl);

Was this article helpful?

What's Next