Skip to content

Chart ui-chart

Try Demo

Provides configuration options to create the following chart types:

Properties

PropDynamicDescription
GroupDefines which group of the UI Dashboard this widget will render in.
SizeControls the width of the button with respect to the parent group. Maximum value is the width of the group.
LabelThe text shown within the button.
ClassThe text shown within the button.
Chart TypeLine | Bar | Scatter
Show LegendDefines whether or not a legend is shown between the title and the chart. Each label is driven by msg.topic.
ActionControls how new data is added to a chart. It will either "append", keeping existing data, or "replace", removing existing data, before adding any newly provided data points.
Point ShapeDefine the shape of the point shown in Scatter & Line charts.
Point RadiusDefine the radius (in pixels) of each point rendered onto a Scatter or Line chart.
X-Axis TypeTimescale | Linear | Categorical
X-Axis FormatHH:mm:ss | HH:mm | YYYY-MM-DD | DD/MM | dd HH:mm | Custom | Auto Defines how the values are displayed on the axis, when X-Axis type is 'timescale'. See here for an overview of all available Luxon tokens.
X-Axis LimitAny data that is before the specific time limit (for time charts) or where there are more data points than the limit specified will be removed from the chart.
PropertiesSeries: Controls how you want to set the Series of data stream into this widget. The default is msg.topic, where separate topics will render to a new line/bar in their respective plots.
X: Only available for Line & Scatter Charts. This defines the key (which can be nested) of the value that should be plotted onto the x-axis. If left blank, the x-value will be calculated as the current timestamp.
Y: Defines the key (which can be nested, e.g. 'nested.value') of the value that should be plotted onto the x-axis. This value is ignored if injecting single numerical values into the chart.
Text ColorOption to override Chart.Js default color for text. At moment overrides the text color for Chart Title, Ticks Text, Axis Title and Legend Text
It is possible to return to Chart.Js defaults by using the checkbox Use ChartJs Default Text Colors
Grid Line ColorOption to override Chart.Js default color for Grid Lines and Axis Border.
It is possible to return to Chart.Js defaults by using the checkbox Use ChartJs Default Grid Colors

Dynamic Properties

Dynamic properties are those that can be overriden at runtime by sending a particular msg to the node.

Where appropriate, the underlying values set within Node-RED will be overriden by the values set in the received messages.

PropPayloadStructuresExample Values
Classmsg.classString

Building Charts

In Node-RED Dashboard 2.0, the ui-chart offers a simple way to render data in a variety of chart types. The chart is configurable to tailor to your data.

To map your data to the chart, the most important properties to configure are:

Example key mapping config for UI ChartExample key mapping config for UI Chart

  • Series: Controls how you want to group your data. On a line chart, different series result in different lines for example, on a bar chart, different series result in different bars for a single x-value (stacked or grouped side-by-side).
  • X: Define where to read the value to plot on the x-axis. If left blank, the x-value will be calculated as the current timestamp.
  • Y: Define where to read the value to plot on the y-axis. If left blank, the y-value will be read from msg.payload directly, and assume it to be a number.

The next most important properties to configure are the "Chart Type" and "X-Axis Type".

  • Chart Type: Choose between a Line, Scatter, or Bar chart.
  • X-Axis Type: Choose between a "Timescale" (for time-based data), "Linear" (for numerical data), or "Categorical" (for non-numeric data). You'll notice that some x-axis types are only available for certain chart types.

Line Charts

Timeseries Data

In this example we wire a Slider to our Chart to plot it's output over time:

Example of a Line ChartExample of a rendered line chart with a "time" x-axis.

A very common use case of Node-RED is to process timeseries data, such as sensor readings. In this case, you would set the following:

PropertyValue
Chart TypeLine
X-Axis TypeTimescale

The value for the x property would then be one of two things:

  • If your data is a simple numerical value, you can leave this blank, and the chart will automatically use the current date/time.
  • If your data is an object, you can provide the key of the timestamp in your data, e.g. {"myTime": 1234567890} would set the "X" property to the type key and the value myTime.

Then, the last piece of the puzzle would be to set the y property would be one of two options:

  • If your data is a simple numerical value, you can leave this blank, and the chart will automatically use the value of msg.payload.
  • If your data is an object, you can provide the key of the value in your data, e.g. {"myTime": 1234567890, "myValue": 123} would set the "Y" property to the type key and the value myValue.

