Widget Registration
Every ui-base
, ui-page
and ui-group
has a .register
function. The core registration function can be found in ui-base
.
This function is used by all of the widgets to inform Dashboard of their existence, and allows the widget to define which group/page/ui it belongs too, along with the relevant properties that widget has and any event handlers (e.g. onInput
or onAction
).
The function is called within the node's Node-RED .js
file, and in th case of a widget registering as part of a group (the most common use case), would look something like this:
module.exports = function (RED) {
function MyNode (config) {
// create node in Node-RED
RED.nodes.createNode(this, config)
// store reference to our Node-RED node
const node = this
// which group are we rendering this widget
const group = RED.nodes.getNode(config.group)
// an object detailing the events to subscribe to
const evts = {}
// inform the dashboard UI that we are adding this node
group.register(node, config, evts)
}
RED.nodes.registerType('ui-mywidget', MyNode)
}
Arguments
The registration function inputs differ slightly depending on whether being called on the ui-group
, ui-page
or ui-base
:
group.register(node, config, evts)
page.register(group, node, config, evts)
base.register(page, group, node, config, evts)
Note though, they do all have 3 inputs in common:
node
This is the this
of your node's constructor, and can be used directly from the value provided from Node-RED.
config
This is made available by Node-RED as the input to the constructor, and can generally passed straight into the .register
function without modification, it will be an object that maps all of the properties and values that have been described in the node's .html
definition.
evts
We expose a range of different event handlers as part of the register
function. All of these handlers run server (Node-RED) side.
In some cases, it is possible to define full functions (that will run at the appropriate point in the event lifecycle), in other occasions, it's only possible to define a true
/false
value that informs Dashboard that you wish for the widget to send or subscribe to that event.
A full breakdown of the event lifecycle can be found here.
const evts = {
onAction: // boolean
onChange: // boolean || function
beforeSend: // function
onInput: // function
onError: // function
onSocket // object
}
Events
All of these event handlers define behaviour that is run server-side (i.e. within Node-RED). If you're looking for client-side event handlers see here.
.onAction
(boolean
)
When set as true
, this flag will trigger the default handler when the Dashboard widgets sends an widget-action
event.
- Assigns the provided value to
msg.payload
- Appends any
msg.topic
defined on the node config - Runs
evts.beforeSend()
(if provided) - Sends the
msg
onwards to any connected nodes usingnode.send(msg)
An example of this is with ui-button
, where the widget's UIButton
contains an @click
function, containing:
this.$socket.emit('widget-action', this.id, msg)
This sends a message via SocketIO to Node-RED, with the topic of the widget's ID. Because the ui-button
has onAction: true
in it's registration, it will consequently run the default handler detailed above.
.onChange
(boolean
|| function
)
Similar to onAction
, when used as a boolean, this flag will trigger the default handler for an onChange
event.
Default onChange
Handler
- Assigns the provided value to
msg.payload
- Appends any
msg.topic
defined on the node config - Runs
evts.beforeSend()
(if provided) - Store the most recent message on the widget under the
._msg
property which will contain the latest state/value of the widget - Pushes a
widget-sync
event to synchronize the widgets in all clients. - Sends the
msg
onwards to any connected nodes
Custom onChange
Handler
Alternatively, you can override this default behaviour by providing a custom onChange
function. An example of this is in the ui-switch
node which needs to do node.status
updates to in order for the Node-RED Editor to reflect it's latest status:
/**
* Handle the input from the widget
* @param {object} msg - the last known msg received (prior to this new value)
* @param {boolean} value - the updated value sent by the widget
* @param {Socket} conn - socket.io socket connecting to the server
* @param {String} id - widget id sending the action
*/
onChange: async function (msg, value, conn, id) {
// ensure we have latest instance of the widget's node
const wNode = RED.nodes.getNode(node.id)
node.status({
fill: value ? 'green' : 'red',
shape: 'ring',
text: value ? states[1] : states[0]
})
// retrieve the assigned on/off value
const on = RED.util.evaluateNodeProperty(config.onvalue, config.onvalueType, wNode)
const off = RED.util.evaluateNodeProperty(config.offvalue, config.offvalueType, wNode)
msg.payload = value ? on : off
// sync this change to all clients with the same widget
const exclude = [conn.id]
base.emit('widget-sync:' + id, msg, node, exclude)
// simulate Node-RED node receiving an input
wNode.send(msg)
}
.beforeSend(msg)
(function
)
This middleware function will run before the node sends any msg
to consequent nodes connected in the Editor (e.g. in onInput
, onAction
and onChange
default handlers).
The function must take msg
as an input, and also return msg
as an output.
In ui-button
, we use beforeSend
evaluate the msg.payload
as we have a TypedInput
(docs. The TypedInput
needs evaluating within Node-RED, as it can reference variables outside of the domain of the button's node (e.g. global
or flow
). The default onInput
handler then takes the output from our beforeSend
and processes it accordingly.
.onInput(msg, send)
(function
)
Defining this function will override the default onInput
handler.
Default onInput
Handler
- Store the most recent message on the widget under the
node._msg
- Appends any
msg.topic
defined on the node config - Checks if the widget has a
passthru
property:- If no
passthru
property is found, runssend(msg)
- If the property is present,
send(msg)
is only run ifpassthru
is set totrue
- If no
Custom onInput
Handler
When provided, this will override the default handler.
We use this in the core widgets in Dashboard with ui-chart
, where we want to be storing the history of recent msg
value, rather than just the most recent value as done in the default handler. We also use it here to ensure we don't have too many data points (as defined in the ui-chart
config).
Another use case here would be if you do not want to pass on any incoming msg
payloads onto connected nodes automatically, for example, you could have a bunch of command-type msg
payloads that instruct your node to do something, that are then not relevant to any preceding nodes in the flow.
.onError(err)
(function
)
This function is called within the handlers for onAction
, onChange
and onInput
. If there is ever an issue with these handlers (including those custom handlers provided), then the onError
function will be called.
.onSocket
(object
)
This is a somewhat unique event handler, that is only used by externally developed widgets (i.e. not part of core Dashboard widgets detailed in this documentation). It is provided so that developers can emit
, and consequently subscribe to, custom SocketIO events that are transmitted by their custom widgets.
You can see a more detailed example in our documentation here.
The general structure of onSocket
is as follows:
const evts = {
onSocket: {
'my-custom-event': function (id, msg) {
console.log('my-custom-event', id, msg)
}
}
}
Note that these events are emitted from the Dashboard, and so, these handlers are run within Node-RED.