Axis configuration
In this tutorial you will learn how to configure and customize the axes of a diagram.
The possible parameters are documented at axis
.
Axes use the Typst package Zero for formatting numbers in a consistent way throughout a document. Please refer to its documentation for configuring how numbers are displayed.
Scales
The scaling of the diagram axes is crucial to how the data is presented. By default, linear scaling is used, mapping the data coordinates proportionally to document coordinates. The scale can be changed through the parameters diagram.xscale
and diagram.yscale
for the two main axes (or axis.scale
in general). Other common scalings are logarithmic ("log"
) and symmetric logarithm ("symlog"
).
#lq.diagram(
xlim: (1, 100),
ylim: (-10, 10),
xscale: "log",
yscale: "symlog"
)
These strings are just shorthands to select predefined scale objects. Equivalently, you can put xscale: lq.scale.log()
. Although it is more verbose, this second way of creating a new scale gives use more control. As an example, let us create a new symlog scale with a different threshold than the default.
#lq.diagram(
ylim: (-10, 10),
yscale: lq.scale.symlog(threshold: 10)
)
You can also make up your own scale with the scale
constructor.
Limits
By default, the axis limits are computed automatically as the maxima and minima of all data points plotted in the diagram. On top, so-called margins (see diagram.margin
) are applied to increase the computed range by some amount.
You can define the limits manually through diagram.xlim
and diagram.ylim
(or axis.lim
in general).
#let xs = lq.linspace(-3, 10)
#lq.diagram(
xlim: (0, 8),
ylim: (0, auto),
lq.plot(xs, xs.map(calc.sin))
)
Note that
- no margins are applied to fixed limits and
- it is possible to leave the upper or lower limit at
auto
while fixing the other.
Axis labels
No data without context! It is important to meaningfully label each axis of a diagram − for this we use diagram.xlabel
and diagram.ylabel
(again axis.label
in general). Instead of passing content to these parameters, you can also create a label
to benefit from its additional parameters.
#lq.diagram(
xlabel: [On the $x$ axis],
ylabel: lq.ylabel(pad: 2em)[and the $y$ axis... ]
)
Through the label
type, we can also apply powerful customization. Let us align the and labels with the right and top edge of the diagram, respectively.
#show lq.selector(lq.label): set align(top + right)
#show: lq.set-label(pad: 1em)
#lq.diagram(
xlabel: [Time],
ylabel: [Value]
)
Since Typst does not yet support user-defined types for which set
and show
rules can be written, Lilaq uses elembic as a temporary solution to work around this limitation.
In the future, you will be able to write just
#show lq.label: set align(top + right)
#set lq.label(pad: 1em)
Ticks
Todo: This is a long story. Let us make this a separate tutorial.
The spine
The axis spine
(the line drawn along the axis) is an element of its own.
#show: lq.set-spine(stroke: 1pt + red)
#show: lq.set-tick(stroke: 0.5pt)
#lq.diagram(width: 2cm, height: 3cm)
By default, tick
inherits its stroke from the spine to make setting thickness and color easier. The spine also has parameters for arrow tips, as demonstrated in this section of the tutorial.
Note that the parameters axis.stroke
, axis.tip
, and axis.toe
are directly forwarded to the spine of the axis and can be used to override the spine settings per-axis.
Placement and mirrors
Usually, the -axis is placed at the bottom and the -axis is placed at the left of a diagram. But what if we wanted to change that up? The parameter axis.position
allows us to do just that.
#lq.diagram(
yaxis: (position: right),
width: 3cm, height: 3cm
)
But now the axis on the left has vanished entirely. Before, there was a copy − a so-called mirror of the axis on the right side (although without the tick labels). This is because when specifying the position explicitly, the mirror is turned off by default.
We can restore the mirror axis by setting axis.mirror
to true
. This parameter also gives us more fine-grained control over the nature of the mirror. By passing (ticks: false)
, we can for example remove the ticks from the mirror.
#lq.diagram(
yaxis: (position: right, mirror: true),
xaxis: (mirror: (ticks: false)),
width: 3cm, height: 3cm
)
With (tick-labels: true)
, it is even possible to show the tick labels on the mirror axis.
Not only can axes be placed at the four sides of the diagram, they can even be moved in- or outside the diagram, as shown in the sections below.
Arrows and schoolbook styles
The previous section leads us to the following question: What if want to have our axis right in the middle, going through the origin of the coordinate system? Let us make some modifications to get a "schoolbook"-style diagram.
-
Instead of passing an
alignment
toaxis.position
, we can also pass afloat
, a data value on the other axis, through which the axis should pass. -
In addition, let us add arrow tips to the axes. These are powered by the package tiptoe.
-
Also, we want to filter the ticks. It doesn't look nice when the ticks overlap the axes at the origin and also, the outermost ticks would overlap with the arrow marks. To avoid that, we install a tick filter that returns
false
when the tick value is0
and its distance to the outer edges of the diagram is less than5pt
. -
Finally, the tick marks on the axis should be centered. We can achieve this by applying a "
set
" rule ontick
.
#import "@preview/tiptoe:0.3.0"
#show: lq.set-tick(inset: 2pt, outset: 2pt, pad: 0.4em)
#let filter(value, distance) = value != 0 and distance >= 5pt
#lq.diagram(
xlim: (-5.8, 5.8),
ylim: (-5.8, 5.8),
xaxis: (position: 0, tip: tiptoe.stealth, filter: filter),
yaxis: (position: 0, tip: tiptoe.stealth, filter: filter),
)
Following the styling and preset tutorial, we can wrap this up in a reusable preset. This is demonstrated in the schoolbook style example.
Additional axes
Sometimes, two is just not enough. In this section, you will learn how to add an arbitrary number of axes to a diagram. We differentiate between three kinds of axes:
- The main axes are just the two default -axis and -axis.
- Dependent axes are axes that show equivalent values for the same data but in different units.
- An independent or twin axis shows data unrelated to the corresponding main or -axis. Twin axis are used to unite two plots in one diagram when they share one common axes.
Dependent axes
A dependent axis is linked to the corresponding main axis ( or ) and defines a pair of functions that transform to and from this secondary unit. The axis limits are automatically inherited from the parent axis and cannot be specified manually.
An example would be a spectrum that is shown in dependence of the wavelength on one side and in terms of photon energy on the opposite side. Note that there is a fixed relation between the wavelength and the energy of a photon, so these two really describe the same data. Another example is the famous Hertzsprung-Russell diagram which commonly shows absolute magnitude and luminosity on the -axis and temperature and the corresponding spectral class on the -axis.
In the following example, we have a velocity along the bottom -axis and the corresponding kinetic energy which scales quadratically with on a secondary axis at the top. First, we remove the mirror of the main -axis (see section on placement and mirrors above). Then we add a second axis
at position: top
. Through axis.functions
, we give the transformation from velocity to energy and its inverse.
#let m = 1 // Let's assume mass of 1
#lq.diagram(
xaxis: (mirror: false),
xlabel: $v$,
xlim: (0, 10),
yaxis: none,
lq.xaxis(
position: top,
label: $E=1/2 m v^2$,
functions: (
v => 0.5 * m * v*v,
E => calc.sqrt(2 * E / m)
)
)
)
Another demonstration can be found in the dual-axis example.
Independent axes (twin axes)
An independent axis is not bound to a parent axis and contains plots as children. It can have its own scaling and individual limits.
One traditional example is a climograph that shows the average temperature and precipitation per month. Both data sets share the “month” axis but have a unique (usually) axis, one for the temperature, one for the precipitation.
First, we turn off the mirror of the main axis and set the main -label. Now we add a secondary axis on the right side to which we add a bar plot of rainfall values.
Finally, we create a temperature plot on the main axes. We use this order to ensure that the temperature is plotted on top of the bars. Alternatively, we could use plot.z-index
to control the plot order.
#lq.diagram(
yaxis: (mirror: false),
ylabel: [Temperature],
lq.yaxis(
position: right,
label: [Precipitation],
lq.bar(
fill: blue.lighten(60%),
(1, 2, 3),
(20, 30, 24)
)
),
lq.plot(
color: red,
(1, 2, 3),
(14, 16, 13)
),
)
The climograph example shows a fully-fledged demo.
The number of axes that can be added to a diagram is in principal unlimited. The axis.position
can also take a dictionary (align: .., offset: ..)
that makes it possible to move axes outside the diagram.
#lq.diagram(
xscale: lq.scale.log(base: 2),
ylim: (-1, 1),
lq.yaxis(
position: (align: right, offset: 20pt),
functions: (x => 2*x, y => 0.5*y)
),
lq.yaxis(
position: (align: right, offset: 50pt),
functions: (x => calc.exp(x), y => calc.ln(y))
),
lq.yaxis(
position: (align: right, offset: 80pt),
functions: (x => (x+1)*(x+1), y => calc.sqrt(y)-1)
),
)