Skip to content

Demo202509 PDF Printing

Summary of Requirements

The Requirement is included in the Create and Print Kanban Card Use Case.

Printing needs to produce high-quality layouts of information that can be printed and used in industrial environments. Previous experience
with printing based on HTML and CSS formatting has shown that it is very hard and requires high technical expertise to achieve these results.

In the long term, the intent is to allow users to define their own layouts and templates to print information, which makes PDF an attractive choice
given the availability of tools and editors.

Printing for the Demo is confined to printing information available in a Kanban Card, including the details of the Item that is associated with it.

There are three initial layouts needed:

  1. Card Layout to be used in Kanban signalling.
  2. Label Layout to be used for location and bin labels in inventory.
  3. Breadcrumb Layout to be used for smaller spaces (e.g., shelf edges)

QR Lookup URL

The format of the URL for the QR Code to be printed in the PDF rendering was introduced in the Format of URL in QR ticket and should follow the following format:

https://{app-hostname}/kanban/cards/{eId}?view=card&src=qr

Where:

  • app-hostname is the hostname of the Arda Front-End Application. See below for known hostnames.
  • eId is the entity id (eId field) of the kanban card.

Known Hostnames

Known hostnames follow the pattern, for different development “purposes”:

Purpose Hostname
dev dev.app.arda.cards
stage stage.app.arda.cards
prod live.app.arda.cards

Internal Hostnames

Other hostnames may be used for special tests but it is discouraged as these may change at any time.

Purpose Hostname
dev dev.alpha002.app.arda.cards
stage stage.alpha002.app.arda.cards
prod prod.alpha001.app.arda.cards

Additional considerations

It is desirable to be able to reuse the layouts and designs already available in the context of the Coda product.

Available Technology

Arda already uses Documint for PDF generation for its Coda based product. Document generation is done through
an intermediate API and Service Layer. The API Endpoint is documented at: PDF Rendering API

Options

For the Demo202509 release there are five options available for PDF printing:

  1. Implement an HTML/CSS based solution in the front end.
  2. Use the Documint service directly:
    1. From the Front End or the BFF.
    2. From the Kanban Card Module in the backend.
  3. Use the current API Endpoint and Service
    1. From the Front End or BFF.
    2. From the Kanban Card Module in the backend.

The first option, given the long term direction of this feature will not be considered further.

Existing Templates and Layouts

Customer Service (Matt Hager) has provided these three Documint Templates for the three required layouts. To access them, use the
credentials available in 1Password under Documint with the user systems-sandbox@arda.cards.

General

These layouts can be used to print multiple cards and are intended to use a Letter paper size, optionally with pre-cut divisions (e.g., Avery labels)

Their data inputs have a structure in rows and columns.

Although not restricted by the schema, the maximum number of column elements per row is 2 to keep the overall grid reasonable to print
on Letter format.

Payload Fields

Although Layouts differ in the contents of their payloads, their field names are common and can be mapped consistently to the information in the Kanban Card Details:

Schema Name Description Source: From KanbanCardDetails
characters A layout control field with values: \({1, 2, 3}\) See below
item_name Name of the Associated Item item.name
notes Notes associated with the Kanban Card notes
print_sku Internal SKU associated with the Item item.internalSku
qr_code URL, to be displayed QR Code format to the Card View Page https://hostname/card/${lookupUrlId}
thin_line A layout control field, URL to a thin line image Static URL (may depend on item.color)
word_minimum An Image sent to Documint for rendering indexed by item.color
text_size A layout control field See below
min_qnty The minimum quantity to keep for an item item.minQty (to be added)
word_location An Image sent to Documint for rendering Indexed by item.color
location The location assigned to the Item item.locator.location
word_order An Image sent to Documint for rendering Indexed by item.color
order_qnty The recommended order quantity for the default Supplier item.defaultSupplier.orderQuantity.amount
order_mechanism The method to order from the default Supplier item.defaultSupplier.orderMethod
word_supplier An Image sent to Documint for rendering Indexed by item.color
supplier The name of the default supplier item.defaultSupply
product_image A Base64 encoded image of the product (GET item.imageUrl).base64()
color_bar A URL to a color bar image based on the item color Static URL
logo URL or Base64 image? Static URL for now. Future: tenant configuration.
thin_light_rectangle URL of an image to send to Documint Indexed by item.color (TBC)
light_rectangle URL of an image to send to Documint Indexed by item.color (TBC)
Missing Fields in current implementation
  • min_qnty: This was omitted in anticipation of multi-facility support. It will be added back for the demo.
  • logo: This should be part of the tenant configuration, which is not in scope for the demo. It could be hardcoded for the demo.
