Define a vega-lite visual
In this tutorial, you will use vega()
to visualise a tabular datastream. You will
- define a chart using
vega()
, and - launch a solution and observe the results.
This lesson will assume that you have an empty project and asset which you can to deploy to a workspace named 03_07_03_define_a_vega_visual
with the following command:
edk template deploy -ycw 03_07_03_define_a_vega_visual
What is Vega-lite?
Vega-lite is an open sourced declarative language, or high-level grammar for interactive data visualisation. Many different types of visualisations can easily be defined using vega-lite. The Elara EDK libraries, make it simple to create Vega-lite visualisations from datastreams.Define and deploy a template
To use the vega()
methods you will first perform the following steps:
- define a datasource using a similar pattern as your previous
My Source
definition. - set the value of based on some generated data of type
Map<string, { date: Date, category: string, value: bigint }>()
value. - add the datasource to a template
In an asset, perform the above steps to create the resulting Typescript code:
import { SourceBuilder, Template } from "@elaraai/core"
const my_source = new SourceBuilder("My Source")
.value({
value: new Map(Array.from({ length: 100 }).map((_, index) => (
[`${index}`, {
date: new Date(new Date().valueOf() + index * 3600 * 1000),
category: `category ${index % 4}`,
value: BigInt(Math.round(200 * Math.sin(Math.PI * index / 100))),
}]
)))
})
export default Template(my_source)
Define a vega-lite view
You can define a single vega-lite view using the vega()
method of VegaBuilder()
, by taking the following steps:
- add a new layout "01 - My Layout", using a prefix to maintain order
- add a vega to the layout using the
vega()
method of theLayoutBuilder
- configure a vega view, by using the
view()
method of theVegaBuilder
- define the datastream for the visualisation with
fromStream()
method of theVegaViewBuilder
- configure a scatter chart with the following series each from a
VegaFieldEncodingBuilder
- build an x series, sort in
ascending
order, fromdate
- build a y series, from
value
- build a color series, from
value
- build a size series, from
value
- build an x series, sort in
- add the new layout to the template
In the definition 01 - My Layout
add the above changes:
import { SourceBuilder, Template, LayoutBuilder } from "@elaraai/core"
const my_source = new SourceBuilder("My Source")
.value({
value: new Map(Array.from({ length: 100 }).map((_, index) => (
[`${index}`, {
date: new Date(new Date().valueOf() + index * 3600 * 1000),
category: `category ${index % 4}`,
value: BigInt(Math.round(200 * Math.sin(Math.PI * index / 100))),
}]
)))
})
const my_layout = new LayoutBuilder("01 - My Layout")
.vega("My Chart", builder => builder
.view(builder => builder
.fromStream(my_source.outputStream())
.scatter({
x: builder => builder.value(fields => fields.date).sort('ascending'),
y: builder => builder.value(fields => fields.value),
color: builder => builder.value(fields => fields.value),
size: builder => builder.value(fields => fields.value)
})
)
)
export default Template(my_source, my_layout);
Navigate to the layout URL located within the tenant you deployed to see your chart:
workspaces/03_07_03_define_a_vega_visual/layouts/01%20-%20My%20Layout/
Series Attributes
Many of the type specific attributes of a series can be customised within each series, with the available methods being exposed by VegaFieldEncodingBuilder
depending on the data type of the chosen field.
For example, given the x series above, the following methods could be used to customise based on the vega-lite specification rules:
Layout | Description |
---|---|
title() | Define the series title |
format() | Define the series label format. |
timeUnit() | Define the series time unit, limited to DateTimeType |
sort() | Define the series sort order. |
bin() | Define binning to perform on the series. |
aggregate() | Define the aggregation to perform. |
normalize() | Define the series stacking/ |
type() | Define the data type, for example ('nominal' , 'ordinal' , 'quantitative' ). |
Define layered vega-lite views
You can define a layered vega-lite visualisations using the layered()
method of VegaBuilder()
, by taking the following steps:
- add a new layout "02 - My Other Layout", using a prefix to maintain order
- add a vega to the layout using the
vega()
method of theLayoutBuilder
- configure a layered vega, by using the
layered()
method of theVegaBuilder
- configure a vega view, by using the
view()
method of theVegaLayeredBuilder
2. define the datastream for the visualisation withfromStream()
method of theVegaViewBuilder
3. configure a column chart with the following series each from aVegaFieldEncodingBuilder
- build an x series, sort in
ascending
order, fromdate
- build a y series, from
value
- build a color series, from
category
- build an x series, sort in
- configure a second vega view, by using the
view()
method of theVegaLayeredBuilder
2. define the datastream for the visualisation withfromStream()
method of theVegaViewBuilder
3. configure a scatter chart with the following series each from aVegaFieldEncodingBuilder
- build an x series, sort in
ascending
order, fromdate
- build a y series, from
value
- build a color series, from
value
- build a size series, from
value
- build an x series, sort in
- configure a vega view, by using the
- add the new layout to the template
In the definition 02 - My Other Layout
add the above changes:
import { SourceBuilder, Template, LayoutBuilder } from "@elaraai/core"
const my_source = new SourceBuilder("My Source")
.value({
value: new Map(Array.from({ length: 100 }).map((_, index) => (
[`${index}`, {
date: new Date(new Date().valueOf() + index * 3600 * 1000),
category: `category ${index % 4}`,
value: BigInt(Math.round(200 * Math.sin(Math.PI * index / 100))),
}]
)))
})
const my_layout = new LayoutBuilder("01 - My Layout")
.vega("My Chart", builder => builder
.view(builder => builder
.fromStream(my_source.outputStream())
.scatter({
x: builder => builder.value(fields => fields.date).sort('ascending'),
y: builder => builder.value(fields => fields.value),
color: builder => builder.value(fields => fields.value),
size: builder => builder.value(fields => fields.value)
})
)
)
const my_other_layout = new LayoutBuilder("02 - My Other Layout")
.vega("My Chart", builder => builder
.layered(builder => builder
.view(builder => builder
.fromStream(my_source.outputStream())
.column({
x: builder => builder.value(fields => fields.date).sort('ascending'),
y: builder => builder.value(fields => fields.value),
color: builder => builder.value(fields => fields.category),
})
)
.view(builder => builder
.fromStream(my_source.outputStream())
.scatter({
x: builder => builder.value(fields => fields.date).sort('ascending'),
y: builder => builder.value(fields => fields.value),
color: builder => builder.value(fields => fields.value),
size: builder => builder.value(fields => fields.value)
})
)
)
)
export default Template(my_source, my_layout, my_other_layout);
Navigate to the layout URL located within the tenant you deployed to see your chart:
workspaces/03_07_03_define_a_vega_visual/layouts/02%20-%20My%20Other%20Layout/
Define a vega-lite specification
You can define a vega-lite specification to display in the user-interface spec()
method of VegaViewBuilder()
, by taking the following steps:
- add a new layout "03 - My Last Layout"
- add a vega to the layout using the
vega()
method - configure a vega view, by using the
view()
method of theVegaBuilder
- define the datastream for the visualisation with
fromStream()
method of theVegaViewBuilder
- configure a spec chart with returning a valid vega-lite specification, based on the field names provided in
fields
- add the new layout to the template
In the definition 03 - My Last Layout
add the above changes:
import { SourceBuilder, Template, LayoutBuilder } from "@elaraai/core"
const my_source = new SourceBuilder("My Source")
.value({
value: new Map(Array.from({ length: 100 }).map((_, index) => (
[`${index}`, {
date: new Date(new Date().valueOf() + index * 3600 * 1000),
category: `category ${index % 4}`,
value: BigInt(Math.round(200 * Math.sin(Math.PI * index / 100))),
}]
)))
})
const my_layout = new LayoutBuilder("01 - My Layout")
.vega("My Chart", builder => builder
.view(builder => builder
.fromStream(my_source.outputStream())
.scatter({
x: builder => builder.value(fields => fields.date).sort('ascending'),
y: builder => builder.value(fields => fields.value),
color: builder => builder.value(fields => fields.value),
size: builder => builder.value(fields => fields.value)
})
)
)
const my_other_layout = new LayoutBuilder("02 - My Other Layout")
.vega("My Chart", builder => builder
.layered(builder => builder
.view(builder => builder
.fromStream(my_source.outputStream())
.column({
x: builder => builder.value(fields => fields.date).sort('ascending'),
y: builder => builder.value(fields => fields.value),
color: builder => builder.value(fields => fields.category),
})
)
.view(builder => builder
.fromStream(my_source.outputStream())
.scatter({
x: builder => builder.value(fields => fields.date).sort('ascending'),
y: builder => builder.value(fields => fields.value),
color: builder => builder.value(fields => fields.value),
size: builder => builder.value(fields => fields.value)
})
)
)
)
const my_last_layout = new LayoutBuilder("03 - My Last Layout")
.vega("My Chart", builder => builder
.view(builder => builder
.fromStream(my_source.outputStream())
.spec((fields) => ({
$schema: "https://vega.github.io/schema/vega-lite/v5.json",
transform: [],
mark: { type: "point" },
encoding: {
x: { field: fields.date, type: 'temporal' },
y: { field: fields.value, type: 'quantitative' },
color: { field: fields.value, type: 'quantitative', },
}
}))
)
)
export default Template(my_source, my_layout, my_other_layout, my_last_layout);
Navigate to the layout URL located within the tenant you deployed to see your chart:
workspaces/03_07_03_define_a_vega_visual/layouts/03%20-%20My%20Last%20Layout/
Observe results
If you do not have access to Elara, the results of the above are shown below:
Example solution
The code for this tutorial is available below:
Next steps
In the next tutorial, you will use the tab()
layout to contain multiple visualisations.