Multiple Lines

Example Line Chart with multiple linesExample Line Chart with multiple lines

You can group data together into multiple lines using the Series property. A common use case here is to use msg.topic, where each message sent to the chart will be assigned to a different line based on the msg.topic value. Alternatively, you can set this to key and provide a key in your data to group by.

If you want a single piece of data to plot multiple lines, you can set the Series property to JSON, and then provide an array of keys (e.g. ["key1", "key2"]), which will plot a data point for each key provided, from a single data point.

Scatter Charts

Example of a Scatter PlotExample of a rendered scatter plot with a "time" x-axis.

We can also use "Series" to group points. Let's take an example with the following data set:

json
[
    { "series": "A", "x": 5, "y": 84 },
    { "series": "A", "x": 9, "y": 10 },
    { "series": "A", "x": 11, "y": 70 },
    { "series": "B", "x": 12, "y": 28 },
    { "series": "B", "x": 15, "y": 35 },
    { "series": "B", "x": 26, "y": 42 },
    { "series": "C", "x": 20, "y": 12 },
    { "series": "C", "x": 24, "y": 54 },
    { "series": "C", "x": 27, "y": 60 },
    { "series": "C", "x": 30, "y": 66 }]

In our flow, we'd have:

With the following configuration:

Example of a Scatter Plot with data grouped into "Series"

Which results in:

Example of a rendered scatter plot with a "Linear" x-axis, and data grouped into "Series"Example of a rendered scatter plot with a "Linear" x-axis, and data grouped into "Series".

Bar Charts

Currently, we only support "Category" x-axis types for Bar Charts. This means that the x-axis values will be a string, and the y-axis will be a numerical value.

Let's take an example of loading data for the Star Wars API:

Example of a bar chart showing character 'height' dataExample of a bar chart showing character "height" data

If we take a look at the configuration for this chart:

Example of a bar chart showing character 'height' data

We could easily modify the "Y" property to plot a different value, without needing to modify our data.

Grouped Bars - Financial Data Example

Here we have an example of some financial data:

json
[
    { "year": 2021, "Q1": 115, "Q2": 207, "Q3": 198, "Q4": 163 },
    { "year": 2022, "Q1": 170, "Q2": 200, "Q3": 230, "Q4": 210 },
    { "year": 2023, "Q1": 86, "Q2": 140, "Q3": 180, "Q4": 138 }
]

Bar charts will automatically group data by common x-axis values, but maintain separate bars for each series. When you select a "Bar" chart, then you can choose the "Group By" option to be "Side-by-Side" or "Stacks".

Default behavior for a Bar Chart is to group content "Side-by-Side".

In our chart config we can define:

Configuration for of a bar chart showing financial data, grouped by yearConfiguration for of a bar chart showing financial data, grouped by year

where we have defined "Series" as a type JSON because we want to render multiple bars for each data point, in this case, one for each quarter:

Example of a bar chart showing financial data, grouped by yearExample of a bar chart showing financial data, grouped by year

If we switch over the "Group By" option to be "Stacks", we'd see:

Example of a bar chart showing the same data, but stackedExample of a bar chart showing the same data, but stacked

Grouped Bars - Election Data Example

Here we have a piece of data for each candidate, for each year, which details the number of "Votes" that candidate won.

json
[
    { "candidate": "Dave", "year": 2019, "votes": 100 },
    { "candidate": "Sarah", "year": 2019, "votes": 90 },
    { "candidate": "Chris", "year": 2019, "votes": 160 },
    { "candidate": "Lucy", "year": 2019, "votes": 125 },
    { "candidate": "Dave", "year": 2024, "votes": 20 },
    { "candidate": "Sarah", "year": 2024, "votes": 170 },
    { "candidate": "Chris", "year": 2024, "votes": 150 },
    { "candidate": "Lucy", "year": 2024, "votes": 60 }
]

We have a couple of different ways we could group this data, firstly, we have a series for each "Year" and the x-value defined as the "candidate":

Configuration for of a bar chart showing election data, grouped by candidate, and a series for each yearConfiguration for of a bar chart showing election data, grouped by candidate, and a series for each year

Resulting in:

