From 0c96e9bbf162b7a4bca569c4a418918c6a18d2a1 Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Tue, 23 Apr 2024 16:52:49 +0100 Subject: [PATCH 1/4] Include functional components. Update README.md with usage examples, acknowledgements and license info Add license Update package.json Add a demo page Rename gitignore to activate it --- gitignore => .gitignore | 0 LICENSE | 177 ++++ README.md | 82 +- demo.html | 94 ++ index.js | 2116 +++++++++++++++++++++++++++++++++++++++ package.json | 2 +- 6 files changed, 2468 insertions(+), 3 deletions(-) rename gitignore => .gitignore (100%) create mode 100644 LICENSE create mode 100644 demo.html create mode 100644 index.js diff --git a/gitignore b/.gitignore similarity index 100% rename from gitignore rename to .gitignore diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..f433b1a --- /dev/null +++ b/LICENSE @@ -0,0 +1,177 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS diff --git a/README.md b/README.md index c780e86..94790cd 100644 --- a/README.md +++ b/README.md @@ -5,13 +5,91 @@ A library to render Node-RED flows in a web page. ## Installation +```bash +npm install @flowfuse/flow-renderer +``` + +## development + +All client-side code is in the `index.js` file for easy inclusion in a web page. + +Run `npm run demo` to test the flow renderer in a browser. ## Usage +Add the following script tag to your HTML file: + +```html + +``` +_or wherever the script is located in your project_ + +Next, add a container element to your HTML file and call the `flowRenderer` function with the flow data and options. + +By default, the flow renderer will render the flow with `gridlines`, `images`, `labels` and `zoom` options enabled. + +To operate the zoom, use the mouse wheel + Ctrl key. +To scroll the container vertically, use the mouse wheel without the Shift key. +To scroll the container horizontally, use the mouse wheel + Shift key. + +### Basic example + +```html +
+``` + +```javascript +const container1 = document.getElementById('nr-flow-1'); +const flow = [{"id": "1001", "type": "inject", "x": 100, "y": 40, "wires": [["1002"]]}, {"id": "1002", "type": "debug", "x":300, "y": 40}] +flowRenderer(flow, { container: container1 }) +``` + +### Inline Options example + +Options can be set by data attributes `scope`, `grid-lines`, `zoom`, `images`, `link-lines`, `labels` + +```html +
+``` + +```javascript +const container2 = document.getElementById('nr-flow-2'); +const flow = [{"id": "1001", "type": "inject", "x": 100, "y": 40, "wires": [["1002"]]}, {"id": "1002", "type": "debug", "x":300, "y": 40}] +flowRenderer(flow, { container: container2 }) +``` + +### Full Options example + +```html +
+``` + +```javascript +const container3 = document.getElementById('nr-flow-3'); +const flow = [{"id": "1001", "type": "inject", "x": 100, "y": 40, "wires": [["1002"]]}, {"id": "1002", "type": "debug", "x":300, "y": 40}] +flowRenderer(flow, { + container: container3, + scope: 'my-scope', // scope for CSS + gridlines: true, // show gridlines + images: true, // show images + linkLines: false, // show link lines + labels: true, // show labels + zoom: true, // enable zoom within the container + flowId: undefined // Id of flow to display +}) +``` + +## Acknowledgements + +This project owes a huge thanks to Gerrit Riessen for his original works on [node-red-flowviewer](https://github.com/gorenje/node-red-flowviewer-js). It was this great contribution that started the ball rolling. Gerrit kindly allowed us relicense the parts we needed to use in this project. -## Contributing +## Limitations +* Only client-side rendering is currently supported +* Mobile pinch zoom is not yet implemented +* The flow renderer does not support the full range of contributed Node-RED nodes however they will render as a generic node type complete with the node's label. +* The flow renderer does not always render the flows and nodes exactly as they appear in the Node-RED editor. This is due in part to being a client-side render with no server-side component to provide full context and partly due to the current limitations of the renderer itself. ## License -TODO \ No newline at end of file +Apache-2.0 diff --git a/demo.html b/demo.html new file mode 100644 index 0000000..4a1b7a7 --- /dev/null +++ b/demo.html @@ -0,0 +1,94 @@ + + + + + + FlowFuse Flow Renderer + + + + + + + + +

FlowFuse Flow Renderer

+
Client side demo
+
+ + +

Node-RED Flow

+ +
+ + +
+ + +

Render (grid, images, labels)

+
+ +
+
+ + +

Render (no grid, no images, labels)

+
+ +
+
+ + +

Render (no grid, images, no labels) using data-options

+
+ + + + + + \ No newline at end of file diff --git a/index.js b/index.js new file mode 100644 index 0000000..6759064 --- /dev/null +++ b/index.js @@ -0,0 +1,2116 @@ + +; (function () { + + if (typeof exports === 'object') { + module.exports = flowRenderer; + } else if (typeof define === 'function' && define.amd) { + define(function () { return flowRenderer; }); + } else { + this.flowRenderer = flowRenderer; + } + + // #region Constants + const styleId = 'flow-renderer-css' + const PORT_WIDTH = 10 + + const portDimensions = { + width: PORT_WIDTH, + height: PORT_WIDTH + } + + const portRadius = { + rx: 3, + ry: 3 + } + + const widthHeightByType = { + "junction": { width: 10, height: 10}, + "link in": { width: 30, height: 30}, + "link out": { width: 30, height: 30}, + "subflow": { width: 40, height: 40}, + "_default": { width: 100, height: 30}, + "_default_no_label": { width: 30, height: 30}, + } + + const imgByType = { + "batch": "batch.svg", + "catch": "alert.svg", + "change": "swap.svg", + "complete": "alert.svg", + "rbe": "rbe.png", + "comment": "comment.svg", + "csv": "parser-csv.svg", + "debug": "debug.svg", + "delay": "timer.svg", + "exec": "cog.svg", + "feedparse": "parser-xml.svg", + "file": "file.svg", + "file in": "file-in.svg", + "file out": "file-out.svg", + "function": "function.svg", + "http response": "white-globe.svg", + "http in": "white-globe.svg", + "http request": "white-globe.svg", + "inject": "inject.svg", + "join": "join.svg", + "json": "parser-json.svg", + "link in": "link-out.svg", + "link out": "link-out.svg", + "link outlink": "link-out.svg", + "link outreturn": "link-return.svg", + "link call": "link-call.svg", + "xml": "parser-xml.svg", + "yaml": "parser-yaml.svg", + + "mqtt in": "bridge.svg", + "mqtt out": "bridge.svg", + "markdown": "parser-markdown.png", + + "postgresql": "db.svg", + "range": "range.svg", + "sort": "sort.svg", + "split": "split.svg", + "subflow": "subflow.svg", + "switch": "switch.svg", + "template": "template.svg", + "trigger": "trigger.svg", + + // dashboard 1.0 + ui_button: "ui_button.png", + ui_template: "ui_template.png", + ui_toast: "ui_toast.png", + ui_audio: "feed.svg", + ui_chart: "ui_chart.png", + ui_colour_picker: "ui_colour_picker.png", + ui_date_picker: "ui_date_picker.png", + ui_dropdown: "ui_dropdown.png", + ui_form: "ui_form.png", + ui_gauge: "ui_gauge.png", + ui_numeric: "ui_numeric.png", + ui_slider: "ui_slider.png", + ui_switch: "ui_switch.png", + ui_text: "ui_text.png", + ui_text_input: "ui_text.png", + + // dashboard 2.0 + "ui-button": "ui_button.png", + "ui-button-group": "ui_button.png", + "ui-chart": "ui_chart.png", + // "ui-control": "ui_control.png", + "ui-dropdown": "ui_dropdown.png", + // "ui-event": "ui_event.png", + "ui-form": "ui_form.png", + "ui-gauge": "ui_gauge.png", + "ui-markdown": "ui_text.png", // temporary + // "ui-notification": "", + // "ui-radio-group": "", + "ui-slider": "ui_slider.png", + "ui-switch": "ui_switch.png", + // "ui-table": "", + "ui-template": "ui_template.png", + "ui-text": "ui_text.png", + "ui-text-input": "ui_text.png", + + "websocket in": "white-globe.svg", + "websocket out": "white-globe.svg", + + "i2c scan": "serial.svg", + "i2c in": "serial.svg", + "i2c out": "serial.svg", + + "cronplus": "timer.svg", + + /* custom, my own private collection */ + "Thought": "alert.svg", + "Idea": "light.svg", + + // FlowFuse specific + "project link out": "ff-logo.svg", + "project link in": "ff-logo.svg", + "project link call": "ff-logo.svg" + } + + const imageNameToContent = { + "alert.svg": "", + "arduino.png": "", + "arrow-in.svg": "", + "batch.svg": "", + "bluetooth.png": "", + "bridge-dash.svg": "", + "bridge.svg": "", + "cog.svg": "", + "comment.svg": "", + "db.svg": "", + "debug.svg": "", + "envelope.svg": "", + "feed.svg": "", + "file-in.svg": "", + "file-out.svg": "", + "file.svg": "", + "function.svg": "", + "hash.svg": "", + "inject.svg": "", + "join.svg": "", + "leveldb.png": "", + "light.svg": "", + "link-call.svg": "", + "link-out.svg": "", + "link-return.svg": "", + "mongodb.png": "", + "mouse.png": "", + "parser-csv.svg": "", + "parser-html.svg": "", + "parser-json.svg": "", + "parser-xml.svg": "", + "parser-yaml.svg": "", + "range.svg": "", + "rbe.png": "", + "redis.png": "", + "rpi.svg": "", + "serial.svg": "", + "sort.svg": "", + "split.svg": "", + "status.svg": "", + "subflow.svg": "", + "swap.svg": "", + "switch.svg": "", + "template.svg": "", + "timer.svg": "", + "trigger.svg": "", + "watch.svg": "", + "white-globe.svg": "", + "parser-markdown.png": "", + "ui_button.png": "", + "ui_chart.png": "", + "ui_colour_picker.png": "", + "ui_date_picker.png": "", + "ui_dropdown.png": "", + "ui_form.png": "", + "ui_gauge.png": "", + "ui_numeric.png": "", + "ui_slider.png": "", + "ui_switch.png": "", + "ui_template.png": "", + "ui_text.png": "", + "ui_toast.png": "", + "ff-logo.svg": "", + } + + const clrByType = { + "base64": _hshClr("#DEBD5C"), + "batch": _hshClr("#E2D96E"), + "catch": _hshClr("#e49191"), + "change": _hshClr("#E2D96E"), + "rbe": _hshClr("#E2D96E"), + "complete": _hshClr("#C0EDC0"), + "comment": _hshClr(), + "csv": _hshClr("#DEBD5C"), + "debug": _hshClr("#87a980"), + "delay": _hshClr("#E6E0F8"), + "exec": _hshClr("darksalmon"), + "feedparse": _hshClr("#C0DEED"), + "file": _hshClr("BurlyWood"), + "file in": _hshClr("BurlyWood"), + "function": _hshClr("#fdd0a2"), + "html": _hshClr("#DEBD5C"), + "http response": _hshClr("rgb(231, 231, 174)"), + "http in": _hshClr("rgb(231, 231, 174)"), + "http request": _hshClr("rgb(231, 231, 174)"), + "inject": _hshClr("#a6bbcf"), + "join": _hshClr("#E2D96E"), + "json": _hshClr("#DEBD5C"), + "junction": _hshClr('rgb(217, 217, 217)'), + "link in": _hshClr("#ddd"), + "link out": _hshClr("#ddd"), + "link call": _hshClr("#ddd"), + "mqtt in": _hshClr("#d8bfd8"), + "mqtt out": _hshClr("#d8bfd8"), + "markdown": _hshClr("#DEBD5C"), + "postgresql": _hshClr("#5b85a7"), + "range": _hshClr("#E2D96E"), + "sort": _hshClr("#E2D96E"), + "split": _hshClr("#E2D96E"), + "subflow": _hshClr("#ddd"), + "switch": _hshClr("#E2D96E"), + "trigger": _hshClr("#E6E0F8"), + "template": _hshClr("rgb(243, 181, 103)"), + + // Dashboard 1.0 + ui_button: _hshClr("rgb(176, 223, 227)"), + ui_list: _hshClr("rgb( 63, 173, 181)"), + ui_svg_graphics: _hshClr("rgb( 63, 173, 181)"), + ui_template: _hshClr("rgb( 63, 173, 181)"), + ui_toast: _hshClr("rgb(119, 198, 204)"), + ui_upload: _hshClr("rgb( 63, 173, 181)"), + ui_audio: _hshClr("rgb(119, 198, 204)"), + ui_chart: _hshClr("rgb(119, 198, 204)"), + ui_gauge: _hshClr("rgb(119, 198, 204)"), + ui_text: _hshClr("rgb(119, 198, 204)"), + ui_date_picker: _hshClr("rgb(176, 223, 227)"), + ui_dropdown: _hshClr("rgb(176, 223, 227)"), + ui_form: _hshClr("rgb(176, 223, 227)"), + ui_numeric: _hshClr("rgb(176, 223, 227)"), + ui_slider: _hshClr("rgb(176, 223, 227)"), + ui_switch: _hshClr("rgb(176, 223, 227)"), + ui_text_input: _hshClr("rgb(176, 223, 227)"), + ui_colour_picker: _hshClr("rgb(176, 223, 227)"), + + // Dashboard 2.0 + "ui-button": _hshClr("rgb(160, 230, 236)"), + "ui-button-group": _hshClr("rgb(160, 230, 236)"), + "ui-chart": _hshClr("rgb(90, 210, 220)"), + "ui-control": _hshClr("rgb(32, 160, 170)"), + "ui-dropdown": _hshClr("rgb(160, 230, 236)"), + "ui-event": _hshClr("rgb(32, 160, 170)"), + "ui-form": _hshClr("rgb(160, 230, 236)"), + "ui-gauge": _hshClr("rgb(90, 210, 220)"), + "ui-markdown": _hshClr("rgb(39, 183, 195)"), + "ui-notification": _hshClr("rgb(90, 210, 220)"), + "ui-radio-group": _hshClr("rgb(160, 230, 236)"), + "ui-slider": _hshClr("rgb(160, 230, 236)"), + "ui-switch": _hshClr("rgb(160, 230, 236)"), + "ui-table": _hshClr("rgb(90, 210, 220)"), + "ui-template": _hshClr("rgb(39, 183, 195)"), + "ui-text": _hshClr("rgb(90, 210, 220)"), + "ui-text-input": _hshClr("rgb(160, 230, 236)"), + + "websocket in": _hshClr("rgb(215, 215, 160)"), + "websocket out": _hshClr("rgb(215, 215, 160)"), + "yaml": _hshClr("#DEBD5C"), + "xml": _hshClr("#DEBD5C"), + + /* private nodes for this instane */ + 'BlogPages': _hshClr("#ddeeff"), + 'BlogDetails': _hshClr("#ddeeff"), + 'BlogPageInfo': _hshClr("#ddeeff"), + 'BlogChanges': _hshClr("#ddeeff"), + 'PubMedium': _hshClr("#ddee44"), + + "Topic": _hshClr('#d0c9f6'), + "Observation": _hshClr('#f4adf3'), + "Question": _hshClr('#e0a4f3'), + "Thought": _hshClr('#cb9cf3'), + "Idea": _hshClr('#88baff'), + "Analogy": _hshClr('#86bfff'), + "Aphorism": _hshClr('#84c3ff'), + "Poesie": _hshClr('#83c7ff'), + "Humour": _hshClr('#81ccff'), + "Treasure": _hshClr('#7fd0ff'), + "Consequence": _hshClr('#f6c1cc'), + "Advantage": _hshClr('#efacbf'), + "Disadvantage": _hshClr('#e796b1'), + "Text": _hshClr('#c8ffb5'), + "Blog-Post": _hshClr('#d0fdc2'), + "Comment": _hshClr('#d9fcce'), + "Code-Base": _hshClr('#e1fbda'), + "Sketch": _hshClr('#e1fbda'), + "Inspiration": _hshClr('#dfdfb6'), + "Quote": _hshClr('#e5e5c0'), + "Definition": _hshClr('#eaebca'), + "Book": _hshClr('#f0f0d4'), + "Author": _hshClr('#f5f6de'), + + 'nnb-input-node': _hshClr('#ffefef'), + 'nnb-layer-node': _hshClr('#ffffef'), + 'nnb-output-node': _hshClr('#efefef'), + 'nnb-backprop': _hshClr('#e3edef'), + 'nnb-trainer': _hshClr('#e5e4ef'), + + 'Seeker': _hshClr('#e5e4ef'), + 'Sink': _hshClr('#e5e4ef'), + 'Screenshot': _hshClr('#e5e4ef'), + 'Orphans': _hshClr('#e5e4ef'), + 'IsMobile': _hshClr('#e5e4ef'), + 'Navigator': _hshClr('#e5e4ef'), + 'DrawSVG': _hshClr('#e5e4ef'), + 'GetFlows': _hshClr('#e5e4ef'), + 'SendFlow': _hshClr('#e5e4ef'), + 'TriggerImport': _hshClr('#e5e4ef'), + + "i2c scan": _hshClr('rgb(227, 82, 83)'), + "i2c in": _hshClr('rgb(227, 82, 83)'), + "i2c out": _hshClr('rgb(227, 82, 83)'), + + 'cronplus': _hshClr('#a6bbcf'), + + 'buffer-maker': _hshClr('#0090d4'), + 'buffer-parser': _hshClr('#0090d4'), + + 'FINS Read': _hshClr('#0090d4'), + 'FINS Write': _hshClr('#0090d4'), + 'FINS Read Multiple': _hshClr('#0090d4'), + 'FINS Transfer': _hshClr('#0090d4'), + 'FINS Fill': _hshClr('#0090d4'), + 'FINS Control': _hshClr('#0090d4'), + + 'MC Read': _hshClr('#0090d4'), + 'MC Write': _hshClr('#0090d4'), + + 'modbus-response': _hshClr('#E9967A'), + 'modbus-read': _hshClr('#E9967A'), + 'modbus-getter': _hshClr('#E9967A'), + 'modbus-write': _hshClr('#E9967A'), + 'modbus-flex-getter': _hshClr('#E9967A'), + 'modbus-flex-write': _hshClr('#E9967A'), + 'modbus-flex-server': _hshClr('#E9967A'), + 'modbus-flex-connector': _hshClr('#E9967A'), + 'modbus-response-filter': _hshClr('#E9967A'), + 'modbus-flex-sequencer': _hshClr('#E9967A'), + 'modbus-flex-fc': _hshClr('#E9967A'), + + 'OpcUa-Item': _hshClr('#3FADB5'), + 'OpcUa-Client': _hshClr('#3FADB5'), + 'OpcUa-Browser': _hshClr('#3FADB5'), + 'OpcUa-Server': _hshClr('#3FADB5'), + 'OpcUa-Event': _hshClr('#3FADB5'), + 'OpcUa-Method': _hshClr('#3FADB5'), + 'OpcUa-Rights': _hshClr('#3FADB5'), + 'OpcUa-Discovery': _hshClr('#3FADB5'), + + 's7 in': _hshClr('#3FADB5'), + 's7 out': _hshClr('#3FADB5'), + 's7 control': _hshClr('#3FADB5'), + + // FlowFuse specific + "project link out": _hshClr('#87D8CF'), + "project link in": _hshClr('#87D8CF'), + "project link call": _hshClr('#87D8CF'), + + "_default": _hshClr(), + } + + // #endregion + + // #region "Helper Functions" + + function getNodeImage (type) { + let imgName = imgByType[type] + if (!imgName) { + if (type.startsWith("subflow:")) { + imgName = imgByType["subflow"] + } else if (type.startsWith("ui-")) { + imgName = imgByType["ui-template"] + } else if (type.startsWith("ui_")) { + imgName = imgByType["ui_template"] + } + } + return imgName ? imageNameToContent[imgName] || "" : "" + } + + function getNodeColor (type) { + let colors = clrByType[type] + if (!colors) { + if (type.startsWith("subflow:")) { + colors = clrByType["subflow"] + } else if (type.startsWith("ui-")) { + colors = clrByType["ui-template"] + } else if (type.startsWith("ui_")) { + colors = clrByType["ui_template"] + } + } + return colors || clrByType["_default"] + } + + function _hshClr (fill, stroke) { + return { + fill: fill || "#ffffff", + stroke: stroke || "rgb(153, 153, 153)" + } + } + + function getDocument() { + if (typeof document !== "undefined") { + return document + } + if (typeof __document !== "undefined") { + return __document + } + + const args = Array.prototype.slice.call(arguments) || [this] + for (const arg of args) { + const argClassName = arg && arg.constructor && arg.constructor.name + if (argClassName === 'HTMLDocument') { + return arg + } else if (argClassName === 'HTMLDivElement' || argClassName === 'HTMLBodyElement' || argClassName === 'HTMLHtmlElement' || argClassName === 'HTMLElement') { + return arg.ownerDocument + } else if (argClassName === 'SVGElement' || argClassName === 'SVGSVGElement') { + return arg.ownerDocument + } + } + // if we reach here, return see if window is available + if (typeof window !== "undefined") { + return window.document + } + // if we reach here, return null + return null + } + + /** + * Create a SVG element with the given type and attributes + * @param {Document} doc - The document to create the SVG element in + * @param {String} type - The type of the SVG element + * @param {Object.} [attributes] - An object with the attributes of the SVG element in the form {attr: value} + * @returns + */ + function createSvgElement(type, attributes) { + const doc = getDocument(this) + const el = doc.createElementNS.apply(doc, ["http://www.w3.org/2000/svg", type]) + if (typeof attributes === "object") { + for (const attr in attributes) { + el.setAttribute(attr, attributes[attr]) + } + } + return el + } + + function clamp (val, min, max, def) { + const isNumber = (typeof val === "number") && !isNaN(val) + if (!isNumber) { + return def || min + } + return Math.min(Math.max(val, min), max) + } + + /** + * + * @param {SVGSVGElement} svg + * @param {HTMLElement} container + */ + function setupZoom(svg, container) { + let originalScale = 1 + const transform = svg.getAttribute('transform'); + if (transform) { + const scale = transform.match(/scale\(([^,]+)\)/); + if (scale) { + originalScale = clamp(parseFloat(scale[1] || 1), 0.25, 3, 1) + } + } + let modifiedScale = originalScale + // get the first child g element of the svg element + const mainSvgGroup = svg.querySelector('g') + mainSvgGroup.setAttribute('transform', `scale(${modifiedScale})`); + container.onwheel = function (e) { + if (e.ctrlKey) { + // zoom in/out + e.preventDefault() + const scale = modifiedScale - (Math.sign(e.deltaY) * 0.075) + modifiedScale = clamp(scale, 0.20, 3, 1) + mainSvgGroup.setAttribute('transform', `scale(${modifiedScale})`) + // recalculate the svg style height and width now that the scale has changed + svg.style.width = `${8000 * modifiedScale}px` + svg.style.height = `${8000 * modifiedScale}px` + } + } + } + + + function createDefaultDivContainer(container) { + const doc = getDocument(container, this) + const div = doc.createElement("div") + div.classList.add("red-ui-workspace-chart") + if (!container) { + return div + } + container.appendChild(div) + return div + } + + /** + * Generates a default SVG element with the required groups: + * + * ```html + * + * + * + * + * + * + * + * ``` + * @param {Document} doc + * @param {HTMLElement} [container] + */ + function createDefaultSVG(container) { + const doc = getDocument(container, this) + const svg = doc.createElementNS("http://www.w3.org/2000/svg", "svg") + const svgG = doc.createElementNS("http://www.w3.org/2000/svg", "g") + svgG.setAttribute("class", "outerContainer") + svg.setAttribute("style", "width:8000px; height:8000px;") + svg.appendChild(svgG) + const gs = ['flow_grid', 'flow_group_elements', 'flow_group_select', 'flow_wires', 'flow_nodes'] + for (let g of gs) { + const group = doc.createElementNS("http://www.w3.org/2000/svg", "g") + group.setAttribute("class", g) + svgG.appendChild(group) + } + // otherwise append the svg to the target + container.appendChild(svg) + return svg + } + + function getCSS(scope) { + // normalize the scope(s) + if (scope && (scope.length && typeof scope !== "string" && typeof scope[0] === "string")) { + scope = [...scope].join(" ") + } + scope = scope || ".flow-renderer" + scope = scope.split(" ").map(s => `.${s}`).join(" ").trim() + const css =` + :root { + --red-ui-primary-background: #fff; + --red-ui-secondary-text-color: #333; + --red-ui-view-grid-color: #eee; + --red-ui-view-border: 1px solid #bbbbbb; + --red-ui-node-border: #999; + --red-ui-node-background-default: #eee; + --red-ui-node-icon-background-color-fill: black; + --red-ui-node-icon-background-color-opacity: 0.1; + --red-ui-node-label-color: #333; + --red-ui-node-port-background: #d9d9d9; + --red-ui-link-color: #999; + } + ::-webkit-scrollbar { + width: 12px; + height: 12px; + } + ::-webkit-scrollbar-track { + -webkit-border-radius: 6px; + border-radius: 6px; + } + ::-webkit-scrollbar-thumb { + -webkit-border-radius: 6px; + border-radius: 6px; + -webkit-box-shadow: inset 0 0 12px rgba(0, 0, 0, 0.5); + } + + ${scope} .red-ui-workspace-chart { + box-sizing: border-box; + border: var(--red-ui-view-border); + overflow: scroll; + height: 100%; + width: 100%; + } + ${scope} svg { + position: relative; + width: 100%; + height: 100%; + min-height: 250px; + margin: auto; + display: block; + border-radius: 2px; + } + ${scope} svg { + cursor: default; + } + ${scope} svg .group-text-label { + font-family: Helvetica Neue, Arial, Helvetica, sans-serif; + font-size: 14px; + } + ${scope} svg .node-text-label { + font-family: Helvetica Neue, Arial, Helvetica, sans-serif; + font-size: 14px; + dominant-baseline: middle; + } + ${scope} svg .subflow-node-text-label { + color: rgb(85, 85, 85); + dominant-baseline: middle; + font-family: Helvetica Neue, Arial, Helvetica, sans-serif; + font-size: 10px; + line-height: 20px; + pointer-events: none; + text-anchor: middle; + user-select: none + } + ${scope} svg .subflow-node-text-label-number { + color: rgb(85, 85, 85); + dominant-baseline: middle; + font-family: Helvetica Neue, Arial, Helvetica, sans-serif; + font-size: 16px; + line-height: 20px; + pointer-events: none; + text-anchor: middle; + user-select: none + } + + ${scope} svg .node { + fill-opacity: 1; + stroke-width: 1px; + } + ${scope} svg .link { + stroke: #999; + stroke-width: 3; + fill: none; + } + ${scope} svg .link-highlight, .node-highlight { + stroke: rgb(255, 127, 14); + } + ${scope} svg .node-highlight { + stroke-width: 3px; + } + + ${scope} svg .node-disabled { + stroke-dasharray: 8,3; + fill-opacity: 0.5; + } + ${scope} svg .group-highlight { + stroke: rgb(255, 127, 14); + stroke-width: 4px; + fill: rgb(255, 127, 14); + fill-opacity: 0.2; + } + ${scope} svg .link-disabled { + stroke-dasharray: 10,8 !important; + stroke-width: 2 !important; + stroke: rgb(204, 204, 204); + } + ${scope} svg .grid-line { + shape-rendering: crispedges; + stroke: rgb(238, 238, 238); + stroke-width: 1px; + fill: none; + } + ${scope} svg .output-deco { + stroke-width: 2px; + stroke-miterlimit: 4; + } + ${scope} svg .input-deco { + stroke-width: 2px; + stroke-miterlimit: 4; + } + ${scope} svg .flow-render-error { + background-color: rgb(54, 52, 52); + color: rgb(196, 59, 59); + width: 100%; + } + ${scope} svg text { + user-select: none; + } + ${scope} .red-ui-tabs { + display: flex; + } + ${scope} .red-ui-tab { + padding: 6px 12px; + box-sizing: border-box; + display: block; + border: 1px solid #bbbbbb; + border-right: none; + background-color: #f0f0f0; + max-width: 200px; + width: 14%; + overflow: hidden; + white-space: nowrap; + position: relative; + margin-top: -1px; + transition: 0.2s background-color; + user-select: none; + position: relative; + z-index: 1; + top: 1px; + } + ${scope} .red-ui-tab-subflow-icon { + mask-image: url(); + display: inline-block; + background-color: grey; + margin-left: -8px; + margin-right: 3px; + margin-top: -1px; + margin-bottom: 1px; + opacity: 1; + width: 16px; + height: 18px; + vertical-align: middle; + mask-size: contain; + mask-position: center; + mask-repeat: no-repeat; + } + ${scope} .red-ui-tab span { + font-size: 0.875rem; + font-family: Helvetica Neue, Arial, Helvetica, sans-serif; + } + ${scope} .red-ui-tab.active { + background-color: white; + border-bottom-color: white; + font-weight: bold; + } + ${scope} .red-ui-tab:hover { + cursor: pointer; + background-color: white; + } + ${scope} .red-ui-tab:last-child { + border-right: 1px solid #bbbbbb; + } + ` + return css + } + + function addStyle(doc, scope) { + const nameSafeId = styleId + (scope ? '--' + scope.toString() : '').replace(/[^a-z0-9]/gi, '-').toLowerCase() + let style = doc.getElementById(nameSafeId) + if (!style) { + style = doc.createElement('style') + style.id = nameSafeId + style.innerHTML = getCSS(scope) + doc.head.appendChild(style) + } + } + + function resetSVG (svg) { + if (svg) { + // TODO: remove all children from the SCG then rebuild it. + // Additionally, use this in the CreateDefaultSVG function + ['flow_grid', 'flow_group_elements', 'flow_group_select', 'flow_wires', 'flow_nodes'].forEach(groupClass => { + const g = svg.querySelector(`.${groupClass}`) || createSvgElement('g', { class: g }) + while (g.firstChild) { + g.removeChild(g.firstChild) + } + }) + } + } + + // #endregion + + // #region "Node-RED src code parts" + + // SRC: https://github.com/node-red/node-red/blob/29ed5b27925e51185098a7fe3180faa4c8a734d7/packages/node_modules/%40node-red/editor-client/src/js/ui/view.js#L1057-L1179 + function generateLinkPath(origX,origY, destX, destY, sc, hasStatus = false) { + // TODO: global? + const lineCurveScale = 0.75 + const node_height = 30 + const node_width = 100 + + var dy = destY-origY; + var dx = destX-origX; + var delta = Math.sqrt(dy*dy+dx*dx); + var scale = lineCurveScale; + var scaleY = 0; + if (dx*sc > 0) { + if (delta < node_width) { + scale = 0.75-0.75*((node_width-delta)/node_width); + // scale += 2*(Math.min(5*node_width,Math.abs(dx))/(5*node_width)); + // if (Math.abs(dy) < 3*node_height) { + // scaleY = ((dy>0)?0.5:-0.5)*(((3*node_height)-Math.abs(dy))/(3*node_height))*(Math.min(node_width,Math.abs(dx))/(node_width)) ; + // } + } + } else { + scale = 0.4-0.2*(Math.max(0,(node_width-Math.min(Math.abs(dx),Math.abs(dy)))/node_width)); + } + function genCP(cp) { + return ` M ${cp[0]-5} ${cp[1]} h 10 M ${cp[0]} ${cp[1]-5} v 10 ` + } + if (dx*sc > 0) { + let cp = [ + [(origX+sc*(node_width*scale)), (origY+scaleY*node_height)], + [(destX-sc*(scale)*node_width), (destY-scaleY*node_height)] + ] + return `M ${origX} ${origY} C ${cp[0][0]} ${cp[0][1]} ${cp[1][0]} ${cp[1][1]} ${destX} ${destY}` + // + ` ${genCP(cp[0])} ${genCP(cp[1])}` + } else { + let topX, topY, bottomX, bottomY + let cp + let midX = Math.floor(destX-dx/2); + let midY = Math.floor(destY-dy/2); + if (Math.abs(dy) < 10) { + bottomY = Math.max(origY, destY) + (hasStatus?35:25) + let startCurveHeight = bottomY - origY + let endCurveHeight = bottomY - destY + cp = [ + [ origX + sc*15 , origY ], + [ origX + sc*25 , origY + 5 ], + [ origX + sc*25 , origY + startCurveHeight/2 ], + + [ origX + sc*25 , origY + startCurveHeight - 5 ], + [ origX + sc*15 , origY + startCurveHeight ], + [ origX , origY + startCurveHeight ], + + [ destX - sc*15, origY + startCurveHeight ], + [ destX - sc*25, origY + startCurveHeight - 5 ], + [ destX - sc*25, destY + endCurveHeight/2 ], + + [ destX - sc*25, destY + 5 ], + [ destX - sc*15, destY ], + [ destX, destY ], + ] + + return "M "+origX+" "+origY+ + " C "+ + cp[0][0]+" "+cp[0][1]+" "+ + cp[1][0]+" "+cp[1][1]+" "+ + cp[2][0]+" "+cp[2][1]+" "+ + " C " + + cp[3][0]+" "+cp[3][1]+" "+ + cp[4][0]+" "+cp[4][1]+" "+ + cp[5][0]+" "+cp[5][1]+" "+ + " h "+dx+ + " C "+ + cp[6][0]+" "+cp[6][1]+" "+ + cp[7][0]+" "+cp[7][1]+" "+ + cp[8][0]+" "+cp[8][1]+" "+ + " C " + + cp[9][0]+" "+cp[9][1]+" "+ + cp[10][0]+" "+cp[10][1]+" "+ + cp[11][0]+" "+cp[11][1]+" " + // +genCP(cp[0])+genCP(cp[1])+genCP(cp[2])+genCP(cp[3])+genCP(cp[4]) + // +genCP(cp[5])+genCP(cp[6])+genCP(cp[7])+genCP(cp[8])+genCP(cp[9])+genCP(cp[10]) + } else { + var cp_height = node_height/2; + var y1 = (destY + midY)/2 + topX = origX + sc*node_width*scale; + topY = dy>0?Math.min(y1 - dy/2 , origY+cp_height):Math.max(y1 - dy/2 , origY-cp_height); + bottomX = destX - sc*node_width*scale; + bottomY = dy>0?Math.max(y1, destY-cp_height):Math.min(y1, destY+cp_height); + var x1 = (origX+topX)/2; + var scy = dy>0?1:-1; + cp = [ + // Orig -> Top + [x1,origY], + [topX,dy>0?Math.max(origY, topY-cp_height):Math.min(origY, topY+cp_height)], + // Top -> Mid + // [Mirror previous cp] + [x1,dy>0?Math.min(midY, topY+cp_height):Math.max(midY, topY-cp_height)], + // Mid -> Bottom + // [Mirror previous cp] + [bottomX,dy>0?Math.max(midY, bottomY-cp_height):Math.min(midY, bottomY+cp_height)], + // Bottom -> Dest + // [Mirror previous cp] + [(destX+bottomX)/2,destY] + ]; + if (cp[2][1] === topY+scy*cp_height) { + if (Math.abs(dy) < cp_height*10) { + cp[1][1] = topY-scy*cp_height/2; + cp[3][1] = bottomY-scy*cp_height/2; + } + cp[2][0] = topX; + } + return "M "+origX+" "+origY+ + " C "+ + cp[0][0]+" "+cp[0][1]+" "+ + cp[1][0]+" "+cp[1][1]+" "+ + topX+" "+topY+ + " S "+ + cp[2][0]+" "+cp[2][1]+" "+ + midX+" "+midY+ + " S "+ + cp[3][0]+" "+cp[3][1]+" "+ + bottomX+" "+bottomY+ + " S "+ + cp[4][0]+" "+cp[4][1]+" "+ + destX+" "+destY + + // +genCP(cp[0])+genCP(cp[1])+genCP(cp[2])+genCP(cp[3])+genCP(cp[4]) + } + } + } + + // SRC: https://github.com/node-red/node-red/blob/29ed5b27925e51185098a7fe3180faa4c8a734d7/packages/node_modules/%40node-red/editor-client/src/js/ui/view.js#L3009C1-L3030C6 + function convertLineBreakCharacter(str) { + var result = []; + var lines = str.split(/\\n /); + if (lines.length > 1) { + var i=0; + for (i=0;i { + return (obj.name || obj.label || obj.info || obj.text || subflowObj.name || obj.type) + }; + + var emptyLabelFunct = (obj, subflowObj, flowdata) => { + return "" + }; + + var linkCallLabelFunct = (obj, subflowObj, flowdata) => { + if (!obj.links || obj.links.length == 0) { return obj.name || obj.type } + + var lbl = undefined; + + flowdata.forEach(function (nde) { + if (lbl) return; + + if (nde.id == obj.links[0]) { + lbl = (labelByFunct[nde.type] || defaultLabelFunct)(nde, subflowObj, flowdata); + } + }); + + /* remember: link calls can reference nodes outside of the current flow, to solve that: given them names */ + return obj.name || lbl || obj.type; + }; + + var catchLabelFunct = (obj, subflowObj, flowdata) => { + var sublabel = ""; + + if (obj.uncaught) { sublabel = ": uncaught" } + if (obj.scope) { sublabel = ": " + obj.scope.length } + if (!obj.scope && !obj.uncaught) { sublabel = ": all" } + + return obj.name || (obj.type + sublabel) + }; + + var mindMapNodeLabelFunct = (obj, subflowObj, flowdata) => { + return (obj.name || obj.label || obj.info || obj.text || "").replace(/(.{40,60})([ \n\t])/g, "$1\\n$2") + + (obj.sumPass ? " ⭄" : "") + + (obj.sumPassPrio && parseInt(obj.sumPassPrio) != 0 ? " (" + obj.sumPassPrio + ")" : ""); + }; + + var blogPageInfoLabel = (obj, subflowObj, flowdata) => { + if (obj.name) { return obj.name } + + var lbl = undefined; + flowdata.forEach(function (nde) { + if (lbl) return; + + if (nde.type == "link in" && nde.name.startsWith("[blog] ") && (nde.wires[0] || []).indexOf(obj.id) > -1) { + lbl = nde.name.substring(7) + } + }); + + return lbl || obj.type; + }; + + var labelByFunct = { + "base64": undefined, + "batch": undefined, + "catch": catchLabelFunct, + "change": undefined, + "comment": undefined, + "csv": undefined, + "debug": undefined, + "exec": undefined, + "file": undefined, + "file in": undefined, + "function": undefined, + "html": undefined, + "http response": (obj, subflowObj, flowdata) => { return (obj.name || ("http" + (obj.statusCode ? " (" + obj.statusCode + ")" : ""))) }, + "http in": (obj, subflowObj, flowdata) => { return (obj.name || ("[" + obj.method + "] " + obj.url)) }, + "http request": undefined, + "inject": undefined, + "join": undefined, + "json": undefined, + "junction": undefined, + "link in": undefined, + "link out": undefined, + "link call": linkCallLabelFunct, + "markdown": undefined, + "postgresql": undefined, + "range": undefined, + "sort": undefined, + "split": undefined, + "switch": undefined, + ui_button: undefined, + ui_list: undefined, + ui_svg_graphics: undefined, + ui_template: undefined, + ui_toast: undefined, + ui_upload: undefined, + "yaml": undefined, + "xml": undefined, + + /* private nodes for this instane */ + 'BlogPages': undefined, + 'BlogDetails': undefined, + 'BlogPageInfo': blogPageInfoLabel, + 'PubMedium': undefined, + + "Topic": mindMapNodeLabelFunct, + "Observation": mindMapNodeLabelFunct, + "Question": mindMapNodeLabelFunct, + "Thought": mindMapNodeLabelFunct, + "Idea": mindMapNodeLabelFunct, + "Analogy": mindMapNodeLabelFunct, + "Aphorism": mindMapNodeLabelFunct, + "Poesie": mindMapNodeLabelFunct, + "Humour": mindMapNodeLabelFunct, + "Treasure": mindMapNodeLabelFunct, + "Consequence": mindMapNodeLabelFunct, + "Advantage": mindMapNodeLabelFunct, + "Disadvantage": mindMapNodeLabelFunct, + "Text": mindMapNodeLabelFunct, + "Blog-Post": mindMapNodeLabelFunct, + "Comment": mindMapNodeLabelFunct, + "Codebase": mindMapNodeLabelFunct, + "Sketch": mindMapNodeLabelFunct, + "Inspiration": mindMapNodeLabelFunct, + "Quote": mindMapNodeLabelFunct, + "Definition": mindMapNodeLabelFunct, + "Book": mindMapNodeLabelFunct, + "Author": mindMapNodeLabelFunct, + + 'nnb-input-node': undefined, + 'nnb-layer-node': (obj, _sub, _flow) => { return (obj.name || (obj.actfunct + ": " + obj.bias + ", " + obj.threshold)) }, + 'nnb-output-node': undefined, + 'nnb-backprop': undefined, + 'nnb-trainer': undefined, + + 'Seeker': undefined, + 'Sink': undefined, + 'Screenshot': undefined, + 'Orphans': undefined, + 'IsMobile': undefined, + 'Navigator': undefined, + 'DrawSVG': undefined, + 'GetFlows': undefined, + + "_default": defaultLabelFunct, + }; + + // #endregion + + // #region "Flow Rendering functions" + + /** + * Get an array of tabs from the flow array. + * @param {Array} flow - Node-RED flow array + * @returns {Array<{id: string, label: string, type: string}>} - Array of tabs + */ + function getFlowTabs(flow) { + const tabs = {} + let sfIndex = 10000 + let fIndex = 0 + for (node of flow) { + if (node.type === 'subflow') { + tabs[node.id] = { + id: node.id, + label: node.name, + type: 'subflow', + order: sfIndex++, + } + } else if (node.type !== "tab" && node.z) { + if (!tabs[node.z]) { + tabs[node.z] = { + id: node.z, + label: 'Flow ' + (fIndex + 1), + type: 'tab', + order: fIndex++, + } + } + } + } + + // if the flow contains real tabs, then update the tabs objects + const flowTabs = flow.filter((d) => d.type === 'tab') + let index = 0 + for (node of flowTabs) { + // Only update if tabs[d.id] exists (i.e. there are nodes to show) + if (tabs[node.id]) { + tabs[node.id].label = node.label || tabs[node.id].label + tabs[node.id].order = index++ + } + } + + // convert tabs to an array and sort by order ascending + const tabsArray = Object.keys(tabs).map((key) => tabs[key]) + tabsArray.sort((a, b) => a.order - b.order) + + return tabsArray + } + + function normaliseFlow(flow) { + if (!Array.isArray(flow)) { + flow = [] + } + for (let i = 0; i < flow.length; i++) { + if (!flow[i].id || typeof flow[i].id !== 'string') { + flow[i].id = Math.random().toString(36).substring(2, 14) + } + } + return flow + } + + function renderFlows(flows, renderOpts) { + renderOpts = renderOpts || {} + flows = normaliseFlow(flows) + + // SETUP + /** @type {Document} */ + const doc = renderOpts.document || this.document + /** @type {HTMLElement} */ + const container = renderOpts.container + + // remove all elements from the container + while (container.firstChild) { + container.removeChild(container.firstChild) + } + + // setup the tabs container + const tabContainer = doc.createElement('div') + tabContainer.classList.add('red-ui-tabs') + container.appendChild(tabContainer) + + // setup the scrollable div container for the svg element + /** @type {HTMLDivElement} */ + const div = createDefaultDivContainer(container) + + // generate the SVG element + /** @type {SVGSVGElement} */ + const svg = createDefaultSVG(div) + resetSVG(svg) + + if (!renderOpts.scope) { + const containerClasses = container.classList + if (containerClasses && containerClasses.length) { + renderOpts.scope = containerClasses + } else { + renderOpts.scope = 'flow-renderer' + container.classList.add(renderOpts.scope) + } + } else { + container.classList.add(renderOpts.scope) + } + + // tab helper functions + const openTab = function (id) { + const tabs = tabContainer.querySelectorAll('.red-ui-tab') + tabs.forEach(tab => tab.classList.remove('active')) + // get the tab element with the data-flow-id attribute equal to the id + const selectedTab = tabContainer.querySelector(`[data-flow-id="${id}"]`) + selectedTab.classList.add('active') + renderOpts.flowId = id + renderFlow(flows, renderOpts) + } + const addTab = function (tab, index) { + const thisId = `red-ui-tab-${index}` + // const name = tab.type === 'tab' ? 'Flow ' + (index + 1) : tab.label + const name = tab.label + let tabEl = container.querySelector(`.${thisId}`) + if (!tabEl) { + tabEl = doc.createElement('div') + tabContainer.appendChild(tabEl) + tabEl.classList.add('red-ui-tab') + tabEl.classList.add('red-ui-tab-' + tab.type) + // add the as a data attribute + tabEl.setAttribute('data-flow-id', tab.id) + } + if (tab.type === 'subflow') { + // + const img = doc.createElement('i') + img.classList.add('red-ui-tab-subflow-icon') + tabEl.appendChild(img) + } + tabEl.title = name + const nameSpan = doc.createElement('span') + nameSpan.textContent = name + tabEl.appendChild(nameSpan) + tabEl.onclick = function (event) { + openTab(tab.id) + } + } + + // add styles + addStyle(doc, renderOpts.scope) + + // delete any existing tabs + let tabEls = container.querySelectorAll('.red-ui-tab') + if (tabEls) { + tabEls.forEach(tab => tab.remove()) + } + + // create tabs for each flow + const tabs = getFlowTabs(flows) + tabs.forEach((tab, index) => { + addTab(tab, index, renderOpts) + }) + + // use renderOpts.flowId (or find a suitable first tab) + if (!renderOpts.flowId) { + // find first tab with .type === 'tab' + const firstTab = tabs.find(tab => tab.type === 'tab') + const firstSubflow = tabs.find(tab => tab.type === 'subflow') + renderOpts.flowId = firstTab ? firstTab.id : firstSubflow ? firstSubflow.id : null + } + if (tabs.length) { + // select the tab (causes the flow to be rendered) + openTab(renderOpts.flowId, renderOpts) + } else { + // no tabs to render + renderFlow(flows, renderOpts) + } + + if (renderOpts.zoom) { + setupZoom(svg, renderOpts.container) + } + + const result = { + svg: svg.innerHTML, + tabs: tabs, + flowId: renderOpts.flowId, + css: getCSS(renderOpts.scope) + } + + return result + } + + function renderFlow(flow, renderOpts) { + flow = normaliseFlow(flow) + /** @type {Document} */ + const doc = renderOpts.document || this.document + const container = renderOpts.container + renderOpts = renderOpts || {} + const nodes = {} + const subflows = {} + const flow_group_select = {} + const subFlowInsOutsStatusNodes = {} + const nodeIdsThatReceiveInput = {} + const flowId = renderOpts.flowId + + /** @type {SVGSVGElement} */ + let svg = container && container.querySelector('svg') + svg = svg || createDefaultSVG(container) + resetSVG(svg) + + /** @type {SVGGElement} */ + const flow_gridEl = svg.querySelector('.flow_grid'); + if (renderOpts.gridlines && flow_gridEl) { + for (var idx = 0; idx < 250; idx++) { + flow_gridEl.append(createSvgElement('line', { + x1: 0, + x2: 8000, + y1: 20 * idx, + y2: 20 * idx, + class: 'grid-line' + })); + } + + for (var idx = 0; idx < 250; idx++) { + flow_gridEl.append(createSvgElement('line', { + x1: 20 * idx, + x2: 20 * idx, + y1: 0, + y2: 8000, + class: 'grid-line' + })); + } + } + + /* this is used to define which nodes get input decoration, this is not clear from the json data so + * we make a guessimate which nodes have inputs by the wiring within the flow + */ + flow.forEach(function (obj) { + if (obj.type == "subflow") { + /* prefix subflow since this is the type of the node that uses this subflow - makes lookup simpler */ + subflows["subflow:" + obj.id] = obj; + for (var idx = 0; idx < obj.in.length; idx++) { + for (var wdx = 0; wdx < obj.in[idx].wires.length; wdx++) { + nodeIdsThatReceiveInput[obj.in[idx].wires[wdx].id] = true + } + } + } + + if (obj.wires && obj.wires.length > 0) { + obj.wires.forEach(function (aryWires) { + aryWires.forEach(function (ndeId) { nodeIdsThatReceiveInput[ndeId] = true }) + }) + } + }) + + /* + * Rendering nodes. + */ + + /** @type {SVGGElement} */ + const flow_nodesEl = svg.querySelector('.flow_nodes') + + /* + * Important lesson: never assume that (x,y) means top-left corner ... in the case of the flows.json, (x,y) is the midpoint of the node. + * For the sack of sanity, compute the top-left corner (x,y) and add it to the node. We also add the bounding-box so we have the width. + */ + flow.forEach(function (obj) { + if (obj.z == flowId || obj.id == flowId /* this is a subflow or tab */) { + + // get node dimensions + var dimensions = widthHeightByType[obj.type] || widthHeightByType["_default"]; + // get node color + var clr = getNodeColor(obj.type); + + switch (obj.type) { + + case "tab": + case "ui_spacer": + case "ui-spacer": + // ignore tabs and spacers + break; + case "group": + /* groups are handled later since we need all bounding boxes for all nodes contained in a group */ + flow_group_select[obj.id] = obj; + break; + + case "subflow": { + /* the type of the node that represents a subflow is subflow:XXXX while the subflow has type 'subflow'. So this occurs if + the flowId is that of a subflow. The obj.id == flowId and its type is 'subflow' and here we are. + */ + subFlowInsOutsStatusNodes[obj.id] = { ...obj }; + + /* input connectors */ + for (var idx = 0; idx < subFlowInsOutsStatusNodes[obj.id].in.length; idx++) { + var inObj = subFlowInsOutsStatusNodes[obj.id].in[idx]; + var grpId = "grp" + Math.random().toString().substring(2); + + const grpObj = createSvgElement('g', { id: grpId }); + flow_nodesEl.appendChild(grpObj); + + grpObj.appendChild(createSvgElement('rect', { + ...clr, + ...dimensions, + rx: 8, + ry: 8, + x: -dimensions.width / 2, + y: -dimensions.height / 2, + "stroke-width": 1, + })) + + grpObj.setAttribute("transform", `translate(${inObj.x}, ${inObj.y})`) + inObj.bbox = grpObj.getBBox(); + inObj.bbox.x = inObj.x - dimensions.width / 2 + inObj.bbox.y = inObj.y - dimensions.height / 2 + + /* add output decoration after computing the bounding box - the decoration extends the bounding box */ + grpObj.appendChild(createSvgElement('rect', { + ...clr, + ...portDimensions, + ...portRadius, + class: "output-deco", + transform: "translate(15, -5)" + })) + + const textElem = createSvgElement('text', { + y: 0, + x: -2, + class: 'subflow-node-text-label' + }); + textElem.textContent = "input"; + grpObj.appendChild(textElem); + } + + /* output connectors */ + for (var idx = 0; idx < subFlowInsOutsStatusNodes[obj.id].out.length; idx++) { + const outObj = subFlowInsOutsStatusNodes[obj.id].out[idx]; + const grpId = "grp" + Math.random().toString().substring(2); + const grpObj = createSvgElement('g', { id: grpId, }) + flow_nodesEl.appendChild(grpObj); + + grpObj.appendChild(createSvgElement('rect', { + ...clr, + ...dimensions, + rx: 8, + ry: 8, + x: -dimensions.width / 2, + y: -dimensions.height / 2, + "stroke-width": 1, + })) + + grpObj.setAttribute("transform", `translate(${outObj.x}, ${outObj.y})`) + outObj.bbox = grpObj.getBBox() + outObj.bbox.x = outObj.x - dimensions.width / 2 + outObj.bbox.y = outObj.y - dimensions.height / 2 + + /* add output decoration after computing the bounding box - the decoration extends the bounding box */ + grpObj.appendChild(createSvgElement('rect', { + ...clr, + ...portDimensions, + ...portRadius, + class: "input-deco", + transform: "translate(-25, -5)", + })) + + /* text that goes "output\n(idx+1)\n" i.e. two lines */ + const textElem1 = createSvgElement('text', { + y: -10, + x: 0, + class: 'subflow-node-text-label' + }) + textElem1.textContent = "output" + grpObj.appendChild(textElem1) + + const textElem2 = createSvgElement('text', { + y: 8, + x: 0, + class: 'subflow-node-text-label-number' + }) + textElem2.textContent = "" + (idx + 1) + grpObj.appendChild(textElem2) + } + + /* status connectors */ + if (subFlowInsOutsStatusNodes[obj.id].status) { + const outObj = subFlowInsOutsStatusNodes[obj.id].status; + const grpId = "grp" + Math.random().toString().substring(2); + const grpObj = createSvgElement('g', { id: grpId, }) + flow_nodesEl.appendChild(grpObj); + + grpObj.appendChild(createSvgElement('rect', { + ...clr, + ...dimensions, + rx: 8, + ry: 8, + x: -dimensions.width / 2, + y: -dimensions.height / 2, + "stroke-width": 1, + })) + + grpObj.setAttribute("transform", `translate(${outObj.x}, ${outObj.y})`) + outObj.bbox = grpObj.getBBox() + outObj.bbox.x = outObj.x - dimensions.width / 2 + outObj.bbox.y = outObj.y - dimensions.height / 2 + + /* add output decoration after computing the bounding box - the decoration extends the bounding box */ + grpObj.appendChild(createSvgElement('path', { + ...clr, + ...portDimensions, + ...portRadius, + class: "input-deco", + transform: "translate(-25, -5)" + })) + + const textElem = createSvgElement('text', { + y: 0, + x: 2, + class: 'subflow-node-text-label' + }); + textElem.textContent = "status"; + grpObj.appendChild(textElem) + } + + break; + } + case "junction": { + const grpId = "grp" + Math.random().toString().substring(2) + const grpObj = createSvgElement('g', { id: grpId, }) + flow_nodesEl.appendChild(grpObj) + + grpObj.appendChild(createSvgElement('rect', { + ...clr, + ...dimensions, + rx: 3, + ry: 3, + x: -5, + y: -5, + "stroke-width": 1, + })) + + grpObj.setAttribute("transform", `translate(${obj.x}, ${obj.y})`) + + obj.bbox = grpObj.getBBox() + obj.bbox.x = obj.x + obj.bbox.y = obj.y + obj.bbox.width = 0 + obj.bbox.height = 0 + + break + } + case "link in": + case "link out": { + const grpId = "grp" + Math.random().toString().substring(2) + const grpObj = createSvgElement('g', { id: grpId, }) + flow_nodesEl.appendChild(grpObj) + + grpObj.prepend(createSvgElement('rect', { + ...clr, + ...dimensions, + rx: 5, + ry: 5, + fill: obj.color || clr.fill, + class: "node " + (" node-" + obj.id) + })) + + grpObj.setAttribute("transform", `translate(${obj.x - dimensions.width / 2}, ${obj.y - dimensions.height / 2})`) + if (obj.d) { + grpObj.setAttribute("class", "node-disabled") + } + obj.bbox = grpObj.getBBox() + obj.bbox.x = obj.x - dimensions.width / 2 + obj.bbox.y = obj.y - dimensions.height / 2 + + /* add image to node */ + if (renderOpts.images) { + grpObj.appendChild(createSvgElement('image', { + "href": getNodeImage(obj.type + (obj.mode || "")), + x: (obj.type == "link in" ? 0 : (obj.mode == "return" ? 1 : 2)), + y: 0, + width: 30, + height: 30 + })) + } + + const transAndPath = { + transform: (obj.type == "link in" ? "translate(25,10)" : "translate(-4,10)"), + d: "M 0.5,9.5 9.5,9.5 9.5,0.5 0.5,0.5 Z", + } + + if (renderOpts.arrows && (obj.type == "link out")) { + transAndPath = { + transform: (obj.type == "link in" ? "translate(27,10)" : "translate(-3,10)"), + d: "M 0,10 9,5 0,0 Z", + } + } + + /* add output decoration after computing the bounding box - the decoration extends the bounding box */ + grpObj.appendChild(createSvgElement('rect', { + ...clr, + ...portDimensions, + ...portRadius, + class: (obj.type == "link in" ? "output-deco" : ("input-deco" + (renderOpts.arrows ? " input-arrows" : ""))), + "stroke-linecap": "round", + "stroke-linejoin": "round", + ...transAndPath + })) + + break + } + default: { + /* the type of the node that represents a subflow is subflow:XXXX while the subflow has type 'subflow' */ + const grpTextId = "grpTxt" + Math.random().toString().substring(2) + const lblFunct = renderOpts.labels ? (labelByFunct[obj.type] || labelByFunct["_default"]) : emptyLabelFunct + const subflowObj = subflows[obj.type] || {} + const textLabels = getLabelParts(lblFunct(obj, subflowObj, flow), "node-text-label") + + const grpText = createSvgElement('g', { id: grpTextId, "transform": "translate(38," + (textLabels.lines.length > 1 ? 18 : 16) + ")" }) + + var ypos = 0 + textLabels.lines.forEach(function (lne) { + var textElem = createSvgElement('text', { + y: ypos, + class: 'node-text-label' + }) + textElem.textContent = lne + grpText.appendChild(textElem) + ypos += 20 + }) + + const grpId = "grp" + Math.random().toString().substring(2) + const grpObj = createSvgElement('g', { id: grpId, }) + flow_nodesEl.appendChild(grpObj) + + grpObj.appendChild(grpText) + + const txtBBox = grpText.getBBox() + const txtWidth = txtBBox.width + 60 + const txtHeight = txtBBox.height + 13.5 + const rectWidth = (dimensions.width > txtWidth ? dimensions.width : txtWidth) + let rectHeight = (dimensions.height > txtHeight ? dimensions.height : txtHeight) + + if ((obj.wires || []).length > 2) { + /* if more than 2 outputs, the node "grows" but the node might already be bigger enough. The base + height of 30 supports two outputs, everything else requires growth in height. */ + rectHeight = Math.max(rectHeight, 15 * obj.wires.length) + + /* move the text block into the middle */ + if (rectHeight > txtHeight) { + const offsetHeight = (rectHeight - txtHeight) / 2 + grpText.setAttributeNS(null, "transform", "translate(38," + ((textLabels.lines.length > 1 ? 16 : 14) + offsetHeight) + ")") + } + } + + grpObj.prepend(createSvgElement('rect', { + ...clr, + rx: 5, + ry: 5, + fill: subflowObj.color || clr.fill, + width: rectWidth, + height: rectHeight, + class: "node " + (" node-" + obj.id) + })) + + grpObj.appendChild(createSvgElement('path', { + d: "M5 0 h25 v" + rectHeight + " h-25 a 5 5 0 0 1 -5 -5 v-" + (rectHeight - 10) + " a 5 5 0 0 1 5 -5", + fill: "rgb(0,0,0)", + "fill-opacity": 0.1, + "stroke": "none" + })) + + grpObj.appendChild(createSvgElement('path', { + d: "M 29.5 0.5 l 0 " + (rectHeight - 1), + fill: "none", + stroke: "rgb(0,0,0)", + "stroke-opacity": 0.1, + "stroke-width": "1px" + })) + + grpObj.setAttribute("transform", `translate(${obj.x - rectWidth / 2}, ${obj.y - rectHeight / 2})`) + obj.bbox = grpObj.getBBox() + obj.bbox.x = obj.x - rectWidth / 2 + obj.bbox.y = obj.y - rectHeight / 2 + + /* Add image - if requested - by type - some types have no image */ + if (renderOpts.images) { + const imgBaseOpts = { + x: 1, + y: Math.max(rectHeight / 2 - 15, 1), + width: 30, + height: 30 + } + const img = getNodeImage(obj.type) + if (img) { + grpObj.appendChild(createSvgElement('image', { + "href": img, + ...imgBaseOpts + })) + } else { + if (obj.type.startsWith("subflow:")) { + const hrefContent = (subflowObj.icon && + imageNameToContent[subflowObj.icon]) || imageNameToContent["subflow.svg"] + + grpObj.appendChild(createSvgElement('image', { "href": hrefContent, ...imgBaseOpts })) + } + } + } + + /* add output decoration after computing the bounding box - the decoration extends the bounding box otherwise */ + if ((subflowObj.in && subflowObj.in.length > 0) || nodeIdsThatReceiveInput[obj.id]) { + if (renderOpts.arrows) { + grpObj.appendChild(createSvgElement('path', { + ...clrByType["junction"], + transform: "translate(-3," + ((obj.bbox.height / 2) - 5) + ")", + d: (renderOpts.arrows ? "M 0,10 9,5 0,0 Z" : "M -1,9.5 8,9.5 8,0.5 -1,0.5 Z"), + class: "input-deco input-arrows", + "stroke-linecap": "round", + "stroke-linejoin": "round", + })) + } else { + grpObj.appendChild(createSvgElement('rect', { + ...clrByType["junction"], + transform: "translate(-5," + ((obj.bbox.height / 2) - 5) + ")", + ...portDimensions, + ...portRadius, + class: "input-deco" + })) + } + } + + const outDecoBaseAttrs = { + ...clrByType["junction"], + ...portDimensions, + ...portRadius, + class: "output-deco" + } + if (obj.wires && Array.isArray(obj.wires)) { + const initFactor = (obj.wires.length == 1 ? ((obj.bbox.height / 2) - 5) : ((obj.wires.length % 2 == 0) ? 5 : 8)) + for (let idx = 0; idx < obj.wires.length; idx++) { + grpObj.appendChild(createSvgElement('rect', { + transform: "translate(" + (obj.bbox.width - 4) + "," + (initFactor + (13 * idx)) + ")", + ...outDecoBaseAttrs + })) + } + } + if (obj.d) { + grpObj.setAttribute('class', 'node-disabled') + } + + break + } + }; + /* since the obj is altered, from here on end, we will be using the altered version of the node */ + nodes[obj.id] = obj + } + }); + + /* + * Rendering groups. + * + * since groups can contain other groups, we have to loop through this until all groups have + * been placed + */ + + const flow_group_selectEl = svg.querySelector('.flow_group_select') + + var doneGroups = [] + var todoGroups = [] + var biscuitBreaker = 50 /* absolute maximum number of enclosure levels */ + + for (var grpId in flow_group_select) { + todoGroups.push(grpId) + } + + // There is also another way of doing this and that is to check for the 'g' attribute on a group + // object. The 'g' attribute is set if the group is contained within another group and the 'g' is the + // id of the other group - but moving up this chain, one gets to the very top group and knows which + // groups need to be drawn to have the size of the top level group. But that is the same as this + // except all groups are drawn, i.e., if there are two groups within one group, going from the first + // group up the tree won't give us the second group contained within the enclosing group. + while (doneGroups.length != todoGroups.length && biscuitBreaker > 0) { + biscuitBreaker -= 1 + + for (var grpId in flow_group_select) { + if (doneGroups.indexOf(grpId) > -1) { continue } + + var grpObj = flow_group_select[grpId] + + /* create a very back-of-a-paper-napkin estimate of the height and width (laziness) + * Could use something like d3.js for doing this automagically, but limit the number + * of dependencies for this code - ideally no jQuery either. + ***/ + + var width = 0, height = 0, oneWasMissing = false + + grpObj.nodes.forEach(function (ndeId) { + var bbox = (nodes[ndeId] || {}).bbox + if (bbox) { + width = Math.max(width, (bbox.x - grpObj.x) + bbox.width) + height = Math.max(height, (bbox.y - grpObj.y) + bbox.height) + } else { + oneWasMissing = true + } + }) + + // Handle groups of groups of groups - i.e. if a group has been defined yet (i.e. has no + // bounding box) then continue with the next group. Each group is only created if all its + // nodes that it contains have been defined. + if (oneWasMissing) { continue } + + var grpRectId = "grpRectId" + Math.random().toString().substring(2) + var grpSvgObj = createSvgElement('g', { + id: grpRectId, + }) + + grpSvgObj.setAttribute("transform", `translate(${grpObj.x}, ${grpObj.y})`) + + grpSvgObj.appendChild(createSvgElement('rect', { + rx: 5, + ry: 5, + width: grpObj.w, + height: grpObj.h, + fill: "none", + "fill-opacity": 0, + "stroke-width": 2, + stroke: "grey", + class: "group-" + grpObj.id, + ...grpObj.style, + })) + + flow_group_selectEl.prepend(grpSvgObj) + + var obj = nodes[grpId] + obj.bbox = grpSvgObj.getBBox() + obj.bbox.x = grpObj.x + obj.bbox.y = grpObj.y + + var labelGrp = createSvgElement('g', {}) + grpSvgObj.appendChild(labelGrp) + + /* this is taken from + * https://github.com/node-red/node-red/blob/7e9042e9f713eec981adeb8ff6af226a40efb5af/packages/node_modules/%40node-red/editor-client/src/js/ui/view.js#L5555 + */ + if (grpObj.style.label && grpObj.name) { + var labelPos = grpObj.style["label-position"] || "nw" + var labels = getLabelParts(grpObj.name, "group-text-label") + + var labelX = 0 + var labelY = 0 + var labelAnchor = "start" + + if (labelPos[0] === 'n') { + labelY = 0 + 15 // Allow for font-height + } else { + labelY = obj.bbox.height - 5 - (labels.lines.length - 1) * 16 + } + if (labelPos[1] === 'w') { + labelX = 5 + labelAnchor = "start" + } else if (labelPos[1] === 'e') { + labelX = obj.bbox.width - 5 + labelAnchor = "end" + } else { + labelX = obj.bbox.width / 2 + labelAnchor = "middle" + } + + labelGrp.setAttribute("transform", `translate(${labelX}, ${labelY})`) + labelGrp.setAttribute("text-anchor", labelAnchor) + if (labels) { + let ypos = 0 + labels.lines.forEach(function (name) { + const tspan = createSvgElement("text", { + class: "group-text-label", + x: 0, + y: ypos, + fill: grpObj.style.color || 'grey', + }) + tspan.textContent = name + labelGrp.appendChild(tspan) + ypos += 16 + }) + } + } + + doneGroups.push(grpId) + } + } + + /* + * Rendering the wires between nodes + */ + const flow_wiresEl = svg.querySelector('.flow_wires') + + var linkOutNodes = [] + + /* rendering subflow? then the subflow hash will be filled */ + var subFlowIds = Object.keys(subFlowInsOutsStatusNodes); + for (var idx = 0; idx < subFlowIds.length; idx++) { + var sfObj = subFlowInsOutsStatusNodes[subFlowIds[idx]]; + + for (var jdx = 0; jdx < sfObj.in.length; jdx++) { + var inObj = sfObj.in[jdx]; + + for (var wdx = 0; wdx < inObj.wires.length; wdx++) { + var otherNode = nodes[inObj.wires[wdx].id]; + + var startX = inObj.bbox.x + inObj.bbox.width; + var startY = inObj.bbox.y + inObj.bbox.height / 2; + var endX = otherNode.bbox.x; + var endY = otherNode.bbox.y + otherNode.bbox.height / 2; + + flow_wiresEl.appendChild(createSvgElement('path', { + d: generateLinkPath(startX, startY, endX, endY, 1), + class: "link " + (otherNode.d ? "link-disabled" : "") + (" link-from-" + sfObj.id + "-to-" + otherNode.id) + })); + } + } + + if (sfObj.status) { sfObj.out.push(sfObj.status) } + for (var jdx = 0; jdx < sfObj.out.length; jdx++) { + var outObj = sfObj.out[jdx] + + for (var wdx = 0; wdx < outObj.wires.length; wdx++) { + var otherNode = nodes[outObj.wires[wdx].id]; + const initFactor = (otherNode.wires.length == 1 ? otherNode.bbox.height / 2 : ((otherNode.wires.length % 2 == 0) ? 10 : 13)); + + var startX = otherNode.bbox.x + otherNode.bbox.width; + var startY = otherNode.bbox.y + (initFactor + (13 * outObj.wires[wdx].port)); + var endX = outObj.bbox.x; + var endY = outObj.bbox.y + outObj.bbox.height / 2; + + flow_wiresEl.appendChild(createSvgElement('path', { + d: generateLinkPath(startX, startY, endX, endY, 1), + class: "link " + (otherNode.d ? "link-disabled" : "") + (" link-from-" + sfObj.id + "-to-" + otherNode.id) + })); + } + } + + } + + for (var ndeId in nodes) { + var nde = nodes[ndeId]; + if (nde.type == "link out") { linkOutNodes.push(nde) } + if ((nde.wires || []).length == 0) { continue } + + const initFactor = (nde.wires.length == 1 ? nde.bbox.height / 2 : ((nde.wires.length % 2 == 0) ? 10 : 13)); + var wireCnt = 0; + nde.wires.forEach(function (wires) { + wires.forEach(function (otherNodeId) { + var otherNode = nodes[otherNodeId]; + + if (otherNode) { + var startX = nde.bbox.x + nde.bbox.width; + var startY = nde.bbox.y + (initFactor + (13 * wireCnt)); + var endX = otherNode.bbox.x; + var endY = otherNode.bbox.y + otherNode.bbox.height / 2; + + flow_wiresEl.appendChild(createSvgElement('path', { + d: generateLinkPath(startX, startY, endX, endY, 1), + class: "link " + (otherNode.d || nde.d ? "link-disabled" : "") + (" link-from-" + nde.id + "-to-" + otherNode.id) + })); + } + }); + wireCnt++; + }); + } + + /* draw the links between link nodes, i.e. link-in and link-out nodes */ + if (renderOpts.linkLines) { + linkOutNodes.forEach(function (nde) { + nde.links.forEach(function (ndeId) { + var otherNode = nodes[ndeId]; + + if (otherNode) { + var startX = nde.bbox.x + nde.bbox.width; + var startY = nde.bbox.y + nde.bbox.height / 2; + var endX = otherNode.bbox.x; + var endY = otherNode.bbox.y + otherNode.bbox.height / 2; + + flow_wiresEl.appendChild(createSvgElement('path', { + d: generateLinkPath(startX, startY, endX, endY, 1), + "stroke-dasharray": "25,4", + class: "link " + (otherNode.d || nde.d ? "link-disabled" : "") + (" link-from-" + nde.id + "-to-" + otherNode.id) + })); + + flow_wiresEl.appendChild(createSvgElement('circle', { + cy: startY, + cx: startX + 6.5, + r: 5, + stroke: 'rgb(170, 170, 170)', + "stroke-width": 1, + stroke: 'rgb(170, 170, 170)', + "fill": 'rgb(238, 238, 238)', + class: (otherNode.d || nde.d ? "link-disabled" : "") + (" link-from-" + nde.id + "-to-" + otherNode.id) + })); + + flow_wiresEl.appendChild(createSvgElement('circle', { + cy: endY, + cx: endX - 6.5, + r: 5, + stroke: 'rgb(170, 170, 170)', + "stroke-width": 1, + stroke: 'rgb(170, 170, 170)', + "fill": 'rgb(238, 238, 238)', + class: (otherNode.d || nde.d ? "link-disabled" : "") + (" link-from-" + nde.id + "-to-" + otherNode.id) + })); + } + }); + }); + } + + /* finally remove our changes to the objects in the flowData array */ + flow.forEach(function (obj) { delete obj.bbox; }); + + const result = { + svg: svg.innerHTML, + flowId: renderOpts.flowId, + css: getCSS(renderOpts.scope) + } + + return result + } + + // #endregion + + /** + * @param {Array} flows - The flows to render. + * @param {FlowRendererOptions} options - The options for rendering the flows. + */ + function flowRenderer(flows, options) { + try { + options = Object.assign({}, flowRenderer.defaults, options) + // see if the container has and data-options and merge those + // data-xxx will enable the option + // data-xxx="true" will enable the option + // data-xxx="false" will disable the option + if (options.container) { + const containerOptions = {} + const dataOptions = ['scope', 'grid-lines', 'arrows', 'zoom', 'images', 'link-lines', 'labels'] + dataOptions.forEach(function (opt) { + if (options.container.hasAttribute("data-" + opt)) { + const optionValue = options.container.getAttribute("data-" + opt) || "true" + containerOptions[opt.replace(/-/g, '')] = optionValue === "true" + } + }) + options = Object.assign({}, options, containerOptions) + + if (typeof window === 'undefined') { + // if we are in a node.js environment, set global.document to the container ownerDocument + options.document = options.document || options.container.ownerDocument + global.__document = options.document + // polyfill getBBox for SVGElement in node env + if (options.document && options.document.createElementNS) { + // get the prototype of the SVGElement without accessing it directly + const SVGElement = options.document.createElementNS('http://www.w3.org/2000/svg', 'rect').constructor + if (!SVGElement.prototype.getBBox) { + SVGElement.prototype.getBBox = function () { + return getBBoxPolyfill(options.document, this) + } + } + } + + } + } + const renderer = renderFlows.bind(this) + return renderer(flows, options) + } catch (e) { + e.message += '\nPlease report this to @flowfuse/flow-renderer.'; + if ((options || flowRenderer.defaults).silent) { + return '

An error occured:

'
+                    + escape(e.message + '', true)
+                    + '
'; + } + throw e; + } + } + + /** + * @typedef {Object} FlowRendererOptions + * @property {boolean} [gridlines=true] - Whether to display grid lines on the flow diagram. Defaults to true. + * @property {boolean} [zoom=true] - Whether to enable zooming on the flow diagram. Defaults to true. + * @property {boolean} [images=true] - Whether to display images on the flow diagram. Defaults to true. + * @property {boolean} [linkLines=false] - Whether to display link lines on the flow diagram. Defaults to false. + * @property {boolean} [labels=true] - Whether to display labels on the flow diagram. Defaults to true. + * @property {Array} flowId - The specific flow(s) to render. If not provided, all flows will be rendered. + * @property {HTMLElement} container - The container div where the SVG diagram and controls will be rendered. + * @property {Document} document - The document object to use for creating SVG elements. Defaults to the global document object. + */ + + /** @type {FlowRendererOptions} */ + flowRenderer.defaults = { + arrows: false, + gridlines: true, + zoom: true, + images: true, + linkLines: false, + labels: true, + flowId: undefined, + container: undefined, + document: this.document + } + + flowRenderer.getStyles = getCSS; + flowRenderer.renderFlows = renderFlows; + flowRenderer.renderFlow = renderFlow; + +}).call(function () { + return this || (typeof window !== 'undefined' ? window : global); +}()); diff --git a/package.json b/package.json index dd6582a..c8895d4 100644 --- a/package.json +++ b/package.json @@ -4,7 +4,7 @@ "description": "Node-RED flow renderer", "main": "index.js", "scripts": { - "demo": "npx http-server", + "demo": "npx http-server -p 8080 -c-1 -o demo.html", "test": "echo \"Error: no test specified\" && exit 1" }, "keywords": [ From f469f59c0f7d90b6edd1f954461a0098978f4b1a Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Wed, 24 Apr 2024 14:50:23 +0100 Subject: [PATCH 2/4] support ESM --- README.md | 47 +++++++++-- index.js | 221 +++++++++++++++++++++++++++------------------------ package.json | 2 +- 3 files changed, 156 insertions(+), 114 deletions(-) diff --git a/README.md b/README.md index 94790cd..e5c295c 100644 --- a/README.md +++ b/README.md @@ -20,11 +20,12 @@ Run `npm run demo` to test the flow renderer in a browser. Add the following script tag to your HTML file: ```html - + ``` _or wherever the script is located in your project_ Next, add a container element to your HTML file and call the `flowRenderer` function with the flow data and options. +NOTE: flow-renderer is an ES Module and requires a modern browser to run. Script tags must have the `type="module"` attribute. By default, the flow renderer will render the flow with `gridlines`, `images`, `labels` and `zoom` options enabled. @@ -32,16 +33,19 @@ To operate the zoom, use the mouse wheel + Ctrl key. To scroll the container vertically, use the mouse wheel without the Shift key. To scroll the container horizontally, use the mouse wheel + Shift key. + + ### Basic example ```html -
+
``` ```javascript +const renderer = new FlowRenderer() const container1 = document.getElementById('nr-flow-1'); const flow = [{"id": "1001", "type": "inject", "x": 100, "y": 40, "wires": [["1002"]]}, {"id": "1002", "type": "debug", "x":300, "y": 40}] -flowRenderer(flow, { container: container1 }) +renderer.renderFlows(flow, { container: container1 }) ``` ### Inline Options example @@ -49,25 +53,28 @@ flowRenderer(flow, { container: container1 }) Options can be set by data attributes `scope`, `grid-lines`, `zoom`, `images`, `link-lines`, `labels` ```html -
+
``` ```javascript +const renderer = new FlowRenderer() const container2 = document.getElementById('nr-flow-2'); const flow = [{"id": "1001", "type": "inject", "x": 100, "y": 40, "wires": [["1002"]]}, {"id": "1002", "type": "debug", "x":300, "y": 40}] -flowRenderer(flow, { container: container2 }) +renderer.renderFlows(flow, { container: container2 }) ``` + ### Full Options example ```html -
+
``` ```javascript +const renderer = new FlowRenderer() const container3 = document.getElementById('nr-flow-3'); const flow = [{"id": "1001", "type": "inject", "x": 100, "y": 40, "wires": [["1002"]]}, {"id": "1002", "type": "debug", "x":300, "y": 40}] -flowRenderer(flow, { +renderer.renderFlows(flow, { container: container3, scope: 'my-scope', // scope for CSS gridlines: true, // show gridlines @@ -79,6 +86,26 @@ flowRenderer(flow, { }) ``` +### Vue example + +```html +
+
+
+ + +``` + ## Acknowledgements This project owes a huge thanks to Gerrit Riessen for his original works on [node-red-flowviewer](https://github.com/gorenje/node-red-flowviewer-js). It was this great contribution that started the ball rolling. Gerrit kindly allowed us relicense the parts we needed to use in this project. @@ -90,6 +117,12 @@ This project owes a huge thanks to Gerrit Riessen for his original works on [nod * The flow renderer does not support the full range of contributed Node-RED nodes however they will render as a generic node type complete with the node's label. * The flow renderer does not always render the flows and nodes exactly as they appear in the Node-RED editor. This is due in part to being a client-side render with no server-side component to provide full context and partly due to the current limitations of the renderer itself. +## Versioning + +While the API is in development, the version number of this package will remain at `0.x.y`. +`x` will be incremented for breaking changes, `y` for new features and patches. +Once the API is stable, the version number will be updated to 1.0.0 and full SemVer rules will be applied. + ## License Apache-2.0 diff --git a/index.js b/index.js index 6759064..360e9c2 100644 --- a/index.js +++ b/index.js @@ -1,18 +1,45 @@ - -; (function () { - - if (typeof exports === 'object') { - module.exports = flowRenderer; - } else if (typeof define === 'function' && define.amd) { - define(function () { return flowRenderer; }); - } else { - this.flowRenderer = flowRenderer; - } - +/** + * @typedef {Object} FlowRendererOptions + * @property {boolean} [gridlines=true] - Whether to display grid lines on the flow diagram. Defaults to true. + * @property {boolean} [zoom=true] - Whether to enable zooming on the flow diagram. Defaults to true. + * @property {boolean} [images=true] - Whether to display images on the flow diagram. Defaults to true. + * @property {boolean} [linkLines=false] - Whether to display link lines on the flow diagram. Defaults to false. + * @property {boolean} [labels=true] - Whether to display labels on the flow diagram. Defaults to true. + * @property {Array} flowId - The specific flow(s) to render. If not provided, all flows will be rendered. + * @property {HTMLElement} container - The container div where the SVG diagram and controls will be rendered. + * @property {Document} document - The document object to use for creating SVG elements. Defaults to the global document object. + */ + +/** + * @typedef {Object} FlowNode - A node in a flow. + * @property {string} id - The unique identifier for the node. + * @property {string} type - The type of node. + */ + +/** + * @typedef {Object} Flows - A collection of flows. + * @property {Array} nodes - The nodes in the flow. + */ + +const FlowRenderer = (function () { + let _this = this || {}; // #region Constants const styleId = 'flow-renderer-css' const PORT_WIDTH = 10 + /** @type {FlowRendererOptions} */ + const defaults = { + arrows: false, + gridlines: true, + zoom: true, + images: true, + linkLines: false, + labels: true, + flowId: undefined, + container: undefined, + document: undefined + } + const portDimensions = { width: PORT_WIDTH, height: PORT_WIDTH @@ -196,7 +223,7 @@ "ff-logo.svg": "", } - const clrByType = { + const colorByType = { "base64": _hshClr("#DEBD5C"), "batch": _hshClr("#E2D96E"), "catch": _hshClr("#e49191"), @@ -398,17 +425,17 @@ } function getNodeColor (type) { - let colors = clrByType[type] + let colors = colorByType[type] if (!colors) { if (type.startsWith("subflow:")) { - colors = clrByType["subflow"] + colors = colorByType["subflow"] } else if (type.startsWith("ui-")) { - colors = clrByType["ui-template"] + colors = colorByType["ui-template"] } else if (type.startsWith("ui_")) { - colors = clrByType["ui_template"] + colors = colorByType["ui_template"] } } - return colors || clrByType["_default"] + return colors || colorByType["_default"] } function _hshClr (fill, stroke) { @@ -1125,7 +1152,7 @@ const tabs = {} let sfIndex = 10000 let fIndex = 0 - for (node of flow) { + for (let node of flow) { if (node.type === 'subflow') { tabs[node.id] = { id: node.id, @@ -1148,7 +1175,7 @@ // if the flow contains real tabs, then update the tabs objects const flowTabs = flow.filter((d) => d.type === 'tab') let index = 0 - for (node of flowTabs) { + for (let node of flowTabs) { // Only update if tabs[d.id] exists (i.e. there are nodes to show) if (tabs[node.id]) { tabs[node.id].label = node.label || tabs[node.id].label @@ -1163,6 +1190,11 @@ return tabsArray } + /** + * Normalise the flows array + * @param {Flows} flow - The flows to normalise + * @returns {Flows} + */ function normaliseFlow(flow) { if (!Array.isArray(flow)) { flow = [] @@ -1175,13 +1207,49 @@ return flow } + /** + * Normalise the options for rendering the flows + * NOTE: This function will also merge defaults and any data-options from the container + * NOTE: Merge occurs in this order: defaults, container data-options, provided options + * @param {FlowRendererOptions} [options] - The options for rendering the flows. + */ + function normaliseOptions(options) { + const mergeItems = [defaults] + // see if the container has and data-options and merge those + // data-xxx will enable the option + // data-xxx="true" will enable the option + // data-xxx="false" will disable the option + if (options) { + if (options.container) { + const containerOptions = {} + const dataOptions = ['scope', 'grid-lines', 'arrows', 'zoom', 'images', 'link-lines', 'labels'] + dataOptions.forEach(function (opt) { + if (options.container.hasAttribute("data-" + opt)) { + const optionValue = options.container.getAttribute("data-" + opt) || "true" + containerOptions[opt.replace(/-/g, '')] = optionValue === "true" + } + }) + mergeItems.push(containerOptions) + } + mergeItems.push(options) + } + // merge the options + const opts = Object.assign({}, ...mergeItems) + return opts + } + + /** + * Render a set of flows + * @param {Flows} flows - The flows to render + * @param {FlowRendererOptions} renderOpts - The options for rendering the flows + */ function renderFlows(flows, renderOpts) { - renderOpts = renderOpts || {} flows = normaliseFlow(flows) + renderOpts = normaliseOptions(renderOpts) // SETUP /** @type {Document} */ - const doc = renderOpts.document || this.document + const doc = getDocument(renderOpts.document, _this.document, renderOpts.container, this) /** @type {HTMLElement} */ const container = renderOpts.container @@ -1298,10 +1366,17 @@ return result } + /** + * Render a single flow tab + * @param {Flows} flow - The flow to render + * @param {FlowRendererOptions} renderOpts - The options for rendering the flows + */ function renderFlow(flow, renderOpts) { flow = normaliseFlow(flow) + renderOpts = normaliseOptions(renderOpts) + /** @type {Document} */ - const doc = renderOpts.document || this.document + const doc = getDocument(renderOpts.document, _this.document, renderOpts.container, this) const container = renderOpts.container renderOpts = renderOpts || {} const nodes = {} @@ -1310,6 +1385,7 @@ const subFlowInsOutsStatusNodes = {} const nodeIdsThatReceiveInput = {} const flowId = renderOpts.flowId + const junctionColor = getNodeColor('junction') /** @type {SVGSVGElement} */ let svg = container && container.querySelector('svg') @@ -1716,7 +1792,7 @@ if ((subflowObj.in && subflowObj.in.length > 0) || nodeIdsThatReceiveInput[obj.id]) { if (renderOpts.arrows) { grpObj.appendChild(createSvgElement('path', { - ...clrByType["junction"], + ...junctionColor, transform: "translate(-3," + ((obj.bbox.height / 2) - 5) + ")", d: (renderOpts.arrows ? "M 0,10 9,5 0,0 Z" : "M -1,9.5 8,9.5 8,0.5 -1,0.5 Z"), class: "input-deco input-arrows", @@ -1725,7 +1801,7 @@ })) } else { grpObj.appendChild(createSvgElement('rect', { - ...clrByType["junction"], + ...junctionColor, transform: "translate(-5," + ((obj.bbox.height / 2) - 5) + ")", ...portDimensions, ...portRadius, @@ -1735,7 +1811,7 @@ } const outDecoBaseAttrs = { - ...clrByType["junction"], + ...junctionColor, ...portDimensions, ...portRadius, class: "output-deco" @@ -2030,87 +2106,20 @@ // #endregion - /** - * @param {Array} flows - The flows to render. - * @param {FlowRendererOptions} options - The options for rendering the flows. - */ - function flowRenderer(flows, options) { - try { - options = Object.assign({}, flowRenderer.defaults, options) - // see if the container has and data-options and merge those - // data-xxx will enable the option - // data-xxx="true" will enable the option - // data-xxx="false" will disable the option - if (options.container) { - const containerOptions = {} - const dataOptions = ['scope', 'grid-lines', 'arrows', 'zoom', 'images', 'link-lines', 'labels'] - dataOptions.forEach(function (opt) { - if (options.container.hasAttribute("data-" + opt)) { - const optionValue = options.container.getAttribute("data-" + opt) || "true" - containerOptions[opt.replace(/-/g, '')] = optionValue === "true" - } - }) - options = Object.assign({}, options, containerOptions) - - if (typeof window === 'undefined') { - // if we are in a node.js environment, set global.document to the container ownerDocument - options.document = options.document || options.container.ownerDocument - global.__document = options.document - // polyfill getBBox for SVGElement in node env - if (options.document && options.document.createElementNS) { - // get the prototype of the SVGElement without accessing it directly - const SVGElement = options.document.createElementNS('http://www.w3.org/2000/svg', 'rect').constructor - if (!SVGElement.prototype.getBBox) { - SVGElement.prototype.getBBox = function () { - return getBBoxPolyfill(options.document, this) - } - } - } - - } - } - const renderer = renderFlows.bind(this) - return renderer(flows, options) - } catch (e) { - e.message += '\nPlease report this to @flowfuse/flow-renderer.'; - if ((options || flowRenderer.defaults).silent) { - return '

An error occured:

'
-                    + escape(e.message + '', true)
-                    + '
'; - } - throw e; - } - } - - /** - * @typedef {Object} FlowRendererOptions - * @property {boolean} [gridlines=true] - Whether to display grid lines on the flow diagram. Defaults to true. - * @property {boolean} [zoom=true] - Whether to enable zooming on the flow diagram. Defaults to true. - * @property {boolean} [images=true] - Whether to display images on the flow diagram. Defaults to true. - * @property {boolean} [linkLines=false] - Whether to display link lines on the flow diagram. Defaults to false. - * @property {boolean} [labels=true] - Whether to display labels on the flow diagram. Defaults to true. - * @property {Array} flowId - The specific flow(s) to render. If not provided, all flows will be rendered. - * @property {HTMLElement} container - The container div where the SVG diagram and controls will be rendered. - * @property {Document} document - The document object to use for creating SVG elements. Defaults to the global document object. - */ - - /** @type {FlowRendererOptions} */ - flowRenderer.defaults = { - arrows: false, - gridlines: true, - zoom: true, - images: true, - linkLines: false, - labels: true, - flowId: undefined, - container: undefined, - document: this.document + return { + renderFlows: renderFlows, + renderFlow: renderFlow, + normaliseOptions: normaliseOptions, + getStyles: getCSS, } +}) - flowRenderer.getStyles = getCSS; - flowRenderer.renderFlows = renderFlows; - flowRenderer.renderFlow = renderFlow; +if (typeof module === 'object' && module.exports) { + module.exports = FlowRenderer +} else if (typeof window === 'object') { + window.FlowRenderer = FlowRenderer +} else { + global.FlowRenderer = FlowRenderer +} -}).call(function () { - return this || (typeof window !== 'undefined' ? window : global); -}()); +export default FlowRenderer diff --git a/package.json b/package.json index c8895d4..1af0df2 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@flowfuse/flow-renderer", - "version": "1.0.0", + "version": "0.1.0", "description": "Node-RED flow renderer", "main": "index.js", "scripts": { From 872d4f422def1fce0e9f8a4c42e96c5b551eb175 Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Wed, 24 Apr 2024 14:50:53 +0100 Subject: [PATCH 3/4] more demos (esm and vue) --- demo-esm.html | 97 ++++++++++++++++++++++++++++++++++++++++ demo-vanilla.html | 95 +++++++++++++++++++++++++++++++++++++++ demo-vue.html | 111 ++++++++++++++++++++++++++++++++++++++++++++++ demo.html | 100 ++++------------------------------------- 4 files changed, 312 insertions(+), 91 deletions(-) create mode 100644 demo-esm.html create mode 100644 demo-vanilla.html create mode 100644 demo-vue.html diff --git a/demo-esm.html b/demo-esm.html new file mode 100644 index 0000000..0b70b43 --- /dev/null +++ b/demo-esm.html @@ -0,0 +1,97 @@ + + + + + + FlowFuse Flow Renderer + + + + + + + + +

FlowFuse Flow Renderer

+
Client side demo
+
+ + +

Node-RED Flow

+ +
+ + +
+ + +

Render (grid, images, labels)

+
+ +
+
+ + +

Render (no grid, no images, labels)

+
+ +
+
+ + +

Render (no grid, images, no labels) using data-options

+
+ + + + + + \ No newline at end of file diff --git a/demo-vanilla.html b/demo-vanilla.html new file mode 100644 index 0000000..75b6280 --- /dev/null +++ b/demo-vanilla.html @@ -0,0 +1,95 @@ + + + + + + FlowFuse Flow Renderer + + + + + + + + +

FlowFuse Flow Renderer

+
Client side demo
+
+ + +

Node-RED Flow

+ +
+ + +
+ + +

Render (grid, images, labels)

+
+ +
+
+ + +

Render (no grid, no images, labels)

+
+ +
+
+ + +

Render (no grid, images, no labels) using data-options

+
+ + + + + + \ No newline at end of file diff --git a/demo-vue.html b/demo-vue.html new file mode 100644 index 0000000..034c653 --- /dev/null +++ b/demo-vue.html @@ -0,0 +1,111 @@ + + + + + + FlowFuse Flow Renderer + + + + + + + +

FlowFuse Flow Renderer

+
Client side demo
+
+ +
+ + +

Node-RED Flow

+ +
+ + +
+ + +

Render (grid, images, labels)

+
+ +
+
+ + +

Render (no grid, no images, labels)

+
+ +
+
+ + +

Render (no grid, images, no labels) using data-options

+
+ +
+ + + + + + + + \ No newline at end of file diff --git a/demo.html b/demo.html index 4a1b7a7..3b33e90 100644 --- a/demo.html +++ b/demo.html @@ -1,94 +1,12 @@ - - - - FlowFuse Flow Renderer - - - - - - - - -

FlowFuse Flow Renderer

-
Client side demo
-
- - -

Node-RED Flow

- -
- - -
- - -

Render (grid, images, labels)

-
- -
-
- - -

Render (no grid, no images, labels)

-
- -
-
- - -

Render (no grid, images, no labels) using data-options

-
- - - - - + +

FlowFuse Flow Renderer

+ + + \ No newline at end of file From c2d9cc75330a3b0c07a2c30c7b8f981cd13111ba Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Wed, 24 Apr 2024 14:58:04 +0100 Subject: [PATCH 4/4] clarify which demo page --- demo-esm.html | 2 +- demo-vanilla.html | 2 +- demo-vue.html | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/demo-esm.html b/demo-esm.html index 0b70b43..7431b94 100644 --- a/demo-esm.html +++ b/demo-esm.html @@ -28,7 +28,7 @@

FlowFuse Flow Renderer

-
Client side demo
+
ESM client side demo

diff --git a/demo-vanilla.html b/demo-vanilla.html index 75b6280..b1592fa 100644 --- a/demo-vanilla.html +++ b/demo-vanilla.html @@ -28,7 +28,7 @@

FlowFuse Flow Renderer

-
Client side demo
+
Vanilla client side demo

diff --git a/demo-vue.html b/demo-vue.html index 034c653..03b7ad8 100644 --- a/demo-vue.html +++ b/demo-vue.html @@ -27,7 +27,7 @@

FlowFuse Flow Renderer

-
Client side demo
+
Vue client side demo