Suppose you've drawn a diagram in some GUI illustration app. You export the diagram to SVG and insert it as a left-aligned block image in an HTML page.
You deliberately scale the diagram so that the size of the text in the diagram matches the 16px reflowable body text in the surrounding content of the HTML page.
In a maximized browser window on a 23-inch monitor, the reflowable body text column is much wider than the diagram, leaving a wide area of white space to the right of the diagram:
It would look better if the diagram was responsive; in this case, where there is an abundance of space, if the diagram filled more of the available width. (Just center-align it and be done with it?!)
However, you don't want to just scale up the entire SVG to fit the available width, because the text in the diagram would also scale up. The text in the diagram would start to shout. You want the text in the diagram to remain at the same 16px size.
And you don't want to just scale up everything but the text. Text labels could be "lost", appear disproportionately small, inside relatively massive shapes.
Ideally, you want a method that distributes—lays out—the shapes in a diagram to fit the available space.
What app or tool can you use to draw diagrams that dynamically resize to fit the width of their HTML container, but with fixed-sized text? Responsive diagrams that don't raise their voice above the surrounding body text when they scale up.
My initial use case is diagrams that are, in formal terms, graphs, including directed graphs: nodes connected by edges.
Example renderings of the same diagram at different widths
Note that, in both cases, the text inside the diagram is the same size as text outside the diagram.
With this rudimentary example diagram, which has only a few nodes and edges, the differences in layout are minimal: in the wider example, the nodes are slightly further apart and the edges are correspondingly longer. With a more complex diagram, and perhaps also with a greater relative difference in width, I can imagine a "layout engine" making more significantly different decisions about how to arrange the nodes and edges to fit the container.
I acknowledge that, especially for such a simple diagram, a layout that fits the size of the container isn't necessarily optimal for readability or looks. In some cases, external white space might be preferable to a "spidery" diagram with extremely long edges, and nodes that shrinkwrap to fit their labels might not be ideal.
Requirements
To achieve what I want, I think I need:
- An abstract definition of a diagram that is independent of the size of the HTML container
- A method for redrawing the diagram to fit the size of the HTML container, based on that description, when the HTML container resizes
My starting point: DOT and d3-graphviz
The DOT language is:
[An] abstract grammar for defining Graphviz nodes, edges, graphs, subgraphs, and clusters
Graphviz renders DOT into various output formats, such as SVG.
d3-graphviz is a D3 plugin based on Graphviz.
I already have experience with d3-graphviz.
Here, for example, is DOT source of the diagram shown previously in the example renderings (note: <width>
and <height>
are placeholders for actual values):
digraph {
ratio="fill"
size="<width>,<height>"
margin=0
pad=0
bgcolor="#f0f0f0"
node [shape=box,style="rounded,filled",color="#ccccff", fontname="Arial", fontsize="12pt"]
edge [color="#6666ff"]
a -> d
b -> d
c -> d
a [label="Alpha"];
b [label="Bravo"];
c [label="Charlie"];
d [label="Width: <width>"];
}
DOT and d3-graphviz are my starting point; they're what I'm familiar with. I'm very open to answers that involve other abstract grammars, other tools.
Use d3-graphviz.
Dynamically inject the width of the HTML container into the DOT source.
When the container resizes, re-render the DOT to match the container width.
Rudimentary examples
Tip: Run the following snippets "full page", and then resize the browser window.
Example 1
Example 2
Known limitations