cuddly/README.md

3.3 KiB

Cuddly

Generate HTML and PDF from KDL structed data.

Installation

git clone https://git.bksp.space/hannaeko/cuddly
cd cuddly
python -m venv env
env/bin/pip install .

Usage

Example

Generating a PDF:

env/bin/cuddly --template samples/cv/cv.template.html --input samples/cv.kdl --output samples/cv.pdf

Generating a HTML file

env/bin/cuddly --template samples/cv/cv.template.html --input samples/cv.kdl --output samples/cv.html

Template usage

The template language used is Jinja, source template need to be a single HTML template file.

Base URL

When generating the PDF the base URL is set to the source template directory. All relative links to other resources (for example CSS and images) must be relative to that directory.

Accessing nodes values and attributes

Nodes are available as global variables in the template. The rendered value of a node is the first value that has been assigned to the node. This value can also be accessed using the value property. The list of values can be accessed using the values property.

my_node "something" "other thing"
{{ my_node }}
{{ my_node.value }}
{{ my_node.values | join(', ')}}

renders to

something
something
something, other thing

Accessing children nodes can be done using properties of the children names.

my_node {
    other_node "test"
}
{{ my_node.other_node.value }}

renders to

test

If a node has multiple children of a same node type, then directly accessing the children properties will fail, a loop will be needed. Note that, on the other hand, looping on the children always work, even if there is only one child of the given type.

my_node {
    other_node "first"
    other_node "second"
}
{% for node in my_node.other_node %}
{{ node.value }}
{% endofr %}

renders to

first
second

Attributes of a node can be accessed through the attributes property:

my_node version=42 "something"
{{ my_node.attributes.version }}

renders to

42

Templating nodes

Node specific templates can be defined in the components element in the HTML template file. Nodes can then be rendered using those micro-templates using the render filter.

The context of the micro-template is the node being rendered. As such, children and properties of that node are in the global scope.

menu "Midnight meal" price="10 €" {
    dish name="Tomato soup" "Fancy tomato soup with a slice of bread"
    dish name="Banana bread" "Amazing banana bread with a scoop of icecream"
}
<!DOCTYPE html>
<html lang="{{ lang }}">
<head>
    <meta charset="UTF-8">
</head>
<components>
    <menu>
        <h1>{{ value }}</h1>
        <dl>
            <dt>Price</dt>
            <dd>{{ attributes.prices }}</dd>
        </dl>

        <dl>
        {% for dish in dish %}
            {{ dish | render }}
        {% endfor %}
        </dl>
    </menu>
    <dish>
        <div>
            <dt>{{ attribute.name }}</dt>
            <dd>{{ value }}</dd>
        </div>
    </dish>
</components>
<body>
    {{ menu | render }}
</body>

Samples

  • samples/cv/cv.template.html: Template to generate a CV, see samples/cv.kdl for the input data.