Special Calculations
  • defaultSupplier
when(item.defaultSupply) {
 item.primarySupply.supplier -> item.primarySupply
 item.secondarySupply.supplier -> item.secondarySupply
 else -> Error("Item ${item.name} `defaultSupply` is not properly configured: ${item.defaultSupply}")
}
  • characters calculation:
    when (item.name.length) {
      in 0..18  -> "1"
      in 19..30 -> "2"
      else      -> "3"
    }
    
  • text_size calculation:

// "text_size" key for the JSON-like object
when(val maxLength = list(
  item.default_supplier.orderQuantityUnits.length(),
  item.minQty.unit.length(),
  item.locator.location.length(),
).max()) {
  in 0..22 -> "1"
  else     -> "2"
}

From Mat Hager, to be removed once clarified:

// Build a list of candidate fields whose length we'll compare
list(
  // Units used for the order quantity (e.g., "pcs")
  thisRow.[Order Units],
  // Units used for the minimum quantity
  thisRow.[Min. Units],
  // Use the normalized Location Name if thisRow.Location matches a known Location
  If(
    Locations.Name.ToText().Contains(thisRow.Location),
    thisRow.Location.Name,
    // Otherwise use whatever is in the row's Location field
    thisRow.Location
  ),
  // Supplier name
  thisRow.Supplier
)
  // Pick the longest entry from that list
  .MaxBy(
    CurrentValue.Length()
  )
  // Measure the character count of that longest entry
  .Length()
  // Name it "length" just for the next step
  .WithName(
    length,
    // If over 22 chars, return 2 (needs more space); else 1
    switchif(length > 22, 2, 1)
  )

Special (Image) fields

This is to be completed.

Format is either Image URL or some string based encoding of the image.
Source is the logic to find the right image to send.

The following fields are URLs to images and determined based on lookup tables in colors.json and color-options.json from
the Item Color.

  • thin_line
  • word_minimum
  • word_location
  • word_order
  • word_supplier
  • color_bar
  • thin_light_rectangle
  • light_rectangle

product_image is an image URL from the Item.imageUrl field.

logo should be the image URL to the Company Logo. For now, it is blank.

Schema

{
  "type":"object",
  "properties":{
    "row":{
      "type":"array",
      "items":{
        "type":"object",
        "properties":{
          "column":{
            "type":"array",
            "items":{
              "type":"object",
              "properties":{
                [Payload Specific]
              }
            }
          }
        }
      }
    }
  }
}

Example.

{
  "row": [
    {
      "column": [
        {
          [payload_1_1]
        },
        {
          [payload_1_2]
        }
      ]
    },
    {
      "column": [
        {
          [payload_2_1]
        },
        {
          [payload_2_2]
        }
      ]
    }
  ]
}

Kanban Card Layout

Documint Template

Card layout preview

Payload Fields

Documint Schema

{
  "type":"object",
  "properties":{
    "characters":{ "type":"string" },
    "item_name":{ "type":"string" },
    "notes":{ "type":"string" },
    "print_sku":{ "type":"string" },
    "qr_code":{ "type":"string" },
    "thin_line":{ "type":"string" },
    "word_minimum":{ "type":"string" },
    "text_size":{ "type":"string" },
    "min_qnty":{ "type":"string" },
    "word_location":{ "type":"string" },
    "location":{ "type":"string" },
    "word_order":{ "type":"string" },
    "order_qnty":{ "type":"string" },
    "order_mechanism":{ "type":"string" },
    "word_supplier":{ "type":"string" },
    "supplier":{ "type":"string" },
    "product_image":{ "type":"string" },
    "color_bar":{ "type":"string" },
    "logo":{ "type":"string" }
  }
}

In addition to the general payload fields, the following fields are specific to the Kanban card layout:

{
  [...]
  "today": { "type": "string" },
  "item_name": { "type": "string" }
  [...]
}

Example

