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 Sourcedefinition.
- 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 ascendingorder, 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 theVegaLayeredBuilder2. define the datastream for the visualisation withfromStream()method of theVegaViewBuilder3. configure a column chart with the following series each from aVegaFieldEncodingBuilder- build an x series, sort in ascendingorder, 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 theVegaLayeredBuilder2. define the datastream for the visualisation withfromStream()method of theVegaViewBuilder3. configure a scatter chart with the following series each from aVegaFieldEncodingBuilder- build an x series, sort in ascendingorder, 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.