Example of a bar chart showing election data, grouped by candidate, and a series for each yearExample of a bar chart showing election data, grouped by candidate, and a series for each year

Alternatively, we could have a series per candidate, and then the x-value defined as the "year":

Configuration for of a bar chart showing election data, grouped by year, and a series for each candidateConfiguration for of a bar chart showing election data, grouped by year, and a series for each candidate

Resulting in:

Example of a bar chart showing election data, grouped by year, and a series for each candidate

Pie/Doughnut Charts

These chart types use "Radial" axes. The "Series" property is used to define the layer that the particular data is rendered in, multiple series results in nested pie/doughnut charts.

The "X" value defines the key within a single Series", and the "Y" property should point to the numerical value that dictates the size of a given segment.

Let's take a look at a couple of examples:

With a sample data set such as:

json
[
    { "year": 2021, "quarter": "Q1", "earnings": 115 },
    { "year": 2021, "quarter": "Q2", "earnings": 120 },
    { "year": 2021, "quarter": "Q3", "earnings": 100 },
    { "year": 2021, "quarter": "Q4", "earnings": 180 }
]

We can configure our chart to render a Pie or Doughnut chart like so:

Configuration for of a Pie and Doughnut Chart

Results in the following, where for the "Doughnut" chart has two "Series" worth of data.

Example of Pie and Doughnut ChartsExample of Pie and Doughnut Charts

Controls

Removing Data

"Append" or "Replace" Added In: v0.11.3

The "Action" property on the chart allows you to control whether you:

  • Append: Any new data provided will be added to the existing data on the chart
  • Replace: Any existing data will first be removed, then new data will be added.

If you ever want to override the property on a message-by-message basis, you can also do this by including a msg.action property, which will override the default behaviour. For example:

js
msg = {
    "action": "append",
    "payload": 1
}

Would append this data point to the chart, leaving existing data, even if the underlying chart has been configured to always "Replace".

Clear All Data

Alternatively, you can remove all data from a chart at any time by sending a msg.payload of [] to the node. Most commonly, this is done by wiring a ui-button to the ui-chart node and configuring the button to send a JSON payload with a value of [].

Nested Data

It is a common use case that you would have data structured as JSON, and want to plot some of it e.g:

js
msg = {
    "payload": {
        "id": "Dataset 1",
        "value": 3,
        "nested": {
            "value": 1
        }
    }
}

Here, we can utilize the "Properties" series, x and y to define which values we want to plot on the chart. To access the relevant data point here you can use the key: type and use dot-notation, e.g: nested.value.

Live Data

If you're producing "live" data (from sensors for example), you do not need to define how the x property should be plotted. Instead, you can leave this blank and the chart will automatically calculate the current date/time.

This works equally well if you use Object formatted data, e.g.

js
msg = {
    "topic": "Sensor A" 
    "payload": {
        "value": 3
    }
}

Where you could set the y property to key:value. The x value, if left blank in the configuration would be calculated as the current date/time.

Building Custom Charts

ChartJS has a rich set of configuration options, of which we only expose a small subsection via the Node-RED configuration. If you want to customise the appearance of your chart further, or even render charts we don't yet support, you can do so with a UI Template node.

Currently, although not ideal, we do need to load the ChartJS library from a CDN, and then watch for the file to have been loaded before we can use it, as per the Loading External Dependencies details in the UI Template documentation.

Example: Static Data

Example of a static 2D bar chart

Here here is the template code that will render this bar chart:

html
<template>
    <canvas ref="chart" />
</template>

<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>

<script>
    export default {
        mounted() {
            // code here when the component is first loaded
            let interval = setInterval(() => {
                if (window.Chart) {
                    // Babylon.js is loaded, so we can now use it
                    clearInterval(interval);
                    this.draw()
                }
            }, 100);
        },
        methods: {
            draw () {
                const ctx = this.$refs.chart
                new Chart(ctx, {
                    type: 'bar',
                    data: {
                        labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'],
                        datasets: [{
                            label: '# of Votes',
                            data: [12, 19, 3, 5, 2, 3],
                            borderWidth: 1
                        }]
                    },
                    options: {
                        scales: {
                            y: {
                                beginAtZero: true
                            }
                        }
                    }
                });
            }
        }
    }
</script>

Example: Plotting Incoming Data