{
  "characters": "3",
  "item_name": "Sample Item",
  "notes": "Sample notes",
  "print_sku": "Sample SKU",
  "qr_code": "Sample QR Code",
  "thin_line": "Sample Thin Line",
  "word_minimum": "Sample Word Minimum",
  "text_size": "Sample Text Size",
  "min_qnty": "Sample Min Quantity",
  "word_location": "Sample Word Location",
  "location": "Sample Location",
  "word_order": "Sample Word Order",
  "order_qnty": "Sample Order Quantity",
  "order_mechanism": "Sample Order Mechanism",
  "word_supplier": "Sample Word Supplier",
  "supplier": "Sample Supplier",
  "product_image": "Sample Product Image",
  "color_bar": "Sample Color Bar",
  "logo": "Sample Logo"
}

Label Layout

Documint Template

Label layout preview

Payload Fields

Documint Schema

{
  "type":"object",
  "properties":{
    "item_name":{ "type":"string" },
    "characters":{ "type":"string" },
    "notes":{ "type":"string" },
    "thin_line":{ "type":"string" },
    "print_sku":{ "type":"string" },
    "qr_code":{ "type":"string" },
    "product_image":{ "type":"string" },
    "color_bar":{ "type":"string" },
    "logo":{ "type":"string" }
  }
}

Example

{
  "item_name": "Sample Item",
  "characters": "Sample Characters",
  "notes": "Sample Notes",
  "thin_line": "https://codaio.imgix.net/docs/y9-aNYpZAA/blobs/bl-jV4BZylBKS/6dbbf1bc3ccc7cd71877cab82cd25471522f28fa6835a6103a0d811843418e916403e9666eb2ac5036f0f1db5dec8727455a2a0047b0c464971a9a0e46095ea2ad317412e9be5e1524195c5ae2d571f5d061dbefb06234fc80ac30efcde49852ec87079d",
  "print_sku": "Sample Print SKU",
  "qr_code": "https://link.arda.cards/lichen/items/1",
  "product_image": "",
  "color_bar": "https://codaio.imgix.net/docs/y9-aNYpZAA/blobs/bl-0u4VcOqeEZ/38e2ec507460280db05be2693d4c5d6da7b425fd3f07692c8b900d9a4e2d9b233f069bec45bc0433ab9732f2c37b5bdb800f4ca8648b792121bc2d700d3608315df1bd5ce86638b8a1fb7e66cbc09a02beb22edf2aa96243ddaaf440cd842852fc81995d?txt64=TWFjaGluaW5nLCAzRCBQcmludGluZw%3D%3D&txt-font=sans-serif&txt-size=46&txt-color=ff8c00&txt-line=1&txt-line-color=ff8c00&txt-shad=1&txt-pad=1&txt-lig=0&txt-align=center%2Cmiddle&txt-width=600&txt-fit=max",
  "logo": "<...>"
}

Documint Template

Breadcrumb layout preview

{
  "type":"object",
  "properties":{
    "item_name":{
      "type":"string"
    },
    "characters":{
      "type":"string"
    },
    "notes":{
      "type":"string"
    },
    "thin_light_rectangle":{
      "type":"string"
    },
    "print_sku":{
      "type":"string"
    },
    "qr_code":{
      "type":"string"
    },
    "logo":{
      "type":"string"
    },
    "product_image":{
      "type":"string"
    },
    "light_rectangle":{
      "type":"string"
    }
  }
}
{
  "item_name": "Sample Item",
  "characters": "Sample Characters",
  "notes": "Sample Notes",
  "thin_light_rectangle": "Sample Thin Light Rectangle",
  "print_sku": "Sample Print SKU",
  "qr_code": "Sample QR Code",
  "logo": "Sample Logo",
  "product_image": "Sample Product Image",
  "light_rectangle": "Sample Light Rectangle"
}

Design Proposal

Given the complex logic and need to retrieve image files, a dedicated module is recommended. This module, initially in the operations component, will work with the Kanban Card Module.

  1. Shared Capabilities with the Kanban Card Module (boundary to be decided)
    1. Selection of Images and Media Files based on Indexing values.
  2. new Pdf-Render Module Capabilities
    1. Storage and Lookup of Templates for different Use Cases and formats
    2. Access to Documint, including formatting of requests, error handling, etc…
  3. Additional Capabilities required in the Item and Kanban Modules
    1. Computation of color based on other item attributes
    2. Adding MinQuantity attribute to Item Entity (even though it will move to a different entity later)
  4. Other
    1. Configuration of Logo based on tenant.

© 2025 Arda Systems, All rights reserved

Comments