Using style cycles
Note that currently style cycles are in a mock-up stage since there are no user-defined types available yet. Instead, we use a huge hack abusing existing types. But this is good for studying how everything should behave.
Style cycles define a repeating sequence of for example colors or marks for consecutive plots. This is useful to
- quickly be able to distinguish multiple plots without explicitly setting their color and
- to ensure consistency between diagrams.
#lq.diagram(
lq.plot(range(4), range(2, 6), label: [1]),
lq.plot(range(4), range(1, 5), label: [2]),
lq.plot(range(4), range(0, 4), label: [3]),
)
Many plot types are affected by style cycles (exceptions are for example boxplot
, colormesh
, and contour
).
Usage
The style cycle can be set through diagram.cycle
. The simple option is to pass an array of colors:
#lq.diagram(
cycle: (red, teal),
lq.plot(range(4), range(2, 6), label: [1]),
lq.plot(range(4), range(1, 5), label: [2]),
)
Alternatively, an array of dictionaries can be used to specify color
, mark
, and stroke
(all optional).
#lq.diagram(
cycle: (
(color: red, mark: "x"),
(color: teal, mark: "+")
),
lq.plot(range(4), range(2, 6), label: [1]),
lq.plot(range(4), range(1, 5), label: [2]),
)
For full control you can pass an array of content-transforming functions:
#lq.diagram(
cycle: (
it => {
set lq.style(fill: red);
set lq.mark(align: lq.marks.s3)
it
},
it => {
set lq.style(fill: teal)
set lq.mark(align: lq.marks.s)
it
},
),
lq.plot(range(4), range(2, 6), label: [1]),
lq.plot(range(4), range(1, 5), label: [2]),
)
Here, we set properties of lq.style
and lq.mark
to define a custom style cycle. These are explained more in-depth in the following section.
Style properties
Currently, there are two things that can be set: lq.style
and lq.mark
.
lq.style
determines the general color and stroke settings that also apply to bar plots etc.lq.style.fill
is the main color. This sets the line, mark and fill color at once. However, this setting has the lowest precedence and is overridden by all other settings if they are not set toauto
.lq.style.stroke
is the main stroke, applied to the line ofplot
,stem
etc. but also tobar
. If the stroke contains no color, the color is inherited fromlq.style.fill
lq.mark
determines all settings concerning markslq.mark.fill
: How to fill the mark. Normally inherited fromlq.style.fill
lq.mark.stroke
: How to stroke the mark. Normally inherited fromlq.style.fill
(!)lq.mark.inset
: The size of the mark. The nameinset
is of course temporary and due to the hack of abusing thegrid
type for marks.lq.mark.align
: The mark to use. This has to be one oflq.marks
or another function that takes a single positional dictionary with the keysfill
,stroke
, andsize
. The namealign
is of course temporary and due to the hack of abusing thegrid
type for marks.
Many parameters of plot
, scatter
etc. are by default set to auto
, e.g., stroke
, mark
, and mark-size
. This means that the fields are inherited from the current style. The current style can of course be overridden per-plot by setting these parameters to explicit values.
Built-in color sequences
Lilaq features a number of built-in color sequences under lq.color.map
. As the default, Lilaq uses the color cycle petroff10
. This is a color sequence introduced by Matthew A. Petroff in https://arxiv.org/abs/2107.02270 that is optimized for distinguishability, aesthetics, and people with color deficiencies.
Also available is the default Matplotlib color cycle and its 20-color pendant.
Generating and folding color cycles
The module lq.cycle
contains helper functions for generating color cycles.