It's unlikely, as per the first example, we just want to render static data - this is Node-RED after all. So as a quick example, we can also wire this example to a ui-slider for a quick demo, here's a flow that can get you started:

and what it'll look like when rendered to the Dashboard:

Example of a line chart plotting incoming data

Taking a deep-dive into the contents of the ui-template for this chart, we can see:

html
<template>
    <canvas ref="chart" />
</template>

<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>

<script>
    export default {
        mounted() {
            // register a listener for incoming data
            this.$socket.on('msg-input:' + this.id, this.onInput)

            // check with ChartJS has loaded
            let interval = setInterval(() => {
                if (window.Chart) {
                    // clear the check for ChartJS
                    clearInterval(interval);
                    // draw our initial chart
                    this.draw()
                }
            }, 100);
        },
        methods: {
            draw () {
                // get reference to the <canvas /> element
                const ctx = this.$refs.chart
                
                // Render the chart
                const chart = new Chart(ctx, {
                    type: 'line',
                    data: {
                        datasets: [{
                            label: "My Label",  // label for the single line we'll render
                            data: []            // start with no data
                        }]
                    },
                    options: {
                        animation: false, // don't run the animation for incoming data
                        responsive: true, // ensure we auto-resize the content
                        scales: {
                            x: {
                                type: 'time' // in this example, we're rendering timestamps
                            }
                        },
                        parsing: {
                            xAxisKey: 'time', // the property to render on the x-axis
                            yAxisKey: 'value' // the property to render on the y-axis
                        },
                        plugins: {
                            legend: {
                                position: 'top',
                            },
                            title: {
                                display: true,
                                text: 'Chart.js Line Chart'
                            }
                        }   
                    },
                });
                // make this available to all elements of the component
                this.chart = chart
            },
            onInput (msg) {
                // add a new data point ot our existing dataset
                this.chart.data.datasets[0].data.push({
                    time: (new Date()).getTime(),
                    value: msg.payload
                }) 
                // ensure the chart re-renders
                this.chart.update()      
            }
        }
    }
</script>

Example: Categorising Data

Let's take a more complex example, where we can render a chart type that we don't currently support in core Dashboard, a Polar Area Chart.

Example of a bar chart categorising incoming data

This example is adapted from this example from the ChartJS documentation

In this example, we wire multiple ui-sliders, each defining a msg.topic of a different color, into our custom chart:

A deep-dive into the contents of the ui-template shows:

html
<template>
    <canvas ref="chart" />
</template>

<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>

<script>
    export default {
        mounted() {
            // register a listener for incoming data
            this.$socket.on('msg-input:' + this.id, this.onInput)

            // code here when the component is first loaded
            let interval = setInterval(() => {
                if (window.Chart) {
                    // Babylon.js is loaded, so we can now use it
                    clearInterval(interval);
                    this.draw()
                }
            }, 100);
        },
        methods: {
            draw () {
                const ctx = this.$refs.chart
                const data = {
                    labels: [],
                    datasets: [{
                        label: 'Colors',
                        data: [],
                        backgroundColor: []
                    }]
                }
                
                // Render the chart
                const chart = new Chart(ctx, {
                    type: 'polarArea',
                    data: data,
                    options: {
                        responsive: true,
                        scales: {
                            r: {
                                pointLabels: {
                                    display: true,
                                    centerPointLabels: true,
                                    font: {
                                        size: 18
                                    }
                                }
                            }
                            },
                            plugins: {
                            legend: {
                                position: 'top',
                            },
                            title: {
                                display: true,
                                text: 'Chart.js Polar Area Chart With Centered Point Labels'
                            }
                        }
                    },
                });
                this.chart = chart
            },
            onInput (msg) {
                // in this example, our topics will be colors
                const color = msg.topic

                // have we seen this color before?
                const index = this.chart.data.labels.indexOf(color)
                
                if (index === -1) {
                    console.log('new color', color)
                    // add new dataset for this topic
                    this.chart.data.labels.push(color)
                    this.chart.data.datasets[0].data.push(msg.payload)
                    this.chart.data.datasets[0].backgroundColor.push(color)
                } else {
                    // we've already got data for this color, update the value
                    this.chart.data.datasets[0].data[index] = msg.payload
                }

                // ensure the chart re-renders
                this.chart.update()      
            }
        }
    }
</script>