24 releases
new 0.9.2 | Jan 13, 2025 |
---|---|
0.8.0 | Oct 20, 2024 |
0.7.0 | Jun 17, 2023 |
0.3.1 | Nov 29, 2021 |
#25 in Multimedia
602 downloads per month
5MB
3K
SLoC
Contains (WOFF font, 59KB) impact.woff2, (WOFF font, 59KB) assets/fonts/impact.woff2, (WOFF font, 59KB) impact.woff2
Collagen — The Collage Generator
Table of Contents
Quick Start
Install Collagen with cargo install collagen
. This will install the
executable clgn
.
Once installed, if you have a manifest file at path
path/to/folder/collagen.jsonnet
(keep reading to learn what goes in
this manifest), you can run the following command:
clgn -i path/to/folder -o output-file.svg
To continuously monitor the input folder and re-run on any changes, you can run Collagen in watch mode:
clgn -i path/to/folder -o output-file.svg --watch
In watch mode, every time a file in path/to/folder
is modified,
Collagen will attempt to regenerate the output file and will either
print a generic success message or log the specific error encountered,
as the case may be. Watch mode will never terminate on its own. As with
most terminal commands, you can terminate it with
Ctrl-C.
This doc has several examples that
can serve as a good starting point for creating a manifest. More
examples are available as test cases in tests/examples
.
Introduction
Collagen — from “collage” and “generate” (and because
collagen is the protein that
holds your body together; s/protein/tool/;s/body/images/
) — is a
program that takes as input a folder containing a JSON or
Jsonnet manifest file that describes the layout
of any number of SVG elements (<rect>
, <line>
, <g>
, etc.), image
files (JPEG, PNG, etc.), other SVGs, and other Collagen folders, and
produces as output a single SVG file with any external assets embedded.
That is, it converts a textual description of an SVG and any provided
assets into a single portable SVG file.
JSON is mapped to SVG in a more or less one-to-one fashion, with a few conveniences thrown in (such as automatically base64-encoding images). For more programmatic conveniences, it is recommended to use Jsonnet, which supports the following:
-
Variables
-
Arithmetic
-
List and object comprehensions (i.e.
for
-loops) -
Conditional logic
-
Functions, including
-
The ability to define and call functions
-
A large standard library of built-in functions
-
-
String interpolation
-
Imports for code sharing across multiple files
-
List and object concatenation/merging
For instance, the following manifest will create a rainbow pinwheel.
collagen.jsonnet
local width = 400;
local height = width;
local n_spokes = 16;
local cx = width / 2;
local cy = height / 2;
local spoke_length = width * 0.75;
// this calls a stdlib function (pi is not built in yet)
local pi = std.acos(-1);
{
attrs: {
// string interpolation
viewBox: "0 0 %d %d" % [width, height],
},
children: [
{
local t = i / n_spokes,
local theta = t * pi,
local dx = (spoke_length / 2) * std.cos(theta),
local dy = (spoke_length / 2) * std.sin(theta),
tag: "line",
attrs: {
x1: cx + dx,
x2: cx - dx,
y1: cy + dy,
y2: cy - dy,
// we can also build strings by adding them together
stroke: "hsl(" + std.toString(360 * t) + ", 100%, 50%)",
"stroke-width": 5,
"stroke-linecap": "round",
},
}
for i in std.range(0, n_spokes - 1)
],
}
Rationale
Creating and editing images is hard. Most of the difficulty is due to lack of programmatic facilities offered by image editing programs. Collagen aims to fill this gap in the following ways:
-
Plain text is portable and trivially editable. When editing an image is done by editing a text file, the only barrier to entry is knowing the format (JSON or Jsonnet), knowing the schema used by Collagen, and knowing the catalog of SVG elements. In addition, a folder containing a text file and some images is simple to share: just zip it and send it along. Finally, Collagen is nondestructive; all assets used in the final image are available in their original form right in that folder.
-
While most image editing programs support “guides” and “snapping” that allow graphical elements to be precisely positioned relative to each other, there is no “glue” to hold the elements together indefinitely; moving one tends not to move the other. For instance, an arrow pointing from a given corner of a rectangle to a fixed point cannot be made to have its tail always lie on that corner while leaving its head stationary. Variables solve this problem by letting the author use the same variables, and perhaps a little arithmetic, to specify the position of both the rectangle’s corner and the arrowhead. More generally, variables support the problem of keeping several values in sync without having to edit multiple hard-coded values.
-
Image editing programs output a single image file which is of one of the well-known image types (JPEG, PNG, etc). Different image formats are optimized for different types of image data, and break down when made to store image data they weren’t optimized for.
-
For instance, JPEG is optimized for images with smoothly varying gradients (which tends to mean photographs taken by a physical camera). Therefore it produces images with ugly compression artifacts when made to store geometric shapes or text. Below are a PNG image of some text, and that same PNG after conversion to JPEG. The JPEG has been enlarged to make the compression artifacts more easily visible.
A screenshot of the word “Collagen” in PNG format
A screenshot of the word “Collagen” in JPG format, zoomed in
-
On the other hand, PNG is optimized for images with long runs of few distinct colors, and requires a massive file size to store the kind of data that JPEG is optimized for. Despite displaying exactly the same image (source), the PNG file below is 6.6 times bigger than the JPEG.
A JPEG, weighing in at 407KB
A PNG, weighing in at 2.7MB
-
JPEGs and PNGs are both raster formats, which means they correspond to a rectangular grid of pixels. A given raster image has a fixed resolution (given in, say, pixels per inch), which is, roughly speaking, the amount of detail present in the image. When you zoom in far enough on a raster image, you’ll be able see the individual pixels that comprise the image. Meanwhile, vector graphics store geometric objects such as lines, rectangles, ellipses, and even text, which have no resolution to speak of — you can zoom infinitely far on them and they’ll always maintain that smooth, pixel-perfect appearance. Without Collagen, if you want to, say, add some text on top of a JPEG, you have no choice to but to rasterize the text, converting the infinitely smooth shapes to a grid of pixels and losing the precision inherent in vector graphics.
Collagen fixes this by allowing JPEGs, PNGs, and any other images supported by browsers to coexist with each other and with vector graphic elements in an SVG file, leading to neither the loss in quality nor the increase in file size that arise when using the wrong image format. (Collagen achieves this by simply base64-encoding the source images and embedding them directly into the SVG.) So you could, for instance, add vector shapes and text on top of an raster image without rasterizing them.
-
-
Creating several similar elements by hand is annoying, and keeping them in sync is even worse. Jsonnet supports “list comprehension”, aka
for
loops, to programmatically create arbitrary numbers of elements, and the children elements can make use of the loop variable to control their behavior. We saw this above in the pinwheel, which used the loop variablei
to set the angle and color of each spoke. Thefor
loop itself had access to then-spokes
variable set at the beginning of the file, which goes back to point 2: variables make things easy. -
Why SVG at all? Why not some other output image format?
-
SVGs can indeed store vector graphics and the different kinds of raster images alongside each other.
-
SVGs are supported by nearly every browser and are widely supported in general.
-
SVGs are "just" a tree of nodes with some attributes, so they’re simple to implement.
-
SVGs are written in XML, which is plain text and simple(-ish) to edit.
-
The above features make Collagen suitable as an “image editor for programmers”. Everybody loves memes, programmers included, so let’s use Collagen to make one.
collagen.json
local width = 800;
{
vars: { width: 800 },
attrs: { viewBox: "0 0 %d 650" % width },
children: [
{
tag: "defs",
children: [
{
tag: "style",
children: {
text: '@import url("https://my-fonts.pages.dev/Impact/impact.css");',
is_preescaped: true,
},
},
],
},
{
image_path: "./drake-small.jpg",
attrs: {
width: width,
},
},
{
local x = 550,
local dy = 50,
tag: "text",
attrs: {
"font-family": "Impact",
"font-size": 50,
color: "black",
"text-anchor": "middle",
"vertical-align": "top",
x: x,
y: 420,
},
children2: [
{
tag: "tspan",
text: [
"Using SVG-based text,",
"which is infinitely",
"zoomable and has",
"no artifacts",
][i],
attrs: { x: x, dy: if i == 0 then 0 else dy },
}
for i in std.range(0, 3)
],
},
],
}
Using Collagen
Definitions
Collagen | The name of this project. |
clgn |
The executable that does the conversion to SVG. |
Manifest | The collagen.json or collagen.jsonnet file residing at the top level inside a skeleton. If both exist, collagen.jsonnet is preferred. |
Skeleton | A folder that is the input to clgn . It must contain a manifest file and any assets specified therein. For instance, if skeleton my_skeleton’s manifest contains `{ "image_path": "path/to/image" } , then my_skeleton/path/to/image must exist. |
In-Depth Description
The input to Collagen is a folder containing, at the bare minimum, a
manifest file named collagen.json
or collagen.jsonnet
. Such a
folder will be referred to as a skeleton. A manifest file is more or
less a JSON-ified version of an SVG (which is itself XML), with some
facilities to make common operations, such as for loops and including an
image by path, more ergonomic. For instance, without Collagen, in order
to embed an image of yours in an SVG, you would have to base64-encode it
and construct that image tag manually, which would look something like
this:
<image href="...(many, many bytes omitted)..."></image>
In contrast, including an image in a Collagen manifest is as simple as including the following JSON object as a descendent of the root tag:
{ "image_path": "path/to/image" }
Collagen handles base64-encoding the image and constructing the
<image>
tag with the correct attributes for you.
Basic Schema
In order to produce an SVG from JSON, Collagen must know how to convert
an object representing a tag into an actual SVG tag, including
performing any additional work (such as base64-encoding an image).
Collagen identifies the type of an object it deserializes simply by the
keys it contains. For instance, the presence of the "image_path"
property tells Collagen that this tag is an <image>
tag with an
associated image file to embed. To avoid ambiguities, it is an error for
an object to contain unexpected keys.
Tags are listed at docs.rs/collagen.
FAQ
-
How is this different from a templating language like Liquid?
Templating languages generally consist of two components: the templating library, which does the rendering of the template, and the template language, which usually resembles HTML with the addition of things like control flow and interpolation. The library is responsible for combining a template file and some external data and turning them into an output file. But you can’t (generally) write your data literally in the template file, which is inconvenient, and the overhead of needing to write down your data separately can be quite large compared to the complexity of the image you would use Collagen to create. In addition, to actually drive the templating library probably requires writing some code in the library’s language and running it in the language’s runtime.
In contrast, Collagen lets you include data directly in the manifest and runs via single executable with no runtime to speak of. It also lets you write your image in a single file (the manifest) instead of two (the template file and the “real code” that creates the output.) In addition, with Collagen, there is no syntax to learn, per se; you simply write JSON or Jsonnet.
-
How does Collagen handle paths across multiple platforms?
In general, filesystem paths are not necessarily valid UTF-8 strings. Furthermore, Windows and *nix systems use different path separators. How, then, does Collagen handle paths to files on disk in a platform-agnostic way? All paths consumed by Collagen must be valid UTF-8 strings using forward slashes (
/
) as the path separator. Forward slashes are replaced with the system path separator before resolving the path. Sopath/to/image
remains unchanged on *nix systems, but becomespath\to\image
on Windows. This means that in order to be portable, path components should not contain the path separator of any system, even if it is legal on the system on which the skeleton is authored. For instance, filenames with backslashes\
are legal on Linux, but would pose a problem when decoding on Windows. Generally speaking, if you restrict your file and folder names to use word characters, hyphens, whitespace, and a limited set of punctuation, you should be fine.Naturally you are also limited by the inherent system limitations on path names. For instance, while
CON
is a valid filename on Linux, it is forbidden by Windows. Collagen makes no effort to do filename validation on behalf of systems on which it may be used; it is up to the author of a skeleton to ensure that it can be decoded on a target device.
Dependencies
~16–28MB
~428K SLoC