diff --git a/docs/assets/images/design-pattern-client.png b/docs/assets/images/design-pattern-client.png
new file mode 100644
index 000000000..add932262
Binary files /dev/null and b/docs/assets/images/design-pattern-client.png differ
diff --git a/docs/assets/images/design-pattern-single.png b/docs/assets/images/design-pattern-single.png
new file mode 100644
index 000000000..605347a50
Binary files /dev/null and b/docs/assets/images/design-pattern-single.png differ
diff --git a/docs/assets/videos/demo-design-patterns.mov b/docs/assets/videos/demo-design-patterns.mov
new file mode 100644
index 000000000..4722f3bdb
Binary files /dev/null and b/docs/assets/videos/demo-design-patterns.mov differ
diff --git a/docs/assets/videos/demo-design-patterns.mp4 b/docs/assets/videos/demo-design-patterns.mp4
new file mode 100755
index 000000000..ed27c5e07
Binary files /dev/null and b/docs/assets/videos/demo-design-patterns.mp4 differ
diff --git a/docs/getting-started.md b/docs/getting-started.md
index a8dc613a9..a0ba1b9c1 100644
--- a/docs/getting-started.md
+++ b/docs/getting-started.md
@@ -2,6 +2,17 @@
description: Kickstart your Node-RED Dashboard 2.0 journey with this getting started guide. Perfect for beginners.
---
+
+
+
# Getting Started
## Installation
@@ -43,7 +54,7 @@ Notice too that Dashboard will automatically create a new group, page, theme and
## Configuring your layout
-Dashboard 2.0 adds a correspinding "Dashboard 2.0" sidebar to the Node-RED editor . This sidebar provides an interface by which to view of your pages, themes, groups and widgets. From here you can add new pages and groups, modify existing settings, and re-order content to your liking.
+Dashboard 2.0 adds a corresponding "Dashboard 2.0" sidebar to the Node-RED editor . This sidebar provides an interface by which to view of your pages, themes, groups and widgets. From here you can add new pages and groups, modify existing settings, and re-order content to your liking.
![Screenshot showing the Dashboard 2.0 sidebar in the Node-RED Editor.](./assets/images/getting-started-sidebar.png){data-zoomable}
_Screenshot showing the Dashboard 2.0 sidebar in the Node-RED Editor._
@@ -78,7 +89,7 @@ Currently, we have three different options for a page's layout:
### Navigation Sidebar
-Built into the framework of the UI is a side navigation bar, along with the top, page-wide "app bar". Configuration options exist such that the side navigation behavior can be controlled. Options include:
+Built into the framework of the UI is a side navigation bar, along with the top, page-wide "app bar". Configuration options exist such that the side navigation behaviour can be controlled. Options include:
- **Collapsing:** When the sidebar is opened the page content will adjust with the width of the sidebar.
- **Fixed:** The full sidebar will always be visible, and the page content will adjust to the width of the sidebar.
@@ -86,6 +97,78 @@ Built into the framework of the UI is a side navigation bar, along with the top,
- **Appear over Content:** When the sidebar is opened, the page is given an overlay, and the sidebar sits on top.
- **Always Hide:** The sidebar will never show, and navigation between pages can instead be driven by [`ui-control`](https://dashboard.flowfuse.com/nodes/widgets/ui-control.html).
+## Design Patterns
+
+There are two core Design Patterns that are possible when building with Dashboard 2.0:
+
+- **Single Source of Truth:** All users of your Dashboard will see the same data. This is useful for industrial IoT or Home Automation applications.
+- **Client-Driven Data:** Data shown in a particular widget is unique to a given client/session/user. This represents a more traditional web application, where each user has their own session and associated data.
+
+It's worth noting that these two patterns can be mixed and matched within a single Dashboard 2.0 application, shown [later](#example).
+
+### Single Source of Truth
+
+![Single Source of Truth](./assets/images/design-pattern-single.png){data-zoomable}
+_Example workflow to demonstrate the "Single Source of Truth" design pattern._
+
+This is the pattern that the original Node-RED Dashboard utilized. In this pattern, all users of the Dashboard will see the same data. Data that populates a widget is generally driven by a piece of hardware or general-purpose API call.
+
+When a user goes to visit a Dashboard, the widgets will load their respective state, and will display it to every user.
+
+An example of this is that if you have interactive elements, e.g. a slider linked to a chart, then one user moving the slider will draw data to the chart of every other user's Dashboards too.
+
+### Client-Driven Data
+
+![Client-Driven Data](./assets/images/design-pattern-client.png){data-zoomable}
+_Example workflow to demonstrate the "Client-Driven Data" design pattern._
+
+In Dashboard 2.0 we can configure a given node type to ["Accept Client Data"](/user/sidebar.html#client-data) from the sidebar:
+
+
+Screenshot of an example "Client Data" tab
+
+If "Include Client Data" is toggled on, then _all_ `msg` objects emitted from _all_ nodes will contain a `msg._client` object, which will at a minimum detail the `socketId` for the connected client. It is possible to add on more data to this object, such as a username, email address, or other unique identifier with Dashboard plugins, e.g. the [FlowFuse User Plugin](https://flowfuse.com/blog/2024/04/displaying-logged-in-users-on-dashboard/).
+
+The "Accept Client Data" table allow configuration over which node types will pay attention to any provided `msg._client` information. Any `msg` sent _to_ one of these nodes can include a `msg._client` value to specify a particular connection (e.g. username, socket ID) that the data should be sent to, rather than to all clients.
+
+For users familiar with the original Node-RED Dashboard, you'll recognise this pattern from what you could do with `ui-notification` and `ui-control`, now, in Dashboard 2.0, it's possible for _all_ widgets.
+
+The key here is that data is generally injected into a node as a consequence of a user action, e.g. clicking a button, viewing a page, or submitting a form, and the responding data is sent _only_ back to that user.
+
+An easy example of this design pattern in Dashboard 2.0 is to utilise the [UI Event](./nodes/widgets/ui-event.md) node. The `ui-event` node emits a `msg` when a user loads a page. Within the `msg` is a full `msg._client` data object available for that client's connection. If this message is then sent onto another node that accept client data, then that full `msg` will _only_ be sent to that specified client.
+
+### Example
+
+Here we have a flow that will produce some client-defined data, and some shared data. When importing, be sure to check that in the Dashboard 2.0 sidebar, both `ui-text` and `ui-template` are checked in the "Accepts Client Data" table.
+
+
+
+In the video above we see that in some cases, data is sent to just the client that triggered it (e.g. button clicks), and in others, the data is shared across all client sessions (e.g. the visualisation of slider value on the chart).
+
+If you'd like to play with this example, the flow is as follows:
+
+
+
+To cover a little more detail about the flow itself:
+
+#### Client-Driven Data
+
+For this use-case we've set `ui-text` and `ui-template` configured in the sidebar to "Accept Client Constraints".
+
+In the top-half, the `ui-event` node will emit a message when a user loads the page. This message will contain a `msg._client` object, which is unique to that user's connection. This message is then sent to a `ui-template` node, which will display the socket ID of the specific user.
+
+Similarly, we also have a button, which will also emit `msg._client` data (as all nodes will do), but this time it will be sent to a `ui-text` node. The `ui-text` will show the timestamp of the last time that the given client/user clicked that button.
+
+
+#### Shared Data (All Clients)
+
+This section of the flow demonstrates how a slider can be used to control a chart, note that we wire the slider straight into the chart because the `ui-chart` has not been configured to "Accept Client Data".
+
+We also connect the `ui-slider` to two `ui-template` nodes. Given that `ui-template` nodes _are_ configured to "Accept Client Data", we can demonstrate both shared and client-specific data in the same flow by deleting `msg._client` data on the way to the lower `ui-template` node. By removing this, any slider data sent here will be sent to _all_ connections, because the `msg` doesn't specify a `_client`. The top `ui-template` will only update for the client that moved the slider.
+
## Contributing
If you would like to run this set of nodes locally, and specifically to help contribute to the development efforts, you can read the [Contributing](./contributing/index.md) documentation.
diff --git a/examples/design-patterns.json b/examples/design-patterns.json
new file mode 100644
index 000000000..af96bc4b9
--- /dev/null
+++ b/examples/design-patterns.json
@@ -0,0 +1,356 @@
+[
+ {
+ "id": "f385539f963b56ce",
+ "type": "ui-text",
+ "z": "3d8c801ff2007261",
+ "group": "2b287eac8c5a64cd",
+ "order": 2,
+ "width": "3",
+ "height": "1",
+ "name": "",
+ "label": "Your Latest Button Click:",
+ "format": "{{msg.payload}}",
+ "layout": "row-left",
+ "style": false,
+ "font": "",
+ "fontSize": 16,
+ "color": "#717171",
+ "className": "",
+ "x": 290,
+ "y": 120,
+ "wires": []
+ },
+ {
+ "id": "ae23d23cc164d27a",
+ "type": "ui-button",
+ "z": "3d8c801ff2007261",
+ "group": "2b287eac8c5a64cd",
+ "name": "",
+ "label": "Click Me!",
+ "order": 3,
+ "width": 0,
+ "height": 0,
+ "emulateClick": false,
+ "tooltip": "",
+ "color": "",
+ "bgcolor": "",
+ "className": "",
+ "icon": "",
+ "iconPosition": "left",
+ "payload": "",
+ "payloadType": "date",
+ "topic": "topic",
+ "topicType": "msg",
+ "x": 100,
+ "y": 120,
+ "wires": [
+ [
+ "f385539f963b56ce"
+ ]
+ ]
+ },
+ {
+ "id": "a7d3cb9fc537d9bb",
+ "type": "ui-slider",
+ "z": "3d8c801ff2007261",
+ "group": "8bee8ca608b26b77",
+ "name": "",
+ "label": "slider",
+ "tooltip": "",
+ "order": 1,
+ "width": 0,
+ "height": 0,
+ "passthru": false,
+ "outs": "all",
+ "topic": "topic",
+ "topicType": "msg",
+ "thumbLabel": true,
+ "min": 0,
+ "max": 10,
+ "step": 1,
+ "className": "",
+ "x": 150,
+ "y": 260,
+ "wires": [
+ [
+ "257625ee5df3a84e",
+ "ea03c8bc066ebdf2",
+ "5ef058757d792100"
+ ]
+ ]
+ },
+ {
+ "id": "257625ee5df3a84e",
+ "type": "ui-chart",
+ "z": "3d8c801ff2007261",
+ "group": "8bee8ca608b26b77",
+ "name": "",
+ "label": "chart",
+ "order": 3,
+ "chartType": "line",
+ "category": "topic",
+ "categoryType": "msg",
+ "xAxisProperty": "",
+ "xAxisPropertyType": "msg",
+ "xAxisType": "time",
+ "yAxisProperty": "",
+ "ymin": "",
+ "ymax": "",
+ "action": "append",
+ "pointShape": "circle",
+ "pointRadius": 4,
+ "showLegend": true,
+ "removeOlder": 1,
+ "removeOlderUnit": "3600",
+ "removeOlderPoints": "",
+ "colors": [
+ "#1f77b4",
+ "#aec7e8",
+ "#ff7f0e",
+ "#2ca02c",
+ "#98df8a",
+ "#d62728",
+ "#ff9896",
+ "#9467bd",
+ "#c5b0d5"
+ ],
+ "width": "9",
+ "height": 8,
+ "className": "",
+ "x": 330,
+ "y": 320,
+ "wires": [
+ []
+ ]
+ },
+ {
+ "id": "ce9b4c9ec9c4ed93",
+ "type": "inject",
+ "z": "3d8c801ff2007261",
+ "name": "Clear Chart",
+ "props": [
+ {
+ "p": "payload"
+ }
+ ],
+ "repeat": "",
+ "crontab": "",
+ "once": false,
+ "onceDelay": 0.1,
+ "topic": "",
+ "payload": "[]",
+ "payloadType": "json",
+ "x": 130,
+ "y": 320,
+ "wires": [
+ [
+ "257625ee5df3a84e"
+ ]
+ ]
+ },
+ {
+ "id": "8aefa358fdb6e177",
+ "type": "ui-event",
+ "z": "3d8c801ff2007261",
+ "ui": "c2e1aa56f50f03bd",
+ "name": "",
+ "x": 100,
+ "y": 80,
+ "wires": [
+ [
+ "0b8294025998e4be"
+ ]
+ ]
+ },
+ {
+ "id": "0b8294025998e4be",
+ "type": "ui-template",
+ "z": "3d8c801ff2007261",
+ "group": "2b287eac8c5a64cd",
+ "page": "",
+ "ui": "",
+ "name": "",
+ "order": 1,
+ "width": 0,
+ "height": 0,
+ "head": "",
+ "format": "\n msg._client:\n