Compare commits
	
		
			1 Commits
		
	
	
		
			0.20.0-bet
			...
			start-proj
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 499d22daca | 
							
								
								
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1 +0,0 @@ | ||||
| /packages/node_modules/** linguist-generated=false | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -8,9 +8,7 @@ If your issue is: | ||||
|   - a feature request or suggestion for a change, | ||||
|   - or problems with 3rd party (`node-red-contrib-`) nodes | ||||
|  | ||||
| please use the [Node-RED Forum](https://discourse.nodered.org) or [slack team](https://nodered.org/slack). | ||||
|  | ||||
| You could also consider asking a question on [Stack Overflow](https://stackoverflow.com/questions/tagged/node-red) and tag it `node-red`. | ||||
| please use the [mailing list](https://groups.google.com/forum/#!forum/node-red), [slack team](https://nodered.org/slack) or ask a question on [Stack Overflow](https://stackoverflow.com/questions/tagged/node-red) and tag it `node-red`. | ||||
|  | ||||
| That way the whole Node-RED user community can help, rather than rely on the core development team. | ||||
|  | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,7 +16,7 @@ Put an `x` in the boxes that apply | ||||
| <!-- | ||||
| If you want to raise a pull-request with a new feature, or a refactoring | ||||
| of existing code, it **may well get rejected** if it hasn't been discussed on | ||||
| the [forum](https://discourse.nodered.org) or | ||||
| the [mailing list](https://groups.google.com/forum/#!forum/node-red) or | ||||
| [slack team](https://nodered.org/slack) first. | ||||
|  | ||||
| --> | ||||
|   | ||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -17,8 +17,3 @@ node_modules | ||||
| public | ||||
| locales/zz-ZZ | ||||
| nodes/core/locales/zz-ZZ | ||||
| !packages/node_modules | ||||
| packages/node_modules/@node-red/editor-client/public | ||||
| !test/**/node_modules | ||||
| docs | ||||
| !packages/node_modules/**/docs | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| /Gruntfile.js | ||||
| /.git/* | ||||
| /lib/* | ||||
| *.backup | ||||
| /public/* | ||||
|   | ||||
							
								
								
									
										7
									
								
								.npmignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| .settings | ||||
| .jshintignore | ||||
| .jshintrc | ||||
| .project | ||||
| .tern-project | ||||
| .travis.yml | ||||
| .git | ||||
							
								
								
									
										26
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						| @@ -1,10 +1,20 @@ | ||||
| sudo: false | ||||
| language: node_js | ||||
| matrix: | ||||
|   include: | ||||
|     - node_js: "10" | ||||
|       script: | ||||
|         - ./node_modules/.bin/grunt && istanbul report text && ( cat coverage/lcov.info | $(npm get prefix)/bin/coveralls || true ) && rm -rf coverage | ||||
|       before_script: | ||||
|         - npm install -g istanbul coveralls | ||||
|     - node_js: "8" | ||||
| env: | ||||
|   - CXX="g++-4.8" | ||||
| addons: | ||||
|   apt: | ||||
|     sources: | ||||
|     - ubuntu-toolchain-r-test | ||||
|     packages: | ||||
|     - g++-4.8 | ||||
|     - gcc-4.8 | ||||
| node_js: | ||||
|   - "8" | ||||
|   - "6" | ||||
|   - "4" | ||||
| script: | ||||
|   - istanbul cover ./node_modules/.bin/grunt --report lcovonly && istanbul report text && ( cat coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js || true ) && rm -rf coverage | ||||
| before_script: | ||||
|   - npm install -g istanbul | ||||
|   - npm install coveralls | ||||
|   | ||||
							
								
								
									
										313
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						| @@ -1,316 +1,3 @@ | ||||
| #### 0.20.0-beta.2: Beta Release | ||||
|  | ||||
|  - Split Node-RED internals into multiple sub-modules | ||||
|  | ||||
| Editor | ||||
|  | ||||
|  - Allow the editor to use a custom admin api url root | ||||
|  - Improve performance of Flow Diff dialog - @TothiViseo #1989 | ||||
|  - Add 'open project' option to Projects Welcome dialog | ||||
|  - Add 'type already registered' check in palette editor | ||||
|  - Handle missing tab.disabled property | ||||
|  - Handle missing wires prop and string x/y props on import | ||||
|  - Add RED.notifications.hide flag - for UI testing | ||||
|  - Improve alignment of node label edit inputs | ||||
|  - Show arrow-in node when invalid font-awesome icon name was specified for default icon | ||||
|  - Add ability to delete context values from sidebar | ||||
|  - Allow copy-to-clipboard copy whole tabs | ||||
|  - Make disabled flows more obvious in editor | ||||
|  - Allow import/export from file in editor | ||||
|  - Allow config nodes to be selected in sidebar and deleted | ||||
|  - Show port label of subflow with input port | ||||
|  - Support ctrl-click selection of flow tabs | ||||
|  - Allow left-hand node button to act as toggle | ||||
|  - Support dbl-click in tab bar to add new flow in position | ||||
|  - Fix duplicate subflow detection on import | ||||
|  - Add import notification with info on what has been imported Closes #1862 | ||||
|  - Show error details when trying to import invalid json | ||||
|  - Show default icon when non-existent font-awesome icon was specified | ||||
|  - Add configurable option for showing node label | ||||
|  - Avoid http redirects as Safari doesn't reuse Auth header Fixes #1903 | ||||
|  - Tidy up ace tooltip styling | ||||
|  - Add event log to editor | ||||
|  - Add tooltips to multiple editor elements | ||||
|  - Allow palette to be hidden | ||||
|  - Add node module into to sidebar and palette popover | ||||
|  - Mark all newly imported nodes as changed | ||||
|  - Allow a node label to be hidden | ||||
|  - Add markdown formatting toolbar | ||||
|  - Add markdown toolbar to various editors | ||||
|  - Fix i18n handling for ja-JP locale on Safari/MacOS | ||||
|  - Add node body tooltip | ||||
|  - Decrease opacity of flow-navigator | ||||
|  - Update tooltip style | ||||
|  - Update ACE to 1.4.1-src-min-noconflict | ||||
|  - Cache node locales by language | ||||
|  - Show icon element with either icon image or fa-icon | ||||
|  - Added font-awesome icons to user defined icon | ||||
|  - Update info side bar with node description section | ||||
|  - One-click search of config node users | ||||
|  - Redesign node edit dialog to tabbed style | ||||
|  - Add 'restart flows' option to deploy menu | ||||
|  - Add node description property UI | ||||
|  | ||||
|  | ||||
| Runtime | ||||
|  | ||||
|  - Allow a project to be loaded from cmdline | ||||
|  - Handle lookup of undefined property in Global context Fixes #1978 | ||||
|  - Refuse to enable Manage Palette if npm too old | ||||
|  - Remove restriction on upgrading non-local modules | ||||
|  - Remove deprecated Buffer constructor usage Fixes #1709 | ||||
|  - Update httpServerOptions doc in settings.js | ||||
|  - Exclude non-testable .js files from the unit tests | ||||
|  - Add --safe mode flag to allow starting without flows running | ||||
|  - Add setting-defined accessToken for automated access to the adminAPI - #1789 | ||||
|  | ||||
| Nodes | ||||
|  | ||||
|  - Move all core node EN help to their own locale files - #1990 | ||||
|  - CSV: better regex for number detection | ||||
|  - Debug: hide button if not configured to send to sidebar | ||||
|  - Delay: report queue activity when in by-topic mode | ||||
|  - Delay: add msg.flush mode | ||||
|  - Exec: Preserve existing properties on msg object | ||||
|  - File: remove CR/LF from incoming filename | ||||
|  - Function: create custom ace javascript mode to handle ES6 Fixes #1911 | ||||
|  - Function: add env.get | ||||
|  - HTTP Request: Add http-proxy config #1913 | ||||
|  - HTTP Request: add msg.redirectList to output | ||||
|  - HTTP Request: add msg.requestTimeout option for per-message setting - @natcl #1959 | ||||
|  - MQTT: add auto-detect and base64 output to mqtt node Fixes #1912 - @DurandA | ||||
|  - MQTT: only unsubscribe node that is being removed | ||||
|  - Sentiment: move to node-red-node-sentiment | ||||
|  - Switch: add missing edit dialog icon | ||||
|  - Tail: move to node-red-node-tail | ||||
|  - TCPGet: clear status if user changes target per message | ||||
|  - Template: tidy up edit dialog | ||||
|  - UDP: more resilient binding to correct port for udp, give input side priority | ||||
|  - Split/Join: add msg.reset to info panel | ||||
|  - Split/Join: reset join without sending part array | ||||
|  - Watch: add msg.filename so can feed direct to file in node | ||||
|  - WebSocket: preserve \_session on msg but don't send as part of wholemsg | ||||
|  | ||||
|  | ||||
| #### 0.19.5: Maintenance Release | ||||
|  | ||||
|  - Recognize pip installs of RPi.GPIO (#1934) | ||||
|  - Merge pull request #1941 from node-red-hitachi/master-batch | ||||
|  - Merge pull request #1931 from node-red-hitachi/master-typedinput | ||||
|  - Set min value of properties and spinners for batch | ||||
|  - Fix that unnecessary optionMenu remains | ||||
|  - Merge pull request #1894 from node-red-hitachi/fix-overlapping-file-node-execution | ||||
|  - Merge pull request #1924 from imZack/patch-1 | ||||
|  - Add missing comma | ||||
|  - Do not disable context sidebar during node edit Fixes #1921 | ||||
|  - Don't allow virtual links to be spliced Fixes #1920 | ||||
|  - Merge project package changes to avoid overwritten changes | ||||
|  - Handle manually added project deps that are unused Fixes #1908 | ||||
|  - update close & input handling of File node | ||||
|  - make close handler argument only one | ||||
|  - Merge pull request #1907 from amilajack/patch-2 | ||||
|  - Change repo badge to point to master branch | ||||
|  - invoke callbacks if async handler is specified | ||||
|  - Merge pull request #1891 from camlow325/resolve-example-path-for-windows-support | ||||
|  - Merge pull request #1900 from kazuhitoyokoi/master-addtestcases4settings.js | ||||
|  - wait closing while pending messages exist | ||||
|  - Add test cases for red/api/editor/settings.js | ||||
|  - Ensure all palette categories are opened properly Closes #1893 | ||||
|  - Resolve path when sending example file for Windows support | ||||
|  - fix multiple input message processing of file node | ||||
|  | ||||
| #### 0.19.4: Maintenance Release | ||||
|  | ||||
|  - Fix race condition in non-cache lfs context Fixes #1888 | ||||
|  - LocalFileSystem Context: Remove extra flush code | ||||
|  - Prevent race condition in caching mode of lfs context (#1889) | ||||
|  - Allow context store name to be provided in the key | ||||
|  - Switch node: only use promises when absolutely necessary | ||||
|  - Fix dbl-click handling on webkit-based browsers | ||||
|  - Ensure context.flow/global cannot be deleted or enumerated | ||||
|  - Handle context.get with multiple levels of unknown key Fixes #1883 | ||||
|  - Fix global.get("foo.bar") for functionGlobalContext set values | ||||
|  - Fix node color bug (#1877) | ||||
|  - Merge pull request #1857 from cclauss/patch-1 | ||||
|  - Define raw_input() in Python 3 & fix time.sleep() | ||||
|  | ||||
| #### 0.19.3: Maintenance Release | ||||
|  | ||||
|  - Split node - fix complete to send msg for k/v object | ||||
|  - Remove unused Join node merged object key typed input | ||||
|  - Set the JavaScript editor to full-screen | ||||
|  - Filter global modules installed locally | ||||
|  - Add svg to permitted icon extension list | ||||
|  - Debug node - indicate status all the time if selected to do so | ||||
|  - pi nodes - increase test coverage slightly | ||||
|  - TCP-request node - only write payload | ||||
|  - JSON schema: perform validation when obj -> obj or str -> str | ||||
|  - JSON schema: add draft-06 support (via $schema keyword) | ||||
|  - Mqtt proxy configuration for websocket connection, #1651. | ||||
|  - Allows MQTT Shared Subscriptions for MQTT-In core node | ||||
|  - Fix use of HTML tag or CSS class specification as icon of typedInput | ||||
|  | ||||
| #### 0.19.2: Maintenance Release | ||||
|  | ||||
|  - Ensure node default colour is used if palette.theme has no match | ||||
|  - fix lost messages / properties in TCPRequest Node; closes #1863 (#1864) | ||||
|  - Fix typo in template.html | ||||
|  - Improve error reporting from context plugin loading | ||||
|  - Prevent no-op edit of node marking as changed due to icon | ||||
|  - Change node must handle empty rule set | ||||
|  | ||||
| #### 0.19.1: Maintenance Release | ||||
|  | ||||
|  - Pull in latest twitter node | ||||
|  - Handle windows paths for context storage | ||||
|  - Handle persisting objects with circular refs in context | ||||
|  - Ensure js editor can expand to fill available space | ||||
|  - Add example localfilesystem contextStorage to settings | ||||
|  - Fix template node handling of nested context tags | ||||
|  | ||||
| #### 0.19: Milestone Release | ||||
|  | ||||
| Editor | ||||
|  | ||||
|  - Add editorTheme.palette.theme to allow overriding colours | ||||
|  - Index all node properties when searching Fixes #1446 | ||||
|  - Handle NaN and Infinity properly in debug sidebar Fixes #1778 #1779 | ||||
|  - Prevent horizontal scroll when palette name cannot wrap | ||||
|  - Ignore middle-click on node/ports to enable panning | ||||
|  - Better wire layout when looping back | ||||
|  - fix appearence of retry button of remote branch management dialog | ||||
|  - Handle releasing ctrl when using quick-add node dialog | ||||
|  - Add $env function to JSONata expressions | ||||
|  - Widen support for env var to use ${} or $() syntax | ||||
|  - Add env-var support to TypedInput | ||||
|  - Show unknown node properties in info tab | ||||
|  - Add node icon picker widget | ||||
|  - Only edit nodes on dbl click on primary button with no modifiers | ||||
|  - Allow subflows to be put in any palette category | ||||
|  - Add flow navigator widget | ||||
|  - Cache flow library result to improve response time Fixes #1753 | ||||
|  - Add middle-button-drag to pan the workspace | ||||
|  - allow multi-line category name in editor | ||||
|  - Redesign sidebar tabs | ||||
|  - Do not disable the export-clipboard menu option with empty selection | ||||
|  | ||||
| Nodes | ||||
|  | ||||
|  - Change: Ensure runtime errors in Change node can be caught Fixes #1769 | ||||
|  - File: Add output to File Out node | ||||
|  - Function: add expandable JavaScript editor pane | ||||
|  - Function: allow id and name reference in function node code (#1731) | ||||
|  - HTTP Request: Move to request module | ||||
|  - HTTP: Ensure apiMaxLength applies to HTTP Nodes Fixes #1278 | ||||
|  - Join: accumulate top level properties | ||||
|  - Join: allow environment variable as reduce init value | ||||
|  - JSON: add JSON schema validation via msg.schema | ||||
|  - Pi: Let nrgpio code work with python 3 | ||||
|  - Pi: let Pi nodes be visible/editable on all platforms | ||||
|  - Switch: add isEmpty rule | ||||
|  - TCP: queue messages while connecting; closes #1414 | ||||
|  - TLS: Add servername option to TLS config node for SNI Fixes #1805 | ||||
|  - UDP: Don't accidentally re-use udp port when set to not do so | ||||
|  | ||||
| Persistent Context | ||||
|  | ||||
|  - Add Context data sidebar | ||||
|  - Add persistable context option | ||||
|  - Add default memory store | ||||
|  - Add file-based context store | ||||
|  - Add async mode to evaluateJSONataExpression | ||||
|  - Update RED.util.evaluateNodeProperty to support context stores | ||||
|  | ||||
| Runtime | ||||
|  | ||||
|  - Support flow.disabled and .info in /flow API | ||||
|  - Node errors should be Strings not Errors Fixes #1781 | ||||
|  - Add detection of connection timeout in git communication Fixes #1770 | ||||
|  - Handle loading empty nodesDir | ||||
|  - Add 'private' property to userDir generated package.json | ||||
|  - Add RED.require to allow nodes to access other modules | ||||
|  - Ensure add/remove modules are run sequentially | ||||
|  | ||||
| #### 0.18.7: Maintenance Release | ||||
|  | ||||
| Editor Fixes | ||||
|  | ||||
|  - Do not trim wires if node declares outputs in defaults but misses value Fixes #1737 | ||||
|  | ||||
| Node Fixes | ||||
|  | ||||
|  - Relax twitter node version ready for major version bump | ||||
|  - Pass Date into the Function node sandbox to fix instanceof tests | ||||
|  - let TCP in node report remote ip and port when in single packet mode | ||||
|  - typo fix in node help (#1735) | ||||
|  | ||||
| Other Fixes | ||||
|  - Tidy up default grunt task and fixup test break due to reorder Fixes #1738 | ||||
|  - Bump jsonata version | ||||
|  | ||||
| #### 0.18.6: Maintenance Release | ||||
|  | ||||
| Editor Fixes | ||||
|  | ||||
|  - Handle a node having wires in the editor on ports it no longer has Fixes #1724 | ||||
|  - Add missing ACE snippet files | ||||
|  - Fix wireClippedNodes is not defined Fixes #1726 | ||||
|  - Split node html to isolate bad nodes when loading | ||||
|  - Avoid unnecessary use of .html() where .text() will do | ||||
|  | ||||
|  - Add editorTheme.projects.enabled to default settings.js" | ||||
|  | ||||
| #### 0.18.5: Maintenance Release | ||||
|  | ||||
| Projects | ||||
|  | ||||
|  - Add clone project to welcome screen | ||||
|  - Handle cloning a project without package.json | ||||
|  - Keep remote branch state in sync between editor and runtime | ||||
|  | ||||
| New Features | ||||
|  | ||||
|  - Add type checks to switch node options (#1714) | ||||
|  - add output property select to HTML parse node (#1701) | ||||
|  - Add Prevent Following Redirect to HTTP Request node (#615) (#1684) | ||||
|  - Add debug and trace functions to function node (#1654) | ||||
|  - Enable user defined icon for subflow | ||||
|  - Add MQTT disconnect message and rework broker node UI (#1719) | ||||
|  - Japanese message catalogue updates (#1723) | ||||
|  - Show node load errors in the Palette Manager view | ||||
|  | ||||
| Editor Fixes | ||||
|  | ||||
|  - Highlight subflow node when log msg comes from inside Fixes #1698 | ||||
|  - Ensure node wires array is not longer than outputs value Fixes #1678 | ||||
|  - Allow importing an unknown config node to be undone Fixes #1681 | ||||
|  - Ensure keyboard shortcuts get saved in runtime settings Fixes #1696 | ||||
|  - Don't mark a subflow changed when actually modified nothing (#1665) | ||||
|  | ||||
| Node Fixes | ||||
|  | ||||
|  - bind to correct port when doing udp broadcast/multicast (#1686) | ||||
|  - Provide full error stack in Function node log message (#1700) | ||||
|  - Fix http request doc type Fixes #1690 | ||||
|  - Make debug slightly larger to pass WCAG AA rating | ||||
|  - Make core nodes labels more consistent, to close #1673 | ||||
|  - Allow template node to be updated more than once Fixes #1671 | ||||
|  - Fix the problem that output labels of switch node sometimes disappear (#1664) | ||||
|  - Chinese translations for core nodes (#1607) | ||||
|  | ||||
| Runtime Fixes | ||||
|  | ||||
|  - Handle and display for invalid flow credentials when project is disabled #1689 (#1694) | ||||
|  - node-red-pi: fix behavior with old bash version (#1713) | ||||
|  - Fix ENOENT error on first start when no user dir (#1711) | ||||
|  - Handle null error object in Flow.handleError Fixes #1721 | ||||
|  - update settings comments to describe how to setup for ipv6 (#1675) | ||||
|  - Remove credential props after diffing flow to prevent future false positives Fixes #1359 | ||||
|  - Log error if settings unavailable when saving user settings Fixes #1645 | ||||
|  - Keep backup of .config.json | ||||
|  - Add warning if using \_credentialSecret from .config.json | ||||
|  - Filter req.user in /settings to prevent potentially leaking info | ||||
|  | ||||
| #### 0.18.4: Maintenance Release | ||||
|  | ||||
| Projects | ||||
|   | ||||
| @@ -30,13 +30,13 @@ At a minimum, please include: | ||||
|  | ||||
| ## Feature requests | ||||
|  | ||||
| For feature requests, please raise them on the [forum](https://discourse.nodered.org). | ||||
| For feature requests, please raise them on the [mailing list](https://groups.google.com/forum/#!forum/node-red). | ||||
|  | ||||
| ## Pull-Requests | ||||
|  | ||||
| If you want to raise a pull-request with a new feature, or a refactoring | ||||
| of existing code, it may well get rejected if you haven't discussed it on | ||||
| the [forum](https://discourse.nodered.org) first. | ||||
| the [mailing list](https://groups.google.com/forum/#!forum/node-red) first. | ||||
|  | ||||
| All contributors need to sign the JS Foundation's Contributor License Agreement. | ||||
| It is an online process and quick to do. You can read the details of the agreement | ||||
|   | ||||
							
								
								
									
										358
									
								
								Gruntfile.js
									
									
									
									
									
								
							
							
						
						| @@ -24,10 +24,6 @@ module.exports = function(grunt) { | ||||
|         nodemonArgs.push(flowFile); | ||||
|     } | ||||
|  | ||||
|     var nonHeadless = grunt.option('non-headless'); | ||||
|     if (nonHeadless) { | ||||
|         process.env.NODE_RED_NON_HEADLESS = 'true'; | ||||
|     } | ||||
|     grunt.initConfig({ | ||||
|         pkg: grunt.file.readJSON('package.json'), | ||||
|         paths: { | ||||
| @@ -42,7 +38,7 @@ module.exports = function(grunt) { | ||||
|                 reporter: 'spec' | ||||
|             }, | ||||
|             all: { src: ['test/**/*_spec.js'] }, | ||||
|             core: { src: ["test/_spec.js","test/unit/**/*_spec.js"]}, | ||||
|             core: { src: ["test/_spec.js","test/red/**/*_spec.js"]}, | ||||
|             nodes: { src: ["test/nodes/**/*_spec.js"]} | ||||
|         }, | ||||
|         webdriver: { | ||||
| @@ -57,11 +53,10 @@ module.exports = function(grunt) { | ||||
|                 ignoreLeaks: false, | ||||
|                 ui: 'bdd', | ||||
|                 reportFormats: ['lcov','html'], | ||||
|                 print: 'both', | ||||
|                 istanbulOptions: ['--no-default-excludes', '-i','**/packages/node_modules/**'] | ||||
|                 print: 'both' | ||||
|             }, | ||||
|             all: { src: ["test/unit/_spec.js","test/unit/**/*_spec.js","test/nodes/**/*_spec.js"] }, | ||||
|             core: { src: ["test/unit/_spec.js","test/unit/**/*_spec.js"]}, | ||||
|             all: { src: ['test/**/*_spec.js'] }, | ||||
|             core: { src: ["test/_spec.js","test/red/**/*_spec.js"]}, | ||||
|             nodes: { src: ["test/nodes/**/*_spec.js"]} | ||||
|         }, | ||||
|         jshint: { | ||||
| @@ -81,14 +76,16 @@ module.exports = function(grunt) { | ||||
|             all: [ | ||||
|                 'Gruntfile.js', | ||||
|                 'red.js', | ||||
|                 'packages/**/*.js' | ||||
|                 'red/**/*.js', | ||||
|                 'nodes/core/*/*.js', | ||||
|                 'editor/js/**/*.js' | ||||
|             ], | ||||
|             core: { | ||||
|                 files: { | ||||
|                     src: [ | ||||
|                         'Gruntfile.js', | ||||
|                         'red.js', | ||||
|                         'packages/**/*.js', | ||||
|                         'red/**/*.js' | ||||
|                     ] | ||||
|                 } | ||||
|             }, | ||||
| @@ -119,84 +116,78 @@ module.exports = function(grunt) { | ||||
|                 src: [ | ||||
|                     // Ensure editor source files are concatenated in | ||||
|                     // the right order | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/jquery-addons.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/red.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/events.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/i18n.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/settings.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/user.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/comms.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/text/bidi.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/text/format.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/state.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/nodes.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/font-awesome.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/history.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/validators.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/utils.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/editableList.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/checkboxSet.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/menu.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/panels.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/popover.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/searchBox.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/tabs.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/stack.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/common/typedInput.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/actions.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/deploy.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/diff.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/keyboard.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/workspaces.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/view.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/view-navigator.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/sidebar.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/palette.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/tab-info.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/tab-config.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/tab-context.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/palette-editor.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/editor.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/editors/*.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/event-log.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/tray.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/clipboard.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/library.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/notifications.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/search.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/typeSearch.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/subflow.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/userSettings.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/projects/projects.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/projects/projectSettings.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/projects/projectUserSettings.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/projects/tab-versionControl.js", | ||||
|                     "packages/node_modules/@node-red/editor-client/src/js/ui/touch/radialMenu.js" | ||||
|                     "editor/js/red.js", | ||||
|                     "editor/js/events.js", | ||||
|                     "editor/js/i18n.js", | ||||
|                     "editor/js/settings.js", | ||||
|                     "editor/js/user.js", | ||||
|                     "editor/js/comms.js", | ||||
|                     "editor/js/text/bidi.js", | ||||
|                     "editor/js/text/format.js", | ||||
|                     "editor/js/ui/state.js", | ||||
|                     "editor/js/nodes.js", | ||||
|                     "editor/js/history.js", | ||||
|                     "editor/js/validators.js", | ||||
|                     "editor/js/ui/utils.js", | ||||
|                     "editor/js/ui/common/editableList.js", | ||||
|                     "editor/js/ui/common/checkboxSet.js", | ||||
|                     "editor/js/ui/common/menu.js", | ||||
|                     "editor/js/ui/common/panels.js", | ||||
|                     "editor/js/ui/common/popover.js", | ||||
|                     "editor/js/ui/common/searchBox.js", | ||||
|                     "editor/js/ui/common/tabs.js", | ||||
|                     "editor/js/ui/common/stack.js", | ||||
|                     "editor/js/ui/common/typedInput.js", | ||||
|                     "editor/js/ui/actions.js", | ||||
|                     "editor/js/ui/deploy.js", | ||||
|                     "editor/js/ui/diff.js", | ||||
|                     "editor/js/ui/keyboard.js", | ||||
|                     "editor/js/ui/workspaces.js", | ||||
|                     "editor/js/ui/view.js", | ||||
|                     "editor/js/ui/sidebar.js", | ||||
|                     "editor/js/ui/palette.js", | ||||
|                     "editor/js/ui/tab-info.js", | ||||
|                     "editor/js/ui/tab-config.js", | ||||
|                     "editor/js/ui/palette-editor.js", | ||||
|                     "editor/js/ui/editor.js", | ||||
|                     "editor/js/ui/tray.js", | ||||
|                     "editor/js/ui/clipboard.js", | ||||
|                     "editor/js/ui/library.js", | ||||
|                     "editor/js/ui/notifications.js", | ||||
|                     "editor/js/ui/search.js", | ||||
|                     "editor/js/ui/typeSearch.js", | ||||
|                     "editor/js/ui/subflow.js", | ||||
|                     "editor/js/ui/userSettings.js", | ||||
|                     "editor/js/ui/projects/projects.js", | ||||
|                     "editor/js/ui/projects/projectSettings.js", | ||||
|                     "editor/js/ui/projects/projectUserSettings.js", | ||||
|                     "editor/js/ui/projects/tab-versionControl.js", | ||||
|                     "editor/js/ui/touch/radialMenu.js" | ||||
|                 ], | ||||
|                 dest: "packages/node_modules/@node-red/editor-client/public/red/red.js" | ||||
|                 dest: "public/red/red.js" | ||||
|             }, | ||||
|             vendor: { | ||||
|                 files: { | ||||
|                     "packages/node_modules/@node-red/editor-client/public/vendor/vendor.js": [ | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/jquery/js/jquery-1.11.3.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/bootstrap/js/bootstrap.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/jquery/js/jquery-ui-1.10.3.custom.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/jquery/js/jquery.ui.touch-punch.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/marked/marked.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/d3/d3.v3.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/i18next/i18next.min.js" | ||||
|                     "public/vendor/vendor.js": [ | ||||
|                         "editor/vendor/jquery/js/jquery-1.11.3.min.js", | ||||
|                         "editor/vendor/bootstrap/js/bootstrap.min.js", | ||||
|                         "editor/vendor/jquery/js/jquery-ui-1.10.3.custom.min.js", | ||||
|                         "editor/vendor/jquery/js/jquery.ui.touch-punch.min.js", | ||||
|                         "editor/vendor/marked/marked.min.js", | ||||
|                         "editor/vendor/d3/d3.v3.min.js", | ||||
|                         "editor/vendor/i18next/i18next.min.js" | ||||
|                     ], | ||||
|                     "packages/node_modules/@node-red/editor-client/public/vendor/vendor.css": [ | ||||
|                     "public/vendor/vendor.css": [ | ||||
|                         // TODO: resolve relative resource paths in | ||||
|                         //       bootstrap/FA/jquery | ||||
|                     ], | ||||
|                     "packages/node_modules/@node-red/editor-client/public/vendor/jsonata/jsonata.min.js": [ | ||||
|                     "public/vendor/jsonata/jsonata.min.js": [ | ||||
|                         "node_modules/jsonata/jsonata-es5.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/jsonata/formatter.js" | ||||
|                         "editor/vendor/jsonata/formatter.js" | ||||
|                     ], | ||||
|                     "packages/node_modules/@node-red/editor-client/public/vendor/ace/worker-jsonata.js": [ | ||||
|                     "public/vendor/ace/worker-jsonata.js": [ | ||||
|                         "node_modules/jsonata/jsonata-es5.min.js", | ||||
|                         "packages/node_modules/@node-red/editor-client/src/vendor/jsonata/worker-jsonata.js" | ||||
|                         "editor/vendor/jsonata/worker-jsonata.js" | ||||
|                     ] | ||||
|                 } | ||||
|             } | ||||
| @@ -204,10 +195,10 @@ module.exports = function(grunt) { | ||||
|         uglify: { | ||||
|             build: { | ||||
|                 files: { | ||||
|                     'packages/node_modules/@node-red/editor-client/public/red/red.min.js': 'packages/node_modules/@node-red/editor-client/public/red/red.js', | ||||
|                     'packages/node_modules/@node-red/editor-client/public/red/main.min.js': 'packages/node_modules/@node-red/editor-client/public/red/main.js', | ||||
|                     'packages/node_modules/@node-red/editor-client/public/vendor/ace/mode-jsonata.js': 'packages/node_modules/@node-red/editor-client/src/vendor/jsonata/mode-jsonata.js', | ||||
|                     'packages/node_modules/@node-red/editor-client/public/vendor/ace/snippets/jsonata.js': 'packages/node_modules/@node-red/editor-client/src/vendor/jsonata/snippets-jsonata.js' | ||||
|                     'public/red/red.min.js': 'public/red/red.js', | ||||
|                     'public/red/main.min.js': 'public/red/main.js', | ||||
|                     'public/vendor/ace/mode-jsonata.js': 'editor/vendor/jsonata/mode-jsonata.js', | ||||
|                     'public/vendor/ace/snippets/jsonata.js': 'editor/vendor/jsonata/snippets-jsonata.js' | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
| @@ -217,50 +208,50 @@ module.exports = function(grunt) { | ||||
|                     outputStyle: 'compressed' | ||||
|                 }, | ||||
|                 files: [{ | ||||
|                     dest: 'packages/node_modules/@node-red/editor-client/public/red/style.min.css', | ||||
|                     src: 'packages/node_modules/@node-red/editor-client/src/sass/style.scss' | ||||
|                     dest: 'public/red/style.min.css', | ||||
|                     src: 'editor/sass/style.scss' | ||||
|                 }, | ||||
|                 { | ||||
|                     dest: 'packages/node_modules/@node-red/editor-client/public/vendor/bootstrap/css/bootstrap.min.css', | ||||
|                     src: 'packages/node_modules/@node-red/editor-client/src/vendor/bootstrap/css/bootstrap.css' | ||||
|                     dest: 'public/vendor/bootstrap/css/bootstrap.min.css', | ||||
|                     src: 'editor/vendor/bootstrap/css/bootstrap.css' | ||||
|                 }] | ||||
|             } | ||||
|         }, | ||||
|         jsonlint: { | ||||
|             messages: { | ||||
|                 src: [ | ||||
|                     'packages/node_modules/@node-red/nodes/locales/**/*.json', | ||||
|                     'packages/node_modules/@node-red/editor-client/locales/**/*.json', | ||||
|                     'packages/node_modules/@node-red/runtime/locales/**/*.json' | ||||
|                     'nodes/core/locales/en-US/messages.json', | ||||
|                     'red/api/locales/en-US/editor.json', | ||||
|                     'red/runtime/locales/en-US/runtime.json' | ||||
|                 ] | ||||
|             }, | ||||
|             keymaps: { | ||||
|                 src: [ | ||||
|                     'packages/node_modules/@node-red/editor-client/src/js/keymap.json' | ||||
|                     'editor/js/keymap.json' | ||||
|                 ] | ||||
|             } | ||||
|         }, | ||||
|         attachCopyright: { | ||||
|             js: { | ||||
|                 src: [ | ||||
|                     'packages/node_modules/@node-red/editor-client/public/red/red.min.js', | ||||
|                     'packages/node_modules/@node-red/editor-client/public/red/main.min.js' | ||||
|                     'public/red/red.min.js', | ||||
|                     'public/red/main.min.js' | ||||
|                 ] | ||||
|             }, | ||||
|             css: { | ||||
|                 src: [ | ||||
|                     'packages/node_modules/@node-red/editor-client/public/red/style.min.css' | ||||
|                     'public/red/style.min.css' | ||||
|                 ] | ||||
|             } | ||||
|         }, | ||||
|         clean: { | ||||
|             build: { | ||||
|                 src: [ | ||||
|                     "packages/node_modules/@node-red/editor-client/public/red", | ||||
|                     "packages/node_modules/@node-red/editor-client/public/index.html", | ||||
|                     "packages/node_modules/@node-red/editor-client/public/favicon.ico", | ||||
|                     "packages/node_modules/@node-red/editor-client/public/icons", | ||||
|                     "packages/node_modules/@node-red/editor-client/public/vendor" | ||||
|                     "public/red", | ||||
|                     "public/index.html", | ||||
|                     "public/favicon.ico", | ||||
|                     "public/icons", | ||||
|                     "public/vendor" | ||||
|                 ] | ||||
|             }, | ||||
|             release: { | ||||
| @@ -272,27 +263,27 @@ module.exports = function(grunt) { | ||||
|         watch: { | ||||
|             js: { | ||||
|                 files: [ | ||||
|                     'packages/node_modules/@node-red/editor-client/src/js/**/*.js' | ||||
|                     'editor/js/**/*.js' | ||||
|                 ], | ||||
|                 tasks: ['copy:build','concat','uglify','attachCopyright:js'] | ||||
|             }, | ||||
|             sass: { | ||||
|                 files: [ | ||||
|                     'packages/node_modules/@node-red/editor-client/src/sass/**/*.scss' | ||||
|                     'editor/sass/**/*.scss' | ||||
|                 ], | ||||
|                 tasks: ['sass','attachCopyright:css'] | ||||
|             }, | ||||
|             json: { | ||||
|                 files: [ | ||||
|                     'packages/node_modules/@node-red/nodes/locales/**/*.json', | ||||
|                     'packages/node_modules/@node-red/editor-client/locales/**/*.json', | ||||
|                     'packages/node_modules/@node-red/runtime/locales/**/*.json' | ||||
|                     'nodes/core/locales/en-US/messages.json', | ||||
|                     'red/api/locales/en-US/editor.json', | ||||
|                     'red/runtime/locales/en-US/runtime.json' | ||||
|                 ], | ||||
|                 tasks: ['jsonlint:messages'] | ||||
|             }, | ||||
|             keymaps: { | ||||
|                 files: [ | ||||
|                     'packages/node_modules/@node-red/editor-client/src/js/keymap.json' | ||||
|                     'editor/js/keymap.json' | ||||
|                 ], | ||||
|                 tasks: ['jsonlint:keymaps','copy:build'] | ||||
|             }, | ||||
| @@ -307,13 +298,12 @@ module.exports = function(grunt) { | ||||
|         nodemon: { | ||||
|             /* uses .nodemonignore */ | ||||
|             dev: { | ||||
|                 script: 'packages/node_modules/node-red/red.js', | ||||
|                 script: 'red.js', | ||||
|                 options: { | ||||
|                     args: nodemonArgs, | ||||
|                     ext: 'js,html,json', | ||||
|                     watch: [ | ||||
|                         'packages/node_modules', | ||||
|                         '!packages/node_modules/@node-red/editor-client' | ||||
|                         'red','nodes' | ||||
|                     ] | ||||
|                 } | ||||
|             } | ||||
| @@ -332,21 +322,21 @@ module.exports = function(grunt) { | ||||
|             build: { | ||||
|                 files:[ | ||||
|                     { | ||||
|                         src: 'packages/node_modules/@node-red/editor-client/src/js/main.js', | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/red/main.js' | ||||
|                         src: 'editor/js/main.js', | ||||
|                         dest: 'public/red/main.js' | ||||
|                     }, | ||||
|                     { | ||||
|                         src: 'packages/node_modules/@node-red/editor-client/src/js/keymap.json', | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/red/keymap.json' | ||||
|                         src: 'editor/js/keymap.json', | ||||
|                         dest: 'public/red/keymap.json' | ||||
|                     }, | ||||
|                     { | ||||
|                         cwd: 'packages/node_modules/@node-red/editor-client/src/images', | ||||
|                         cwd: 'editor/images', | ||||
|                         src: '**', | ||||
|                         expand: true, | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/red/images/' | ||||
|                         dest: 'public/red/images/' | ||||
|                     }, | ||||
|                     { | ||||
|                         cwd: 'packages/node_modules/@node-red/editor-client/src/vendor', | ||||
|                         cwd: 'editor/vendor', | ||||
|                         src: [ | ||||
|                             'ace/**', | ||||
|                             //'bootstrap/css/**', | ||||
| @@ -355,35 +345,46 @@ module.exports = function(grunt) { | ||||
|                             'font-awesome/**' | ||||
|                         ], | ||||
|                         expand: true, | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/vendor/' | ||||
|                         dest: 'public/vendor/' | ||||
|                     }, | ||||
|                     { | ||||
|                         cwd: 'packages/node_modules/@node-red/editor-client/src/icons', | ||||
|                         cwd: 'editor/icons', | ||||
|                         src: '**', | ||||
|                         expand: true, | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/icons/' | ||||
|                         dest: 'public/icons/' | ||||
|                     }, | ||||
|                     { | ||||
|                         expand: true, | ||||
|                         src: ['packages/node_modules/@node-red/editor-client/src/index.html','packages/node_modules/@node-red/editor-client/src/favicon.ico'], | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/', | ||||
|                         src: ['editor/index.html','editor/favicon.ico'], | ||||
|                         dest: 'public/', | ||||
|                         flatten: true | ||||
|                     }, | ||||
|                     { | ||||
|                         src: 'CHANGELOG.md', | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/red/about' | ||||
|                     }, | ||||
|                     { | ||||
|                         src: 'CHANGELOG.md', | ||||
|                         dest: 'packages/node_modules/node-red/' | ||||
|                     }, | ||||
|                     { | ||||
|                         cwd: 'packages/node_modules/@node-red/editor-client/src/ace/bin/', | ||||
|                         src: '**', | ||||
|                         expand: true, | ||||
|                         dest: 'packages/node_modules/@node-red/editor-client/public/vendor/ace/' | ||||
|                         dest: 'public/red/about' | ||||
|                     } | ||||
|                 ] | ||||
|             }, | ||||
|             release: { | ||||
|                 files: [{ | ||||
|                     mode: true, | ||||
|                     expand: true, | ||||
|                     src: [ | ||||
|                         '*.md', | ||||
|                         'LICENSE', | ||||
|                         'package.json', | ||||
|                         'settings.js', | ||||
|                         'red.js', | ||||
|                         'lib/.gitignore', | ||||
|                         'nodes/*.demo', | ||||
|                         'nodes/core/**', | ||||
|                         'red/**', | ||||
|                         'public/**', | ||||
|                         'editor/templates/**', | ||||
|                         'bin/**' | ||||
|                     ], | ||||
|                     dest: path.resolve('<%= paths.dist %>/node-red-<%= pkg.version %>') | ||||
|                 }] | ||||
|             } | ||||
|         }, | ||||
|         chmod: { | ||||
| @@ -392,79 +393,19 @@ module.exports = function(grunt) { | ||||
|             }, | ||||
|             release: { | ||||
|                 src: [ | ||||
|                     "packages/node_modules/@node-red/nodes/core/hardware/nrgpio", | ||||
|                     "packages/node_modules/@node-red/runtime/lib/storage/localfilesystem/projects/git/node-red-*sh" | ||||
|                     path.resolve('<%= paths.dist %>/node-red-<%= pkg.version %>/nodes/core/hardware/nrgpio*'), | ||||
|                     path.resolve('<%= paths.dist %>/node-red-<%= pkg.version %>/red/runtime/storage/localfilesystem/projects/git/node-red-*sh') | ||||
|                 ] | ||||
|             } | ||||
|         }, | ||||
|         'npm-command': { | ||||
|             options: { | ||||
|                 cmd: "pack", | ||||
|                 cwd: "<%= paths.dist %>/modules" | ||||
|             }, | ||||
|             'node-red': { options: { args: [__dirname+'/packages/node_modules/node-red'] } }, | ||||
|             '@node-red/editor-api': { options: { args: [__dirname+'/packages/node_modules/@node-red/editor-api'] } }, | ||||
|             '@node-red/editor-client': { options: { args: [__dirname+'/packages/node_modules/@node-red/editor-client'] } }, | ||||
|             '@node-red/nodes': { options: { args: [__dirname+'/packages/node_modules/@node-red/nodes'] } }, | ||||
|             '@node-red/registry': { options: { args: [__dirname+'/packages/node_modules/@node-red/registry'] } }, | ||||
|             '@node-red/runtime': { options: { args: [__dirname+'/packages/node_modules/@node-red/runtime'] } }, | ||||
|             '@node-red/util': { options: { args: [__dirname+'/packages/node_modules/@node-red/util'] } } | ||||
|  | ||||
|  | ||||
|         }, | ||||
|         mkdir: { | ||||
|             release: { | ||||
|                 options: { | ||||
|                     create: ['<%= paths.dist %>/modules'] | ||||
|                 }, | ||||
|             }, | ||||
|         }, | ||||
|         compress: { | ||||
|             release: { | ||||
|                 options: { | ||||
|                     archive: '<%= paths.dist %>/node-red-<%= pkg.version %>.zip' | ||||
|                 }, | ||||
|                 expand: true, | ||||
|                 cwd: 'packages/node_modules/', | ||||
|                 src: [ | ||||
|                     '**', | ||||
|                     '!@node-red/editor-client/src/**' | ||||
|                 ] | ||||
|             } | ||||
|         }, | ||||
|         jsdoc : { | ||||
|             modules: { | ||||
|                 src: [ | ||||
|                     'packages/node_modules/node-red/lib/red.js', | ||||
|                     'packages/node_modules/@node-red/runtime/lib/index.js', | ||||
|                     'packages/node_modules/@node-red/runtime/lib/api/*.js', | ||||
|                     'packages/node_modules/@node-red/runtime/lib/events.js', | ||||
|                     'packages/node_modules/@node-red/util/**/*.js', | ||||
|                     ], | ||||
|                 options: { | ||||
|                     destination: 'docs', | ||||
|                     configure: './jsdoc.json' | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         jsdoc2md: { | ||||
|             runtimeAPI: { | ||||
|                 options: { | ||||
|                     separators: true | ||||
|                 }, | ||||
|                 src: [ | ||||
|                     'packages/node_modules/@node-red/runtime/lib/index.js', | ||||
|                     'packages/node_modules/@node-red/runtime/lib/api/*.js', | ||||
|                     'packages/node_modules/@node-red/runtime/lib/events.js' | ||||
|                 ], | ||||
|                 dest: 'packages/node_modules/@node-red/runtime/docs/api.md' | ||||
|             }, | ||||
|             nodeREDUtil: { | ||||
|                 options: { | ||||
|                     separators: true | ||||
|                 }, | ||||
|                 src: 'packages/node_modules/@node-red/util/**/*.js', | ||||
|                 dest: 'packages/node_modules/@node-red/util/docs/api.md' | ||||
|                 cwd: '<%= paths.dist %>/', | ||||
|                 src: ['node-red-<%= pkg.version %>/**'] | ||||
|             } | ||||
|         } | ||||
|     }); | ||||
| @@ -484,10 +425,6 @@ module.exports = function(grunt) { | ||||
|     grunt.loadNpmTasks('grunt-jsonlint'); | ||||
|     grunt.loadNpmTasks('grunt-mocha-istanbul'); | ||||
|     grunt.loadNpmTasks('grunt-webdriver'); | ||||
|     grunt.loadNpmTasks('grunt-jsdoc'); | ||||
|     grunt.loadNpmTasks('grunt-jsdoc-to-markdown'); | ||||
|     grunt.loadNpmTasks('grunt-npm-command'); | ||||
|     grunt.loadNpmTasks('grunt-mkdir'); | ||||
|  | ||||
|     grunt.registerMultiTask('attachCopyright', function() { | ||||
|         var files = this.data.src; | ||||
| @@ -529,18 +466,6 @@ module.exports = function(grunt) { | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     grunt.registerTask('verifyPackageDependencies', function() { | ||||
|         var done = this.async(); | ||||
|         var verifyDependencies = require("./scripts/verify-package-dependencies.js"); | ||||
|         verifyDependencies().then(function(failures) { | ||||
|             if (failures.length > 0) { | ||||
|                 failures.forEach(f => grunt.log.error(f)); | ||||
|                 grunt.fail.fatal("Failed to verify package dependencies"); | ||||
|             } | ||||
|             done(); | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     grunt.registerTask('setDevEnv', | ||||
|         'Sets NODE_ENV=development so non-minified assets are used', | ||||
|             function () { | ||||
| @@ -549,7 +474,7 @@ module.exports = function(grunt) { | ||||
|  | ||||
|     grunt.registerTask('default', | ||||
|         'Builds editor content then runs code style checks and unit tests on all components', | ||||
|         ['build','verifyPackageDependencies','jshint:editor','mocha_istanbul:all']); | ||||
|         ['build','test-core','test-editor','test-nodes']); | ||||
|  | ||||
|     grunt.registerTask('test-core', | ||||
|         'Runs code style check and unit tests on core runtime code', | ||||
| @@ -577,18 +502,9 @@ module.exports = function(grunt) { | ||||
|  | ||||
|     grunt.registerTask('release', | ||||
|         'Create distribution zip file', | ||||
|         ['build','verifyPackageDependencies','clean:release','mkdir:release','chmod:release','compress:release','pack-modules']); | ||||
|  | ||||
|     grunt.registerTask('pack-modules', | ||||
|         'Create module pack files for release', | ||||
|         ['mkdir:release','npm-command']); | ||||
|  | ||||
|         ['build','clean:release','copy:release','chmod:release','compress:release']); | ||||
|  | ||||
|     grunt.registerTask('coverage', | ||||
|         'Run Istanbul code test coverage task', | ||||
|         ['build','mocha_istanbul:all']); | ||||
|  | ||||
|     grunt.registerTask('docs', | ||||
|         'Generates API documentation', | ||||
|         ['jsdoc','jsdoc2md']); | ||||
| }; | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| http://nodered.org | ||||
|  | ||||
| [](https://travis-ci.org/node-red/node-red) | ||||
| [](https://travis-ci.org/node-red/node-red) | ||||
| [](https://coveralls.io/r/node-red/node-red?branch=master) | ||||
|  | ||||
| A visual tool for wiring the Internet of Things. | ||||
| @@ -22,7 +22,8 @@ started. | ||||
|  | ||||
| More documentation can be found [here](http://nodered.org/docs). | ||||
|  | ||||
| For further help, or general discussion, please use the [Node-RED Forum](https://discourse.nodered.org) or [slack team](https://nodered.org/slack). | ||||
| For further help, or general discussion, please use the | ||||
| [mailing list](https://groups.google.com/forum/#!forum/node-red). | ||||
|  | ||||
| ## Developers | ||||
|  | ||||
| @@ -44,6 +45,9 @@ If you want to run the latest code from git, here's how to get started: | ||||
| 4. Run | ||||
|  | ||||
|         npm start | ||||
|    or | ||||
|  | ||||
|         node red.js | ||||
|  | ||||
| ## Contributing | ||||
|  | ||||
|   | ||||
| @@ -31,7 +31,7 @@ done | ||||
| # Find the real location of this script | ||||
| CURRENT_PATH=`pwd` | ||||
| SCRIPT_PATH="${BASH_SOURCE[0]}"; | ||||
| while [ -h "${SCRIPT_PATH}" ]; do | ||||
| while([ -h "${SCRIPT_PATH}" ]); do | ||||
|     cd "`dirname "${SCRIPT_PATH}"`" | ||||
|     SCRIPT_PATH="$(readlink "`basename "${SCRIPT_PATH}"`")"; | ||||
| done | ||||
| Before Width: | Height: | Size: 17 KiB After Width: | Height: | Size: 17 KiB | 
| Before Width: | Height: | Size: 308 B After Width: | Height: | Size: 308 B | 
| Before Width: | Height: | Size: 603 B After Width: | Height: | Size: 603 B | 
| Before Width: | Height: | Size: 393 B After Width: | Height: | Size: 393 B | 
| Before Width: | Height: | Size: 712 B After Width: | Height: | Size: 712 B | 
| Before Width: | Height: | Size: 2.3 KiB After Width: | Height: | Size: 2.3 KiB | 
| Before Width: | Height: | Size: 508 B After Width: | Height: | Size: 508 B | 
| Before Width: | Height: | Size: 575 B After Width: | Height: | Size: 575 B | 
| Before Width: | Height: | Size: 493 B After Width: | Height: | Size: 493 B | 
| Before Width: | Height: | Size: 601 B After Width: | Height: | Size: 601 B | 
| Before Width: | Height: | Size: 459 B After Width: | Height: | Size: 459 B | 
| Before Width: | Height: | Size: 218 B After Width: | Height: | Size: 218 B | 
| Before Width: | Height: | Size: 324 B After Width: | Height: | Size: 324 B | 
| Before Width: | Height: | Size: 378 B After Width: | Height: | Size: 378 B | 
| Before Width: | Height: | Size: 255 B After Width: | Height: | Size: 255 B | 
| Before Width: | Height: | Size: 457 B After Width: | Height: | Size: 457 B | 
| Before Width: | Height: | Size: 502 B After Width: | Height: | Size: 502 B | 
| Before Width: | Height: | Size: 449 B After Width: | Height: | Size: 449 B | 
| Before Width: | Height: | Size: 253 B After Width: | Height: | Size: 253 B | 
| Before Width: | Height: | Size: 1.6 KiB After Width: | Height: | Size: 1.6 KiB | 
| Before Width: | Height: | Size: 639 B After Width: | Height: | Size: 639 B | 
| Before Width: | Height: | Size: 402 B After Width: | Height: | Size: 402 B | 
| Before Width: | Height: | Size: 414 B After Width: | Height: | Size: 414 B | 
| Before Width: | Height: | Size: 671 B After Width: | Height: | Size: 671 B | 
| Before Width: | Height: | Size: 386 B After Width: | Height: | Size: 386 B | 
| Before Width: | Height: | Size: 386 B After Width: | Height: | Size: 386 B | 
| Before Width: | Height: | Size: 413 B After Width: | Height: | Size: 413 B | 
| Before Width: | Height: | Size: 393 B After Width: | Height: | Size: 393 B | 
| Before Width: | Height: | Size: 467 B After Width: | Height: | Size: 467 B | 
| Before Width: | Height: | Size: 393 B After Width: | Height: | Size: 393 B | 
| Before Width: | Height: | Size: 423 B After Width: | Height: | Size: 423 B | 
| Before Width: | Height: | Size: 360 B After Width: | Height: | Size: 360 B | 
| Before Width: | Height: | Size: 736 B After Width: | Height: | Size: 736 B | 
| Before Width: | Height: | Size: 482 B After Width: | Height: | Size: 482 B | 
| Before Width: | Height: | Size: 273 B After Width: | Height: | Size: 273 B | 
| Before Width: | Height: | Size: 793 B After Width: | Height: | Size: 793 B | 
| Before Width: | Height: | Size: 256 B After Width: | Height: | Size: 256 B | 
| Before Width: | Height: | Size: 439 B After Width: | Height: | Size: 439 B | 
| Before Width: | Height: | Size: 592 B After Width: | Height: | Size: 592 B | 
| Before Width: | Height: | Size: 509 B After Width: | Height: | Size: 509 B | 
| Before Width: | Height: | Size: 488 B After Width: | Height: | Size: 488 B | 
| Before Width: | Height: | Size: 628 B After Width: | Height: | Size: 628 B | 
| Before Width: | Height: | Size: 258 B After Width: | Height: | Size: 258 B | 
| Before Width: | Height: | Size: 404 B After Width: | Height: | Size: 404 B | 
| Before Width: | Height: | Size: 591 B After Width: | Height: | Size: 591 B | 
| Before Width: | Height: | Size: 707 B After Width: | Height: | Size: 707 B | 
| Before Width: | Height: | Size: 291 B After Width: | Height: | Size: 291 B | 
| Before Width: | Height: | Size: 386 B After Width: | Height: | Size: 386 B | 
| Before Width: | Height: | Size: 289 B After Width: | Height: | Size: 289 B | 
| Before Width: | Height: | Size: 368 B After Width: | Height: | Size: 368 B | 
| Before Width: | Height: | Size: 290 B After Width: | Height: | Size: 290 B | 
| Before Width: | Height: | Size: 392 B After Width: | Height: | Size: 392 B | 
| Before Width: | Height: | Size: 192 B After Width: | Height: | Size: 192 B | 
| Before Width: | Height: | Size: 8.3 KiB After Width: | Height: | Size: 8.3 KiB | 
| Before Width: | Height: | Size: 2.3 KiB After Width: | Height: | Size: 2.3 KiB | 
| Before Width: | Height: | Size: 3.3 KiB After Width: | Height: | Size: 3.3 KiB | 
| Before Width: | Height: | Size: 1019 B After Width: | Height: | Size: 1019 B | 
| Before Width: | Height: | Size: 600 B After Width: | Height: | Size: 600 B | 
| Before Width: | Height: | Size: 410 B After Width: | Height: | Size: 410 B | 
| Before Width: | Height: | Size: 638 B After Width: | Height: | Size: 638 B | 
| Before Width: | Height: | Size: 546 B After Width: | Height: | Size: 546 B | 
| Before Width: | Height: | Size: 638 B After Width: | Height: | Size: 638 B | 
| Before Width: | Height: | Size: 646 B After Width: | Height: | Size: 646 B | 
| Before Width: | Height: | Size: 563 B After Width: | Height: | Size: 563 B | 
| Before Width: | Height: | Size: 588 B After Width: | Height: | Size: 588 B | 
| Before Width: | Height: | Size: 502 B After Width: | Height: | Size: 502 B | 
| @@ -28,24 +28,14 @@ RED.comms = (function() { | ||||
| 
 | ||||
|     function connectWS() { | ||||
|         active = true; | ||||
|         var wspath; | ||||
| 
 | ||||
|         if (RED.settings.apiRootUrl) { | ||||
|             var m = /^(https?):\/\/(.*)$/.exec(RED.settings.apiRootUrl); | ||||
|             if (m) { | ||||
|                 console.log(m); | ||||
|                 wspath = "ws"+(m[1]==="https"?"s":"")+"://"+m[2]+"comms"; | ||||
|             } | ||||
|         } else { | ||||
|             var path = location.hostname; | ||||
|             var port = location.port; | ||||
|             if (port.length !== 0) { | ||||
|                 path = path+":"+port; | ||||
|             } | ||||
|             path = path+document.location.pathname; | ||||
|             path = path+(path.slice(-1) == "/"?"":"/")+"comms"; | ||||
|             wspath = "ws"+(document.location.protocol=="https:"?"s":"")+"://"+path; | ||||
|         var path = location.hostname; | ||||
|         var port = location.port; | ||||
|         if (port.length !== 0) { | ||||
|             path = path+":"+port; | ||||
|         } | ||||
|         path = path+document.location.pathname; | ||||
|         path = path+(path.slice(-1) == "/"?"":"/")+"comms"; | ||||
|         path = "ws"+(document.location.protocol=="https:"?"s":"")+"://"+path; | ||||
| 
 | ||||
|         var auth_tokens = RED.settings.get("auth-tokens"); | ||||
|         pendingAuth = (auth_tokens!=null); | ||||
| @@ -58,7 +48,7 @@ RED.comms = (function() { | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         ws = new WebSocket(wspath); | ||||
|         ws = new WebSocket(path); | ||||
|         ws.onopen = function() { | ||||
|             reconnectAttempts = 0; | ||||
|             if (errornotification) { | ||||
| @@ -91,15 +91,12 @@ RED.history = (function() { | ||||
|             } else if (ev.t == "delete") { | ||||
|                 if (ev.workspaces) { | ||||
|                     for (i=0;i<ev.workspaces.length;i++) { | ||||
|                         RED.nodes.addWorkspace(ev.workspaces[i],ev.workspaces[i]._index); | ||||
|                         RED.workspaces.add(ev.workspaces[i],undefined,ev.workspaces[i]._index); | ||||
|                         delete ev.workspaces[i]._index; | ||||
|                         RED.nodes.addWorkspace(ev.workspaces[i]); | ||||
|                         RED.workspaces.add(ev.workspaces[i]); | ||||
|                     } | ||||
|                 } | ||||
|                 if (ev.subflows) { | ||||
|                     for (i=0;i<ev.subflows.length;i++) { | ||||
|                         RED.nodes.addSubflow(ev.subflows[i]); | ||||
|                     } | ||||
|                 if (ev.subflow && ev.subflow.subflow) { | ||||
|                     RED.nodes.addSubflow(ev.subflow.subflow); | ||||
|                 } | ||||
|                 if (ev.subflowInputs && ev.subflowInputs.length > 0) { | ||||
|                     subflow = RED.nodes.subflow(ev.subflowInputs[0].z); | ||||
| @@ -16,13 +16,10 @@ | ||||
| 
 | ||||
| RED.i18n = (function() { | ||||
| 
 | ||||
|     var apiRootUrl; | ||||
| 
 | ||||
|     return { | ||||
|         init: function(options, done) { | ||||
|             apiRootUrl = options.apiRootUrl||""; | ||||
|         init: function(done) { | ||||
|             i18n.init({ | ||||
|                 resGetPath: apiRootUrl+'locales/__ns__?lng=__lng__', | ||||
|                 resGetPath: 'locales/__ns__?lng=__lng__', | ||||
|                 dynamicLoad: false, | ||||
|                 load:'current', | ||||
|                 ns: { | ||||
| @@ -39,7 +36,7 @@ RED.i18n = (function() { | ||||
|             } | ||||
| 
 | ||||
|         }, | ||||
|         loadNodeCatalog: function(namespace,done) { | ||||
|         loadCatalog: function(namespace,done) { | ||||
|             var languageList = i18n.functions.toLanguages(i18n.detectLanguage()); | ||||
|             var toLoad = languageList.length; | ||||
|             languageList.forEach(function(lang) { | ||||
| @@ -48,7 +45,7 @@ RED.i18n = (function() { | ||||
|                         "Accept":"application/json" | ||||
|                     }, | ||||
|                     cache: false, | ||||
|                     url: apiRootUrl+'nodes/'+namespace+'/messages?lng='+lang, | ||||
|                     url: 'locales/'+namespace+'?lng='+lang, | ||||
|                     success: function(data) { | ||||
|                         i18n.addResourceBundle(lang,namespace,data); | ||||
|                         toLoad--; | ||||
| @@ -71,7 +68,7 @@ RED.i18n = (function() { | ||||
|                         "Accept":"application/json" | ||||
|                     }, | ||||
|                     cache: false, | ||||
|                     url: apiRootUrl+'nodes/messages?lng='+lang, | ||||
|                     url: 'locales/nodes?lng='+lang, | ||||
|                     success: function(data) { | ||||
|                         var namespaces = Object.keys(data); | ||||
|                         namespaces.forEach(function(ns) { | ||||
| @@ -10,23 +10,14 @@ | ||||
|         "ctrl-g i": "core:show-info-tab", | ||||
|         "ctrl-g d": "core:show-debug-tab", | ||||
|         "ctrl-g c": "core:show-config-tab", | ||||
|         "ctrl-g x": "core:show-context-tab", | ||||
|         "ctrl-e": "core:show-export-dialog", | ||||
|         "ctrl-i": "core:show-import-dialog", | ||||
|         "ctrl-space": "core:toggle-sidebar", | ||||
|         "ctrl-p": "core:toggle-palette", | ||||
|         "ctrl-,": "core:show-user-settings", | ||||
|         "ctrl-alt-r": "core:show-remote-diff", | ||||
|         "ctrl-alt-n": "core:new-project", | ||||
|         "ctrl-alt-o": "core:open-project", | ||||
|         "ctrl-g v": "core:show-version-control-tab", | ||||
|         "ctrl-shift-l": "core:show-event-log" | ||||
|     }, | ||||
|     "sidebar-node-config": { | ||||
|         "backspace": "core:delete-config-selection", | ||||
|         "delete": "core:delete-config-selection", | ||||
|         "ctrl-a": "core:select-all-config-nodes", | ||||
|         "ctrl-z": "core:undo" | ||||
|         "ctrl-g v": "core:show-version-control-tab" | ||||
|     }, | ||||
|     "workspace": { | ||||
|         "backspace": "core:delete-selection", | ||||
| @@ -13,45 +13,7 @@ | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  **/ | ||||
| var RED = (function() { | ||||
| 
 | ||||
|     function appendNodeConfig(nodeConfig,done) { | ||||
|         done = done || function(){}; | ||||
|         var m = /<!-- --- \[red-module:(\S+)\] --- -->/.exec(nodeConfig.trim()); | ||||
|         var moduleId; | ||||
|         if (m) { | ||||
|             moduleId = m[1]; | ||||
|         } else { | ||||
|             moduleId = "unknown"; | ||||
|         } | ||||
|         try { | ||||
|             var hasDeferred = false; | ||||
| 
 | ||||
|             var nodeConfigEls = $("<div>"+nodeConfig+"</div>"); | ||||
|             nodeConfigEls.find("script").each(function(i,el) { | ||||
|                 var srcUrl = $(el).attr('src'); | ||||
|                 if (srcUrl && !/^\s*(https?:|\/|\.)/.test(srcUrl)) { | ||||
|                     $(el).remove(); | ||||
|                     var newScript = document.createElement("script"); | ||||
|                     newScript.onload = function() { $("body").append(nodeConfigEls); done() } | ||||
|                     $('body').append(newScript); | ||||
|                     newScript.src = RED.settings.apiRootUrl+srcUrl; | ||||
|                     hasDeferred = true; | ||||
|                 } | ||||
|             }) | ||||
|             if (!hasDeferred) { | ||||
|                 $("body").append(nodeConfigEls); | ||||
|                 done(); | ||||
|             } | ||||
|         } catch(err) { | ||||
|             RED.notify(RED._("notification.errors.failedToAppendNode",{module:moduleId, error:err.toString()}),{ | ||||
|                 type: "error", | ||||
|                 timeout: 10000 | ||||
|             }); | ||||
|             console.log("["+moduleId+"] "+err.toString()); | ||||
|             done(); | ||||
|         } | ||||
|     } | ||||
| (function() { | ||||
| 
 | ||||
|     function loadNodeList() { | ||||
|         $.ajax({ | ||||
| @@ -93,41 +55,33 @@ var RED = (function() { | ||||
|             cache: false, | ||||
|             url: 'nodes', | ||||
|             success: function(data) { | ||||
|                 var configs = data.trim().split(/(?=<!-- --- \[red-module:\S+\] --- -->)/); | ||||
|                 var stepConfig = function() { | ||||
|                     if (configs.length === 0) { | ||||
|                         $("body").i18n(); | ||||
|                         $("#palette > .palette-spinner").hide(); | ||||
|                         $(".palette-scroll").removeClass("hide"); | ||||
|                         $("#palette-search").removeClass("hide"); | ||||
|                         loadFlows(function() { | ||||
|                             if (RED.settings.theme("projects.enabled",false)) { | ||||
|                                 RED.projects.refresh(function(activeProject) { | ||||
|                                     RED.sidebar.info.refresh() | ||||
|                                     if (!activeProject) { | ||||
|                                         // Projects enabled but no active project
 | ||||
|                                         RED.menu.setDisabled('menu-item-projects-open',true); | ||||
|                                         RED.menu.setDisabled('menu-item-projects-settings',true); | ||||
|                                         if (activeProject === false) { | ||||
|                                             // User previously decline the migration to projects.
 | ||||
|                                         } else { // null/undefined
 | ||||
|                                             RED.projects.showStartup(); | ||||
|                                         } | ||||
|                                     } | ||||
|                                     completeLoad(); | ||||
|                                 }); | ||||
|                             } else { | ||||
|                                 // Projects disabled by the user
 | ||||
|                                 RED.sidebar.info.refresh() | ||||
|                                 completeLoad(); | ||||
|                 $("body").append(data); | ||||
|                 $("body").i18n(); | ||||
|                 $("#palette > .palette-spinner").hide(); | ||||
|                 $(".palette-scroll").removeClass("hide"); | ||||
|                 $("#palette-search").removeClass("hide"); | ||||
|                 loadFlows(function() { | ||||
|                     if (RED.settings.theme("projects.enabled",false)) { | ||||
|                         RED.projects.refresh(function(activeProject) { | ||||
|                             RED.sidebar.info.refresh() | ||||
|                             if (!activeProject) { | ||||
|                                 // Projects enabled but no active project
 | ||||
|                                 RED.menu.setDisabled('menu-item-projects-open',true); | ||||
|                                 RED.menu.setDisabled('menu-item-projects-settings',true); | ||||
|                                 if (activeProject === false) { | ||||
|                                     // User previously decline the migration to projects.
 | ||||
|                                 } else { // null/undefined
 | ||||
|                                     RED.projects.showStartup(); | ||||
|                                 } | ||||
|                             } | ||||
|                             completeLoad(); | ||||
|                         }); | ||||
|                     } else { | ||||
|                         var config = configs.shift(); | ||||
|                         appendNodeConfig(config,stepConfig); | ||||
|                         // Projects disabled by the user
 | ||||
|                         RED.sidebar.info.refresh() | ||||
|                         completeLoad(); | ||||
|                     } | ||||
|                 } | ||||
|                 stepConfig(); | ||||
|                 }); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| @@ -176,13 +130,13 @@ var RED = (function() { | ||||
|                     loadFlows(function() { | ||||
|                         var project = RED.projects.getActiveProject(); | ||||
|                         var message = { | ||||
|                             "change-branch": RED._("notification.project.change-branch", {project: project.git.branches.local}), | ||||
|                             "merge-abort": RED._("notification.project.merge-abort"), | ||||
|                             "loaded": RED._("notification.project.loaded", {project: msg.project}), | ||||
|                             "updated": RED._("notification.project.updated", {project: msg.project}), | ||||
|                             "pull": RED._("notification.project.pull", {project: msg.project}), | ||||
|                             "revert": RED._("notification.project.revert", {project: msg.project}), | ||||
|                             "merge-complete": RED._("notification.project.merge-complete") | ||||
|                             "change-branch":"Change to local branch '"+project.git.branches.local+"'", | ||||
|                             "merge-abort":"Git merge aborted", | ||||
|                             "loaded":"Project '"+msg.project+"' loaded", | ||||
|                             "updated":"Project '"+msg.project+"' updated", | ||||
|                             "pull":"Project '"+msg.project+"' reloaded", | ||||
|                             "revert": "Project '"+msg.project+"' reloaded", | ||||
|                             "merge-complete":"Git merge completed" | ||||
|                         }[msg.action]; | ||||
|                         RED.notify("<p>"+message+"</p>"); | ||||
|                         RED.sidebar.info.refresh() | ||||
| @@ -201,32 +155,23 @@ var RED = (function() { | ||||
|                     id: notificationId | ||||
|                 } | ||||
|                 if (notificationId === "runtime-state") { | ||||
|                     if (msg.error === "safe-mode") { | ||||
|                         options.buttons = [ | ||||
|                             { | ||||
|                                 text: RED._("common.label.close"), | ||||
|                                 click: function() { | ||||
|                                     persistentNotifications[notificationId].hideNotification(); | ||||
|                                 } | ||||
|                             } | ||||
|                         ] | ||||
|                     } else if (msg.error === "missing-types") { | ||||
|                     if (msg.error === "missing-types") { | ||||
|                         text+="<ul><li>"+msg.types.join("</li><li>")+"</li></ul>"; | ||||
|                         if (!!RED.projects.getActiveProject()) { | ||||
|                             options.buttons = [ | ||||
|                                 { | ||||
|                                     text: RED._("notification.label.manage-project-dep"), | ||||
|                                     text: "Manage project dependencies", | ||||
|                                     click: function() { | ||||
|                                         persistentNotifications[notificationId].hideNotification(); | ||||
|                                         RED.projects.settings.show('deps'); | ||||
|                                     } | ||||
|                                 } | ||||
|                             ] | ||||
|                             // } else if (RED.settings.theme('palette.editable') !== false) {
 | ||||
|                         // } else if (RED.settings.theme('palette.editable') !== false) {
 | ||||
|                         } else { | ||||
|                             options.buttons = [ | ||||
|                                 { | ||||
|                                     text: RED._("common.label.close"), | ||||
|                                     text: "Close", | ||||
|                                     click: function() { | ||||
|                                         persistentNotifications[notificationId].hideNotification(); | ||||
|                                     } | ||||
| @@ -234,25 +179,13 @@ var RED = (function() { | ||||
|                             ] | ||||
|                         } | ||||
|                     } else if (msg.error === "credentials_load_failed") { | ||||
|                         if (RED.settings.theme("projects.enabled",false)) { | ||||
|                             // projects enabled
 | ||||
|                             if (RED.user.hasPermission("projects.write")) { | ||||
|                                 options.buttons = [ | ||||
|                                     { | ||||
|                                         text: "Setup credentials", | ||||
|                                         click: function() { | ||||
|                                             persistentNotifications[notificationId].hideNotification(); | ||||
|                                             RED.projects.showCredentialsPrompt(); | ||||
|                                         } | ||||
|                                     } | ||||
|                                 ] | ||||
|                             } | ||||
|                         } else { | ||||
|                         if (RED.user.hasPermission("projects.write")) { | ||||
|                             options.buttons = [ | ||||
|                                 { | ||||
|                                     text: "Close", | ||||
|                                     text: "Setup credentials", | ||||
|                                     click: function() { | ||||
|                                         persistentNotifications[notificationId].hideNotification(); | ||||
|                                         RED.projects.showCredentialsPrompt(); | ||||
|                                     } | ||||
|                                 } | ||||
|                             ] | ||||
| @@ -269,18 +202,6 @@ var RED = (function() { | ||||
|                                 } | ||||
|                             ] | ||||
|                         } | ||||
|                     } else if (msg.error === "missing_package_file") { | ||||
|                         if (RED.user.hasPermission("projects.write")) { | ||||
|                             options.buttons = [ | ||||
|                                 { | ||||
|                                     text: "Create default package file", | ||||
|                                     click: function() { | ||||
|                                         persistentNotifications[notificationId].hideNotification(); | ||||
|                                         RED.projects.createDefaultPackageFile(); | ||||
|                                     } | ||||
|                                 } | ||||
|                             ] | ||||
|                         } | ||||
|                     } else if (msg.error === "project_empty") { | ||||
|                         if (RED.user.hasPermission("projects.write")) { | ||||
|                             options.buttons = [ | ||||
| @@ -349,9 +270,9 @@ var RED = (function() { | ||||
|                     var id = m.id; | ||||
|                     RED.nodes.addNodeSet(m); | ||||
|                     addedTypes = addedTypes.concat(m.types); | ||||
|                     RED.i18n.loadNodeCatalog(id, function() { | ||||
|                     RED.i18n.loadCatalog(id, function() { | ||||
|                         $.get('nodes/'+id, function(data) { | ||||
|                             appendNodeConfig(data); | ||||
|                             $("body").append(data); | ||||
|                         }); | ||||
|                     }); | ||||
|                 }); | ||||
| @@ -379,7 +300,7 @@ var RED = (function() { | ||||
|                         RED.notify(RED._("palette.event.nodeEnabled", {count:msg.types.length})+typeList,"success"); | ||||
|                     } else { | ||||
|                         $.get('nodes/'+msg.id, function(data) { | ||||
|                             appendNodeConfig(data); | ||||
|                             $("body").append(data); | ||||
|                             typeList = "<ul><li>"+msg.types.join("</li><li>")+"</li></ul>"; | ||||
|                             RED.notify(RED._("palette.event.nodeAdded", {count:msg.types.length})+typeList,"success"); | ||||
|                         }); | ||||
| @@ -398,17 +319,13 @@ var RED = (function() { | ||||
|             // Refresh flow library to ensure any examples are updated
 | ||||
|             RED.library.loadFlowLibrary(); | ||||
|         }); | ||||
|         RED.comms.subscribe("event-log/#", function(topic,payload) { | ||||
|             var id = topic.substring(9); | ||||
|             RED.eventLog.log(id,payload); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     function showAbout() { | ||||
|         $.get('red/about', function(data) { | ||||
|             var aboutHeader = '<div style="text-align:center;">'+ | ||||
|             '<img width="50px" src="red/images/node-red-icon.svg" />'+ | ||||
|             '</div>'; | ||||
|                                 '<img width="50px" src="red/images/node-red-icon.svg" />'+ | ||||
|                               '</div>'; | ||||
| 
 | ||||
|             RED.sidebar.info.set(aboutHeader+marked(data)); | ||||
|             RED.sidebar.info.show(); | ||||
| @@ -418,10 +335,10 @@ var RED = (function() { | ||||
|     function loadEditor() { | ||||
|         var menuOptions = []; | ||||
|         if (RED.settings.theme("projects.enabled",false)) { | ||||
|             menuOptions.push({id:"menu-item-projects-menu",label:RED._("menu.label.projects"),options:[ | ||||
|                 {id:"menu-item-projects-new",label:RED._("menu.label.projects-new"),disabled:false,onselect:"core:new-project"}, | ||||
|                 {id:"menu-item-projects-open",label:RED._("menu.label.projects-open"),disabled:false,onselect:"core:open-project"}, | ||||
|                 {id:"menu-item-projects-settings",label:RED._("menu.label.projects-settings"),disabled:false,onselect:"core:show-project-settings"} | ||||
|             menuOptions.push({id:"menu-item-projects-menu",label:"Projects",options:[ | ||||
|                 {id:"menu-item-projects-new",label:"New",disabled:false,onselect:"core:new-project"}, | ||||
|                 {id:"menu-item-projects-open",label:"Open",disabled:false,onselect:"core:open-project"}, | ||||
|                 {id:"menu-item-projects-settings",label:"Project Settings",disabled:false,onselect:"core:show-project-settings"} | ||||
|             ]}); | ||||
|         } | ||||
| 
 | ||||
| @@ -438,9 +355,7 @@ var RED = (function() { | ||||
|             //     {id:"menu-item-bidi-auto",toggle:"text-direction",label:RED._("menu.label.view.auto"), onselect:function(s) { if(s){RED.text.bidi.setTextDirection("auto")}}}
 | ||||
|             // ]},
 | ||||
|             // null,
 | ||||
|             {id:"menu-item-palette",label:RED._("menu.label.palette.show"),toggle:true,onselect:"core:toggle-palette", selected: true}, | ||||
|             {id:"menu-item-sidebar",label:RED._("menu.label.sidebar.show"),toggle:true,onselect:"core:toggle-sidebar", selected: true}, | ||||
|             {id:"menu-item-event-log",label:RED._("eventLog.title"),onselect:"core:show-event-log"}, | ||||
|             null | ||||
|         ]}); | ||||
|         menuOptions.push(null); | ||||
| @@ -448,8 +363,8 @@ var RED = (function() { | ||||
|             {id:"menu-item-import-clipboard",label:RED._("menu.label.clipboard"),onselect:"core:show-import-dialog"}, | ||||
|             {id:"menu-item-import-library",label:RED._("menu.label.library"),options:[]} | ||||
|         ]}); | ||||
|         menuOptions.push({id:"menu-item-export",label:RED._("menu.label.export"),options:[ | ||||
|             {id:"menu-item-export-clipboard",label:RED._("menu.label.clipboard"),onselect:"core:show-export-dialog"}, | ||||
|         menuOptions.push({id:"menu-item-export",label:RED._("menu.label.export"),disabled:true,options:[ | ||||
|             {id:"menu-item-export-clipboard",label:RED._("menu.label.clipboard"),disabled:true,onselect:"core:show-export-dialog"}, | ||||
|             {id:"menu-item-export-library",label:RED._("menu.label.library"),disabled:true,onselect:"core:library-export"} | ||||
|         ]}); | ||||
|         menuOptions.push(null); | ||||
| @@ -488,7 +403,6 @@ var RED = (function() { | ||||
|         RED.library.init(); | ||||
|         RED.keyboard.init(); | ||||
|         RED.palette.init(); | ||||
|         RED.eventLog.init(); | ||||
|         if (RED.settings.theme('palette.editable') !== false) { | ||||
|             RED.palette.editor.init(); | ||||
|         } else { | ||||
| @@ -525,25 +439,16 @@ var RED = (function() { | ||||
|         loadNodeList(); | ||||
|     } | ||||
| 
 | ||||
|     var initialised = false; | ||||
|     $(function() { | ||||
| 
 | ||||
|     function init(options) { | ||||
|         if (initialised) { | ||||
|             throw new Error("RED already initialised"); | ||||
|         if ((window.location.hostname !== "localhost") && (window.location.hostname !== "127.0.0.1")) { | ||||
|             document.title = document.title+" : "+window.location.hostname; | ||||
|         } | ||||
|         initialised = true; | ||||
| 
 | ||||
|         ace.require("ace/ext/language_tools"); | ||||
|         options = options || {}; | ||||
|         options.apiRootUrl = options.apiRootUrl || ""; | ||||
|         if (options.apiRootUrl && !/\/$/.test(options.apiRootUrl)) { | ||||
|             options.apiRootUrl = options.apiRootUrl+"/"; | ||||
|         } | ||||
|         RED.i18n.init(options, function() { | ||||
|             RED.settings.init(options, loadEditor); | ||||
|         }) | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|         init: init | ||||
|     } | ||||
|         RED.i18n.init(function() { | ||||
|             RED.settings.init(loadEditor); | ||||
|         }) | ||||
|     }); | ||||
| })(); | ||||
| @@ -133,7 +133,7 @@ RED.nodes = (function() { | ||||
|             registerNodeType: function(nt,def) { | ||||
|                 nodeDefinitions[nt] = def; | ||||
|                 def.type = nt; | ||||
|                 if (nt.substring(0,8) != "subflow:") { | ||||
|                 if (def.category != "subflows") { | ||||
|                     def.set = nodeSets[typeToId[nt]]; | ||||
|                     nodeSets[typeToId[nt]].added = true; | ||||
|                     nodeSets[typeToId[nt]].enabled = true; | ||||
| @@ -174,7 +174,6 @@ RED.nodes = (function() { | ||||
|             }, | ||||
|             setIconSets: function(sets) { | ||||
|                 iconSets = sets; | ||||
|                 iconSets["font-awesome"] = RED.nodes.fontAwesome.getIconList(); | ||||
|             }, | ||||
|             getIconSets: function() { | ||||
|                 return iconSets; | ||||
| @@ -298,14 +297,10 @@ RED.nodes = (function() { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     function addWorkspace(ws,targetIndex) { | ||||
|     function addWorkspace(ws) { | ||||
|         workspaces[ws.id] = ws; | ||||
|         ws._def = RED.nodes.getType('tab'); | ||||
|         if (targetIndex === undefined) { | ||||
|             workspacesOrder.push(ws.id); | ||||
|         } else { | ||||
|             workspacesOrder.splice(targetIndex,0,ws.id); | ||||
|         } | ||||
|         workspacesOrder.push(ws.id); | ||||
|     } | ||||
|     function getWorkspace(id) { | ||||
|         return workspaces[id]; | ||||
| @@ -359,8 +354,9 @@ RED.nodes = (function() { | ||||
|         subflows[sf.id] = sf; | ||||
|         RED.nodes.registerType("subflow:"+sf.id, { | ||||
|             defaults:{name:{value:""}}, | ||||
|             info: sf.info, | ||||
|             icon: function() { return sf.icon||"subflow.png" }, | ||||
|             category: sf.category || "subflows", | ||||
|             category: "subflows", | ||||
|             inputs: sf.in.length, | ||||
|             outputs: sf.out.length, | ||||
|             color: "#da9", | ||||
| @@ -495,9 +491,7 @@ RED.nodes = (function() { | ||||
|             for (var j=0;j<wires.length;j++) { | ||||
|                 var w = wires[j]; | ||||
|                 if (w.target.type != "subflow") { | ||||
|                     if (w.sourcePort < node.wires.length) { | ||||
|                         node.wires[w.sourcePort].push(w.target.id); | ||||
|                     } | ||||
|                     node.wires[w.sourcePort].push(w.target.id); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
| @@ -513,15 +507,6 @@ RED.nodes = (function() { | ||||
|                     node.icon = n.icon; | ||||
|                 } | ||||
|             } | ||||
|             if ((!n._def.defaults || !n._def.defaults.hasOwnProperty("l")) && n.hasOwnProperty('l')) { | ||||
|                 var isLink = /^link (in|out)$/.test(node.type); | ||||
|                 if (isLink == n.l) { | ||||
|                     node.l = n.l; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         if (n.info) { | ||||
|             node.info = n.info; | ||||
|         } | ||||
|         return node; | ||||
|     } | ||||
| @@ -532,7 +517,6 @@ RED.nodes = (function() { | ||||
|         node.type = n.type; | ||||
|         node.name = n.name; | ||||
|         node.info = n.info; | ||||
|         node.category = n.category; | ||||
|         node.in = []; | ||||
|         node.out = []; | ||||
| 
 | ||||
| @@ -653,39 +637,42 @@ RED.nodes = (function() { | ||||
|     } | ||||
| 
 | ||||
|     function checkForMatchingSubflow(subflow,subflowNodes) { | ||||
|         subflowNodes = subflowNodes || []; | ||||
|         var i; | ||||
|         var match = null; | ||||
|         RED.nodes.eachSubflow(function(sf) { | ||||
|             if (sf.name != subflow.name || | ||||
|                 sf.info != subflow.info || | ||||
|                 sf.in.length != subflow.in.length || | ||||
|                 sf.out.length != subflow.out.length) { | ||||
|         try { | ||||
|             RED.nodes.eachSubflow(function(sf) { | ||||
|                 if (sf.name != subflow.name || | ||||
|                     sf.info != subflow.info || | ||||
|                     sf.in.length != subflow.in.length || | ||||
|                     sf.out.length != subflow.out.length) { | ||||
|                         return; | ||||
|                 } | ||||
|                 var sfNodes = RED.nodes.filterNodes({z:sf.id}); | ||||
|                 if (sfNodes.length != subflowNodes.length) { | ||||
|                     return; | ||||
|             } | ||||
|             var sfNodes = RED.nodes.filterNodes({z:sf.id}); | ||||
|             if (sfNodes.length != subflowNodes.length) { | ||||
|                 return; | ||||
|             } | ||||
|                 } | ||||
| 
 | ||||
|             var subflowNodeSet = [subflow].concat(subflowNodes); | ||||
|             var sfNodeSet = [sf].concat(sfNodes); | ||||
|                 var subflowNodeSet = [subflow].concat(subflowNodes); | ||||
|                 var sfNodeSet = [sf].concat(sfNodes); | ||||
| 
 | ||||
|             var exportableSubflowNodes = JSON.stringify(subflowNodeSet); | ||||
|             var exportableSFNodes = JSON.stringify(createExportableNodeSet(sfNodeSet)); | ||||
|             var nodeMap = {}; | ||||
|             for (i=0;i<sfNodes.length;i++) { | ||||
|                 exportableSubflowNodes = exportableSubflowNodes.replace(new RegExp("\""+subflowNodes[i].id+"\"","g"),'"'+sfNodes[i].id+'"'); | ||||
|             } | ||||
|             exportableSubflowNodes = exportableSubflowNodes.replace(new RegExp("\""+subflow.id+"\"","g"),'"'+sf.id+'"'); | ||||
|                 var exportableSubflowNodes = JSON.stringify(subflowNodeSet); | ||||
|                 var exportableSFNodes = JSON.stringify(createExportableNodeSet(sfNodeSet)); | ||||
|                 var nodeMap = {}; | ||||
|                 for (i=0;i<sfNodes.length;i++) { | ||||
|                     exportableSubflowNodes = exportableSubflowNodes.replace(new RegExp("\""+subflowNodes[i].id+"\"","g"),'"'+sfNodes[i].id+'"'); | ||||
|                 } | ||||
|                 exportableSubflowNodes = exportableSubflowNodes.replace(new RegExp("\""+subflow.id+"\"","g"),'"'+sf.id+'"'); | ||||
| 
 | ||||
|             if (exportableSubflowNodes !== exportableSFNodes) { | ||||
|                 return; | ||||
|             } | ||||
|                 if (exportableSubflowNodes !== exportableSFNodes) { | ||||
|                     return; | ||||
|                 } | ||||
| 
 | ||||
|             match = sf; | ||||
|             return false; | ||||
|         }); | ||||
|                 match = sf; | ||||
|                 throw new Error(); | ||||
|             }); | ||||
|         } catch(err) { | ||||
|             console.log(err.stack); | ||||
|         } | ||||
|         return match; | ||||
|     } | ||||
|     function compareNodes(nodeA,nodeB,idMustMatch) { | ||||
| @@ -765,7 +752,8 @@ RED.nodes = (function() { | ||||
|         } | ||||
|         if (!isInitialLoad && unknownTypes.length > 0) { | ||||
|             var typeList = "<ul><li>"+unknownTypes.join("</li><li>")+"</li></ul>"; | ||||
|             RED.notify("<p>"+RED._("clipboard.importUnrecognised",{count:unknownTypes.length})+"</p>"+typeList,"error",false,10000); | ||||
|             var type = "type"+(unknownTypes.length > 1?"s":""); | ||||
|             RED.notify("<strong>"+RED._("clipboard.importUnrecognised",{count:unknownTypes.length})+"</strong>"+typeList,"error",false,10000); | ||||
|         } | ||||
| 
 | ||||
|         var activeWorkspace = RED.workspaces.active(); | ||||
| @@ -913,14 +901,7 @@ RED.nodes = (function() { | ||||
|                 } | ||||
| 
 | ||||
|                 if (!existingConfigNode || existingConfigNode._def.exclusive) { //} || !compareNodes(existingConfigNode,n,true) || existingConfigNode.z !== n.z) {
 | ||||
|                     configNode = { | ||||
|                         id:n.id, | ||||
|                         z:n.z, | ||||
|                         type:n.type, | ||||
|                         info: n.info, | ||||
|                         users:[], | ||||
|                         _config:{} | ||||
|                     }; | ||||
|                     configNode = {id:n.id, z:n.z, type:n.type, users:[], _config:{}}; | ||||
|                     for (d in def.defaults) { | ||||
|                         if (def.defaults.hasOwnProperty(d)) { | ||||
|                             configNode[d] = n[d]; | ||||
| @@ -955,21 +936,17 @@ RED.nodes = (function() { | ||||
|                 def = registry.getNodeType(n.type); | ||||
|                 if (!def || def.category != "config") { | ||||
|                     var node = { | ||||
|                         x:parseFloat(n.x || 0), | ||||
|                         y:parseFloat(n.y || 0), | ||||
|                         x:n.x, | ||||
|                         y:n.y, | ||||
|                         z:n.z, | ||||
|                         type:0, | ||||
|                         wires:n.wires||[], | ||||
|                         wires:n.wires, | ||||
|                         inputLabels: n.inputLabels, | ||||
|                         outputLabels: n.outputLabels, | ||||
|                         icon: n.icon, | ||||
|                         info: n.info, | ||||
|                         changed:false, | ||||
|                         _config:{} | ||||
|                     }; | ||||
|                     if (n.hasOwnProperty('l')) { | ||||
|                         node.l = n.l; | ||||
|                     } | ||||
|                     if (createNewIds) { | ||||
|                         if (subflow_blacklist[n.z]) { | ||||
|                             continue; | ||||
| @@ -1035,13 +1012,6 @@ RED.nodes = (function() { | ||||
|                                     set: registry.getNodeSet("node-red/unknown") | ||||
|                                 }; | ||||
|                                 node.users = []; | ||||
|                                 // This is a config node, so delete the default
 | ||||
|                                 // non-config node properties
 | ||||
|                                 delete node.x; | ||||
|                                 delete node.y; | ||||
|                                 delete node.wires; | ||||
|                                 delete node.inputLabels; | ||||
|                                 delete node.outputLabels; | ||||
|                             } | ||||
|                             var orig = {}; | ||||
|                             for (var p in n) { | ||||
| @@ -1054,31 +1024,10 @@ RED.nodes = (function() { | ||||
|                             node.type = "unknown"; | ||||
|                         } | ||||
|                         if (node._def.category != "config") { | ||||
|                             if (n.hasOwnProperty('inputs')) { | ||||
|                                 node.inputs = n.inputs; | ||||
|                                 node._config.inputs = JSON.stringify(n.inputs); | ||||
|                             } else { | ||||
|                                 node.inputs = node._def.inputs; | ||||
|                             } | ||||
|                             if (n.hasOwnProperty('outputs')) { | ||||
|                                 node.outputs = n.outputs; | ||||
|                                 node._config.outputs = JSON.stringify(n.outputs); | ||||
|                             } else { | ||||
|                                 node.outputs = node._def.outputs; | ||||
|                             } | ||||
|                             if (node.hasOwnProperty('wires') && node.wires.length > node.outputs) { | ||||
|                                 if (!node._def.defaults.hasOwnProperty("outputs") || !isNaN(parseInt(n.outputs))) { | ||||
|                                     // If 'wires' is longer than outputs, clip wires
 | ||||
|                                     console.log("Warning: node.wires longer than node.outputs - trimming wires:",node.id," wires:",node.wires.length," outputs:",node.outputs); | ||||
|                                     node.wires = node.wires.slice(0,node.outputs); | ||||
|                                 } else { | ||||
|                                     // The node declares outputs in its defaults, but has not got a valid value
 | ||||
|                                     // Defer to the length of the wires array
 | ||||
|                                     node.outputs = node.wires.length; | ||||
|                                 } | ||||
|                             } | ||||
|                             node.inputs = n.inputs||node._def.inputs; | ||||
|                             node.outputs = n.outputs||node._def.outputs; | ||||
|                             for (d in node._def.defaults) { | ||||
|                                 if (node._def.defaults.hasOwnProperty(d) && d !== 'inputs' && d !== 'outputs') { | ||||
|                                 if (node._def.defaults.hasOwnProperty(d)) { | ||||
|                                     node[d] = n[d]; | ||||
|                                     node._config[d] = JSON.stringify(n[d]); | ||||
|                                 } | ||||
| @@ -1098,9 +1047,7 @@ RED.nodes = (function() { | ||||
|                     addNode(node); | ||||
|                     RED.editor.validateNode(node); | ||||
|                     node_map[n.id] = node; | ||||
|                     // If an 'unknown' config node, it will not have been caught by the
 | ||||
|                     // proper config node handling, so needs adding to new_nodes here
 | ||||
|                     if (node.type === "unknown" || node._def.category !== "config") { | ||||
|                     if (node._def.category != "config") { | ||||
|                         new_nodes.push(node); | ||||
|                     } | ||||
|                 } | ||||
| @@ -1382,41 +1329,31 @@ RED.nodes = (function() { | ||||
| 
 | ||||
|         eachNode: function(cb) { | ||||
|             for (var n=0;n<nodes.length;n++) { | ||||
|                 if (cb(nodes[n]) === false) { | ||||
|                     break; | ||||
|                 } | ||||
|                 cb(nodes[n]); | ||||
|             } | ||||
|         }, | ||||
|         eachLink: function(cb) { | ||||
|             for (var l=0;l<links.length;l++) { | ||||
|                 if (cb(links[l]) === false) { | ||||
|                     break; | ||||
|                 } | ||||
|                 cb(links[l]); | ||||
|             } | ||||
|         }, | ||||
|         eachConfig: function(cb) { | ||||
|             for (var id in configNodes) { | ||||
|                 if (configNodes.hasOwnProperty(id)) { | ||||
|                     if (cb(configNodes[id]) === false) { | ||||
|                         break; | ||||
|                     } | ||||
|                     cb(configNodes[id]); | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         eachSubflow: function(cb) { | ||||
|             for (var id in subflows) { | ||||
|                 if (subflows.hasOwnProperty(id)) { | ||||
|                     if (cb(subflows[id]) === false) { | ||||
|                         break; | ||||
|                     } | ||||
|                     cb(subflows[id]); | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         eachWorkspace: function(cb) { | ||||
|             for (var i=0;i<workspacesOrder.length;i++) { | ||||
|                 if (cb(workspaces[workspacesOrder[i]]) === false) { | ||||
|                     break; | ||||
|                 } | ||||
|                 cb(workspaces[workspacesOrder[i]]); | ||||
|             } | ||||
|         }, | ||||
| 
 | ||||
| @@ -13,5 +13,4 @@ | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  **/ | ||||
| 
 | ||||
| module.exports = false | ||||
| var RED = {}; | ||||
| @@ -89,22 +89,18 @@ RED.settings = (function () { | ||||
|         userSettings = data; | ||||
|     } | ||||
| 
 | ||||
|     var init = function (options, done) { | ||||
|     var init = function (done) { | ||||
|         var accessTokenMatch = /[?&]access_token=(.*?)(?:$|&)/.exec(window.location.search); | ||||
|         if (accessTokenMatch) { | ||||
|             var accessToken = accessTokenMatch[1]; | ||||
|             RED.settings.set("auth-tokens",{access_token: accessToken}); | ||||
|             window.location.search = ""; | ||||
|         } | ||||
|         RED.settings.apiRootUrl = options.apiRootUrl; | ||||
| 
 | ||||
|         $.ajaxSetup({ | ||||
|             beforeSend: function(jqXHR,settings) { | ||||
|                 // Only attach auth header for requests to relative paths
 | ||||
|                 if (!/^\s*(https?:|\/|\.)/.test(settings.url)) { | ||||
|                     if (options.apiRootUrl) { | ||||
|                         settings.url = options.apiRootUrl+settings.url; | ||||
|                     } | ||||
|                     var auth_tokens = RED.settings.get("auth-tokens"); | ||||
|                     if (auth_tokens) { | ||||
|                         jqXHR.setRequestHeader("Authorization","Bearer "+auth_tokens.access_token); | ||||
| @@ -12,9 +12,9 @@ RED.actions = (function() { | ||||
|     function getAction(name) { | ||||
|         return actions[name]; | ||||
|     } | ||||
|     function invokeAction(name,args) { | ||||
|     function invokeAction(name) { | ||||
|         if (actions.hasOwnProperty(name)) { | ||||
|             actions[name](args); | ||||
|             actions[name](); | ||||
|         } | ||||
|     } | ||||
|     function listActions() { | ||||
| @@ -22,8 +22,6 @@ RED.clipboard = (function() { | ||||
|     var exportNodesDialog; | ||||
|     var importNodesDialog; | ||||
|     var disabled = false; | ||||
|     var popover; | ||||
|     var currentPopoverError; | ||||
| 
 | ||||
|     function setupDialogs() { | ||||
|         dialog = $('<div id="clipboard-dialog" class="hide node-red-dialog"><form class="dialog-form form-horizontal"></form></div>') | ||||
| @@ -49,21 +47,6 @@ RED.clipboard = (function() { | ||||
|                             $( this ).dialog( "close" ); | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         id: "clipboard-dialog-download", | ||||
|                         class: "primary", | ||||
|                         text: RED._("clipboard.download"), | ||||
|                         click: function() { | ||||
|                             var element = document.createElement('a'); | ||||
|                             element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent($("#clipboard-export").val())); | ||||
|                             element.setAttribute('download', "flows.json"); | ||||
|                             element.style.display = 'none'; | ||||
|                             document.body.appendChild(element); | ||||
|                             element.click(); | ||||
|                             document.body.removeChild(element); | ||||
|                             $( this ).dialog( "close" ); | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         id: "clipboard-dialog-copy", | ||||
|                         class: "primary", | ||||
| @@ -90,10 +73,6 @@ RED.clipboard = (function() { | ||||
|                     $(this).parent().find(".ui-dialog-titlebar-close").hide(); | ||||
|                 }, | ||||
|                 close: function(e) { | ||||
|                     if (popover) { | ||||
|                         popover.close(true); | ||||
|                         currentPopoverError = null; | ||||
|                     } | ||||
|                 } | ||||
|             }); | ||||
| 
 | ||||
| @@ -107,7 +86,7 @@ RED.clipboard = (function() { | ||||
|                     '<a id="export-range-flow" class="editor-button toggle" href="#" data-i18n="clipboard.export.current"></a>'+ | ||||
|                     '<a id="export-range-full" class="editor-button toggle" href="#" data-i18n="clipboard.export.all"></a>'+ | ||||
|                 '</span>'+ | ||||
|             '</div>'+ | ||||
|                 '</div>'+ | ||||
|             '<div class="form-row">'+ | ||||
|                 '<textarea readonly style="resize: none; width: 100%; border-radius: 4px;font-family: monospace; font-size: 12px; background:#f3f3f3; padding-left: 0.5em; box-sizing:border-box;" id="clipboard-export" rows="5"></textarea>'+ | ||||
|             '</div>'+ | ||||
| @@ -118,13 +97,10 @@ RED.clipboard = (function() { | ||||
|                 '</span>'+ | ||||
|             '</div>'; | ||||
| 
 | ||||
|         importNodesDialog = | ||||
|             '<div class="form-row"><span data-i18n="clipboard.pasteNodes"></span>'+ | ||||
|                 ' <a class="editor-button" id="import-file-upload-btn"><i class="fa fa-upload"></i> <span data-i18n="clipboard.selectFile"></span></a>'+ | ||||
|                 '<input type="file" id="import-file-upload" accept=".json" style="display:none">'+ | ||||
|             '</div>'+ | ||||
|             '<div class="form-row">'+ | ||||
|                 '<textarea style="resize: none; width: 100%; border-radius: 0px;font-family: monospace; font-size: 12px; background:#eee; padding-left: 0.5em; box-sizing:border-box;" id="clipboard-import" rows="5"></textarea>'+ | ||||
|         importNodesDialog = '<div class="form-row">'+ | ||||
|             '<textarea style="resize: none; width: 100%; border-radius: 0px;font-family: monospace; font-size: 12px; background:#eee; padding-left: 0.5em; box-sizing:border-box;" id="clipboard-import" rows="5" placeholder="'+ | ||||
|             RED._("clipboard.pasteNodes")+ | ||||
|             '"></textarea>'+ | ||||
|             '</div>'+ | ||||
|             '<div class="form-row">'+ | ||||
|             '<label style="width:auto;margin-right: 10px;" data-i18n="clipboard.import.import"></label>'+ | ||||
| @@ -135,98 +111,21 @@ RED.clipboard = (function() { | ||||
|             '</div>'; | ||||
|     } | ||||
| 
 | ||||
|     var validateImportTimeout; | ||||
| 
 | ||||
|     function validateImport() { | ||||
|         if (validateImportTimeout) { | ||||
|             clearTimeout(validateImportTimeout); | ||||
|         var importInput = $("#clipboard-import"); | ||||
|         var v = importInput.val(); | ||||
|         v = v.substring(v.indexOf('['),v.lastIndexOf(']')+1); | ||||
|         try { | ||||
|             JSON.parse(v); | ||||
|             importInput.removeClass("input-error"); | ||||
|             importInput.val(v); | ||||
|             $("#clipboard-dialog-ok").button("enable"); | ||||
|         } catch(err) { | ||||
|             if (v !== "") { | ||||
|                 importInput.addClass("input-error"); | ||||
|             } | ||||
|             $("#clipboard-dialog-ok").button("disable"); | ||||
|         } | ||||
|         validateImportTimeout = setTimeout(function() { | ||||
|             var importInput = $("#clipboard-import"); | ||||
|             var v = importInput.val().trim(); | ||||
|             if (v === "") { | ||||
|                 popover.close(true); | ||||
|                 currentPopoverError = null; | ||||
|                 importInput.removeClass("input-error"); | ||||
|                 $("#clipboard-dialog-ok").button("disable"); | ||||
|                 return; | ||||
|             } | ||||
|             try { | ||||
|                 if (!/^\[[\s\S]*\]$/m.test(v)) { | ||||
|                     throw new Error(RED._("clipboard.import.errors.notArray")); | ||||
|                 } | ||||
|                 var res = JSON.parse(v); | ||||
|                 for (var i=0;i<res.length;i++) { | ||||
|                     if (typeof res[i] !== "object") { | ||||
|                         throw new Error(RED._("clipboard.import.errors.itemNotObject",{index:i})); | ||||
|                     } | ||||
|                     if (!res[i].hasOwnProperty('id')) { | ||||
|                         throw new Error(RED._("clipboard.import.errors.missingId",{index:i})); | ||||
|                     } | ||||
|                     if (!res[i].hasOwnProperty('type')) { | ||||
|                         throw new Error(RED._("clipboard.import.errors.missingType",{index:i})); | ||||
|                     } | ||||
|                 } | ||||
|                 currentPopoverError = null; | ||||
|                 popover.close(true); | ||||
|                 importInput.removeClass("input-error"); | ||||
|                 importInput.val(v); | ||||
|                 $("#clipboard-dialog-ok").button("enable"); | ||||
|             } catch(err) { | ||||
|                 if (v !== "") { | ||||
|                     importInput.addClass("input-error"); | ||||
|                     var errString = err.toString(); | ||||
|                     if (errString !== currentPopoverError) { | ||||
|                         // Display the error as-is.
 | ||||
|                         // Error messages are only in English. Each browser has its
 | ||||
|                         // own set of messages with very little consistency.
 | ||||
|                         // To provide translated messages this code will either need to:
 | ||||
|                         // - reduce everything down to 'unexpected token at position x'
 | ||||
|                         //   which is the least useful, but most consistent message
 | ||||
|                         // - use a custom/library parser that gives consistent messages
 | ||||
|                         //   which can be translated.
 | ||||
|                         var message = $('<div class="clipboard-import-error"></div>').text(errString); | ||||
|                         var errorPos; | ||||
|                         // Chrome error messages
 | ||||
|                         var m = /at position (\d+)/i.exec(errString); | ||||
|                         if (m) { | ||||
|                             errorPos = parseInt(m[1]); | ||||
|                         } else { | ||||
|                             // Firefox error messages
 | ||||
|                             m = /at line (\d+) column (\d+)/i.exec(errString); | ||||
|                             if (m) { | ||||
|                                 var line = parseInt(m[1])-1; | ||||
|                                 var col = parseInt(m[2])-1; | ||||
|                                 var lines = v.split("\n"); | ||||
|                                 errorPos = 0; | ||||
|                                 for (var i=0;i<line;i++) { | ||||
|                                     errorPos += lines[i].length+1; | ||||
|                                 } | ||||
|                                 errorPos += col; | ||||
|                             } else { | ||||
|                                 // Safari doesn't provide any position information
 | ||||
|                                 // IE: tbd
 | ||||
|                             } | ||||
|                         } | ||||
| 
 | ||||
|                         if (errorPos !== undefined) { | ||||
|                             v = v.replace(/\n/g,"↵"); | ||||
|                             var index = parseInt(m[1]); | ||||
|                             var parseError = $('<div>').appendTo(message); | ||||
|                             var code = $('<pre>').appendTo(parseError); | ||||
|                             $('<span>').text(v.substring(errorPos-12,errorPos)).appendTo(code) | ||||
|                             $('<span class="error">').text(v.charAt(errorPos)).appendTo(code); | ||||
|                             $('<span>').text(v.substring(errorPos+1,errorPos+12)).appendTo(code); | ||||
|                         } | ||||
|                         popover.close(true).setContent(message).open(); | ||||
|                         currentPopoverError = errString; | ||||
|                     } | ||||
|                 } else { | ||||
|                     currentPopoverError = null; | ||||
|                 } | ||||
|                 $("#clipboard-dialog-ok").button("disable"); | ||||
|             } | ||||
|         },100); | ||||
|     } | ||||
| 
 | ||||
|     function importNodes() { | ||||
| @@ -241,7 +140,6 @@ RED.clipboard = (function() { | ||||
|         $("#clipboard-dialog-cancel").show(); | ||||
|         $("#clipboard-dialog-close").hide(); | ||||
|         $("#clipboard-dialog-copy").hide(); | ||||
|         $("#clipboard-dialog-download").hide(); | ||||
|         $("#clipboard-dialog-ok").button("disable"); | ||||
|         $("#clipboard-import").keyup(validateImport); | ||||
|         $("#clipboard-import").on('paste',function() { setTimeout(validateImport,10)}); | ||||
| @@ -255,26 +153,7 @@ RED.clipboard = (function() { | ||||
|             $(this).addClass('selected'); | ||||
|         }); | ||||
| 
 | ||||
|         $("#import-file-upload").change(function() { | ||||
|             var fileReader = new FileReader(); | ||||
|             fileReader.onload = function () { | ||||
|                 $("#clipboard-import").val(fileReader.result); | ||||
|                 validateImport(); | ||||
|             }; | ||||
|             fileReader.readAsText($(this).prop('files')[0]); | ||||
|         }) | ||||
|         $("#import-file-upload-btn").click(function(evt) { | ||||
|             evt.preventDefault(); | ||||
|             $("#import-file-upload").click(); | ||||
|         }) | ||||
| 
 | ||||
|         dialog.dialog("option","title",RED._("clipboard.importNodes")).dialog("open"); | ||||
|         popover = RED.popover.create({ | ||||
|             target: $("#clipboard-import"), | ||||
|             trigger: "manual", | ||||
|             direction: "bottom", | ||||
|             content: "" | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     function exportNodes() { | ||||
| @@ -323,18 +202,9 @@ RED.clipboard = (function() { | ||||
|             var flow = ""; | ||||
|             var nodes = null; | ||||
|             if (type === 'export-range-selected') { | ||||
|                 var selection = RED.workspaces.selection(); | ||||
|                 if (selection.length > 0) { | ||||
|                     nodes = []; | ||||
|                     selection.forEach(function(n) { | ||||
|                         nodes.push(n); | ||||
|                         nodes = nodes.concat(RED.nodes.filterNodes({z:n.id})); | ||||
|                     }); | ||||
|                 } else { | ||||
|                     nodes = RED.view.selection().nodes||[]; | ||||
|                 } | ||||
|                 var selection = RED.view.selection(); | ||||
|                 // Don't include the subflow meta-port nodes in the exported selection
 | ||||
|                 nodes = RED.nodes.createExportableNodeSet(nodes.filter(function(n) { return n.type !== 'subflow'})); | ||||
|                 nodes = RED.nodes.createExportableNodeSet(selection.nodes.filter(function(n) { return n.type !== 'subflow'})); | ||||
|             } else if (type === 'export-range-flow') { | ||||
|                 var activeWorkspace = RED.workspaces.active(); | ||||
|                 nodes = RED.nodes.filterNodes({z:activeWorkspace}); | ||||
| @@ -364,17 +234,12 @@ RED.clipboard = (function() { | ||||
|         $("#clipboard-dialog-cancel").hide(); | ||||
|         $("#clipboard-dialog-copy").hide(); | ||||
|         $("#clipboard-dialog-close").hide(); | ||||
|         var selection = RED.workspaces.selection(); | ||||
|         if (selection.length > 0) { | ||||
|         var selection = RED.view.selection(); | ||||
|         if (selection.nodes) { | ||||
|             $("#export-range-selected").click(); | ||||
|         } else { | ||||
|             selection = RED.view.selection(); | ||||
|             if (selection.nodes) { | ||||
|                 $("#export-range-selected").click(); | ||||
|             } else { | ||||
|                 $("#export-range-selected").addClass('disabled').removeClass('selected'); | ||||
|                 $("#export-range-flow").click(); | ||||
|             } | ||||
|             $("#export-range-selected").addClass('disabled').removeClass('selected'); | ||||
|             $("#export-range-flow").click(); | ||||
|         } | ||||
|         if (format === "export-format-full") { | ||||
|             $("#export-format-full").click(); | ||||
| @@ -400,8 +265,6 @@ RED.clipboard = (function() { | ||||
|             $("#clipboard-dialog-cancel").show(); | ||||
|             $("#clipboard-dialog-copy").show(); | ||||
|         } | ||||
|         $("#clipboard-dialog-download").show(); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     function hideDropTarget() { | ||||
| @@ -413,20 +276,9 @@ RED.clipboard = (function() { | ||||
|         if (typeof value !== "string" ) { | ||||
|             value = JSON.stringify(value, function(key,value) { | ||||
|                 if (value !== null && typeof value === 'object') { | ||||
|                     if (value.__enc__) { | ||||
|                         if (value.hasOwnProperty('data') && value.hasOwnProperty('length')) { | ||||
|                             truncated = value.data.length !== value.length; | ||||
|                             return value.data; | ||||
|                         } | ||||
|                         if (value.type === 'function' || value.type === 'internal') { | ||||
|                             return undefined | ||||
|                         } | ||||
|                         if (value.type === 'number') { | ||||
|                             // Handle NaN and Infinity - they are not permitted
 | ||||
|                             // in JSON. We can either substitute with a String
 | ||||
|                             // representation or null
 | ||||
|                             return null; | ||||
|                         } | ||||
|                     if (value.__encoded__ && value.hasOwnProperty('data') && value.hasOwnProperty('length')) { | ||||
|                         truncated = value.data.length !== value.length; | ||||
|                         return value.data; | ||||
|                     } | ||||
|                 } | ||||
|                 return value; | ||||
| @@ -457,6 +309,18 @@ RED.clipboard = (function() { | ||||
| 
 | ||||
|             $('<input type="text" id="clipboard-hidden">').appendTo("body"); | ||||
| 
 | ||||
|             RED.events.on("view:selection-changed",function(selection) { | ||||
|                 if (!selection.nodes) { | ||||
|                     RED.menu.setDisabled("menu-item-export",true); | ||||
|                     RED.menu.setDisabled("menu-item-export-clipboard",true); | ||||
|                     RED.menu.setDisabled("menu-item-export-library",true); | ||||
|                 } else { | ||||
|                     RED.menu.setDisabled("menu-item-export",false); | ||||
|                     RED.menu.setDisabled("menu-item-export-clipboard",false); | ||||
|                     RED.menu.setDisabled("menu-item-export-library",false); | ||||
|                 } | ||||
|             }); | ||||
| 
 | ||||
|             RED.actions.add("core:show-export-dialog",exportNodes); | ||||
|             RED.actions.add("core:show-import-dialog",importNodes); | ||||
| 
 | ||||
							
								
								
									
										81
									
								
								editor/js/ui/common/panels.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,81 @@ | ||||
| /** | ||||
|  * Copyright JS Foundation and other contributors, http://js.foundation | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  * http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  **/ | ||||
|  | ||||
|  | ||||
| RED.panels = (function() { | ||||
|  | ||||
|     function createPanel(options) { | ||||
|         var container = options.container || $("#"+options.id); | ||||
|         var children = container.children(); | ||||
|         if (children.length !== 2) { | ||||
|             throw new Error("Container must have exactly two children"); | ||||
|         } | ||||
|  | ||||
|         container.addClass("red-ui-panels"); | ||||
|         var separator = $('<div class="red-ui-panels-separator"></div>').insertAfter(children[0]); | ||||
|         var startPosition; | ||||
|         var panelHeights = []; | ||||
|         var modifiedHeights = false; | ||||
|         var panelRatio; | ||||
|  | ||||
|         separator.draggable({ | ||||
|                 axis: "y", | ||||
|                 containment: container, | ||||
|                 scroll: false, | ||||
|                 start:function(event,ui) { | ||||
|                     var height = container.height(); | ||||
|                     startPosition = ui.position.top; | ||||
|                     panelHeights = [$(children[0]).height(),$(children[1]).height()]; | ||||
|                 }, | ||||
|                 drag: function(event,ui) { | ||||
|                     var height = container.height(); | ||||
|                     var delta = ui.position.top-startPosition; | ||||
|                     var newHeights = [panelHeights[0]+delta,panelHeights[1]-delta]; | ||||
|                     $(children[0]).height(newHeights[0]); | ||||
|                     $(children[1]).height(newHeights[1]); | ||||
|                     if (options.resize) { | ||||
|                         options.resize(newHeights[0],newHeights[1]); | ||||
|                     } | ||||
|                     ui.position.top -= delta; | ||||
|                     panelRatio = newHeights[0]/height; | ||||
|                 }, | ||||
|                 stop:function(event,ui) { | ||||
|                     modifiedHeights = true; | ||||
|                 } | ||||
|         }); | ||||
|  | ||||
|         return { | ||||
|             resize: function(height) { | ||||
|                 var panelHeights = [$(children[0]).height(),$(children[1]).height()]; | ||||
|                 container.height(height); | ||||
|                 if (modifiedHeights) { | ||||
|                     var topPanelHeight = panelRatio*height; | ||||
|                     var bottomPanelHeight = height - topPanelHeight - 48; | ||||
|                     panelHeights = [topPanelHeight,bottomPanelHeight]; | ||||
|                     $(children[0]).height(panelHeights[0]); | ||||
|                     $(children[1]).height(panelHeights[1]); | ||||
|                 } | ||||
|                 if (options.resize) { | ||||
|                     options.resize(panelHeights[0],panelHeights[1]); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         create: createPanel | ||||
|     } | ||||
| })(); | ||||
							
								
								
									
										149
									
								
								editor/js/ui/common/popover.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,149 @@ | ||||
| /** | ||||
|  * Copyright JS Foundation and other contributors, http://js.foundation | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  * http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  **/ | ||||
|  | ||||
| RED.popover = (function() { | ||||
|     var deltaSizes = { | ||||
|         "default": { | ||||
|             top: 10, | ||||
|             leftRight: 17, | ||||
|             leftLeft: 25 | ||||
|         }, | ||||
|         "small": { | ||||
|             top: 5, | ||||
|             leftRight: 17, | ||||
|             leftLeft: 16 | ||||
|         } | ||||
|     } | ||||
|     function createPopover(options) { | ||||
|         var target = options.target; | ||||
|         var direction = options.direction || "right"; | ||||
|         var trigger = options.trigger; | ||||
|         var content = options.content; | ||||
|         var delay = options.delay; | ||||
|         var autoClose = options.autoClose; | ||||
|         var width = options.width||"auto"; | ||||
|         var size = options.size||"default"; | ||||
|         if (!deltaSizes[size]) { | ||||
|             throw new Error("Invalid RED.popover size value:",size); | ||||
|         } | ||||
|  | ||||
|         var timer = null; | ||||
|         var active; | ||||
|         var div; | ||||
|  | ||||
|         var openPopup = function(instant) { | ||||
|             if (active) { | ||||
|                 div = $('<div class="red-ui-popover red-ui-popover-'+direction+'"></div>').appendTo("body"); | ||||
|                 if (size !== "default") { | ||||
|                     div.addClass("red-ui-popover-size-"+size); | ||||
|                 } | ||||
|                 if (typeof content === 'function') { | ||||
|                     content.call(res).appendTo(div); | ||||
|                 } else { | ||||
|                     div.html(content); | ||||
|                 } | ||||
|                 if (width !== "auto") { | ||||
|                     div.width(width); | ||||
|                 } | ||||
|  | ||||
|  | ||||
|                 var targetPos = target.offset(); | ||||
|                 var targetWidth = target.width(); | ||||
|                 var targetHeight = target.height(); | ||||
|                 var divHeight = div.height(); | ||||
|                 var divWidth = div.width(); | ||||
|                 if (direction === 'right') { | ||||
|                     div.css({top: targetPos.top+targetHeight/2-divHeight/2-deltaSizes[size].top,left:targetPos.left+targetWidth+deltaSizes[size].leftRight}); | ||||
|                 } else if (direction === 'left') { | ||||
|                     div.css({top: targetPos.top+targetHeight/2-divHeight/2-deltaSizes[size].top,left:targetPos.left-deltaSizes[size].leftLeft-divWidth}); | ||||
|                 } | ||||
|                 if (instant) { | ||||
|                     div.show(); | ||||
|                 } else { | ||||
|                     div.fadeIn("fast"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         var closePopup = function(instant) { | ||||
|             if (!active) { | ||||
|                 if (div) { | ||||
|                     if (instant) { | ||||
|                         $(this).remove(); | ||||
|                     } else { | ||||
|                         div.fadeOut("fast",function() { | ||||
|                             $(this).remove(); | ||||
|                         }); | ||||
|                     } | ||||
|                     div = null; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (trigger === 'hover') { | ||||
|             target.on('mouseenter',function(e) { | ||||
|                 clearTimeout(timer); | ||||
|                 active = true; | ||||
|                 timer = setTimeout(openPopup,delay.show); | ||||
|             }); | ||||
|             target.on('mouseleave', function(e) { | ||||
|                 if (timer) { | ||||
|                     clearTimeout(timer); | ||||
|                 } | ||||
|                 active = false; | ||||
|                 setTimeout(closePopup,delay.hide); | ||||
|             }); | ||||
|         } else if (trigger === 'click') { | ||||
|             target.click(function(e) { | ||||
|                 e.preventDefault(); | ||||
|                 e.stopPropagation(); | ||||
|                 active = !active; | ||||
|                 if (!active) { | ||||
|                     closePopup(); | ||||
|                 } else { | ||||
|                     openPopup(); | ||||
|                 } | ||||
|             }); | ||||
|         } else if (autoClose) { | ||||
|             setTimeout(function() { | ||||
|                 active = false; | ||||
|                 closePopup(); | ||||
|             },autoClose); | ||||
|         } | ||||
|         var res = { | ||||
|             setContent: function(_content) { | ||||
|                 content = _content; | ||||
|                 return res; | ||||
|             }, | ||||
|             open: function (instant) { | ||||
|                 active = true; | ||||
|                 openPopup(instant); | ||||
|                 return res; | ||||
|             }, | ||||
|             close: function (instant) { | ||||
|                 active = false; | ||||
|                 closePopup(instant); | ||||
|                 return res; | ||||
|             } | ||||
|         } | ||||
|         return res; | ||||
|  | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         create: createPopover | ||||
|     } | ||||
|  | ||||
| })(); | ||||
							
								
								
									
										379
									
								
								editor/js/ui/common/tabs.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,379 @@ | ||||
| /** | ||||
|  * Copyright JS Foundation and other contributors, http://js.foundation | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  * http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  **/ | ||||
|  | ||||
|  | ||||
|  | ||||
| RED.tabs = (function() { | ||||
|     function createTabs(options) { | ||||
|         var tabs = {}; | ||||
|         var currentTabWidth; | ||||
|         var currentActiveTabWidth = 0; | ||||
|  | ||||
|         var ul = options.element || $("#"+options.id); | ||||
|         var wrapper = ul.wrap( "<div>" ).parent(); | ||||
|         var scrollContainer = ul.wrap( "<div>" ).parent(); | ||||
|         wrapper.addClass("red-ui-tabs"); | ||||
|         if (options.vertical) { | ||||
|             wrapper.addClass("red-ui-tabs-vertical"); | ||||
|         } | ||||
|         if (options.addButton && typeof options.addButton === 'function') { | ||||
|             wrapper.addClass("red-ui-tabs-add"); | ||||
|             var addButton = $('<div class="red-ui-tab-button"><a href="#"><i class="fa fa-plus"></i></a></div>').appendTo(wrapper); | ||||
|             addButton.find('a').click(function(evt) { | ||||
|                 evt.preventDefault(); | ||||
|                 options.addButton(); | ||||
|             }) | ||||
|  | ||||
|         } | ||||
|         var scrollLeft; | ||||
|         var scrollRight; | ||||
|  | ||||
|         if (options.scrollable) { | ||||
|             wrapper.addClass("red-ui-tabs-scrollable"); | ||||
|             scrollContainer.addClass("red-ui-tabs-scroll-container"); | ||||
|             scrollContainer.scroll(updateScroll); | ||||
|             scrollLeft = $('<div class="red-ui-tab-button red-ui-tab-scroll red-ui-tab-scroll-left"><a href="#" style="display:none;"><i class="fa fa-caret-left"></i></a></div>').appendTo(wrapper).find("a"); | ||||
|             scrollLeft.on('mousedown',function(evt) { scrollEventHandler(evt,'-=150') }).on('click',function(evt){ evt.preventDefault();}); | ||||
|             scrollRight = $('<div class="red-ui-tab-button red-ui-tab-scroll red-ui-tab-scroll-right"><a href="#" style="display:none;"><i class="fa fa-caret-right"></i></a></div>').appendTo(wrapper).find("a"); | ||||
|             scrollRight.on('mousedown',function(evt) { scrollEventHandler(evt,'+=150') }).on('click',function(evt){ evt.preventDefault();}); | ||||
|         } | ||||
|         function scrollEventHandler(evt,dir) { | ||||
|             evt.preventDefault(); | ||||
|             if ($(this).hasClass('disabled')) { | ||||
|                 return; | ||||
|             } | ||||
|             var currentScrollLeft = scrollContainer.scrollLeft(); | ||||
|             scrollContainer.animate( { scrollLeft: dir }, 100); | ||||
|             var interval = setInterval(function() { | ||||
|                 var newScrollLeft = scrollContainer.scrollLeft() | ||||
|                 if (newScrollLeft === currentScrollLeft) { | ||||
|                     clearInterval(interval); | ||||
|                     return; | ||||
|                 } | ||||
|                 currentScrollLeft = newScrollLeft; | ||||
|                 scrollContainer.animate( { scrollLeft: dir }, 100); | ||||
|             },100); | ||||
|             $(this).one('mouseup',function() { | ||||
|                 clearInterval(interval); | ||||
|             }) | ||||
|         } | ||||
|  | ||||
|  | ||||
|         ul.children().first().addClass("active"); | ||||
|         ul.children().addClass("red-ui-tab"); | ||||
|  | ||||
|         function onTabClick() { | ||||
|             if (options.onclick) { | ||||
|                 options.onclick(tabs[$(this).attr('href').slice(1)]); | ||||
|             } | ||||
|             activateTab($(this)); | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         function updateScroll() { | ||||
|             if (ul.children().length !== 0) { | ||||
|                 var sl = scrollContainer.scrollLeft(); | ||||
|                 var scWidth = scrollContainer.width(); | ||||
|                 var ulWidth = ul.width(); | ||||
|                 if (sl === 0) { | ||||
|                     scrollLeft.hide(); | ||||
|                 } else { | ||||
|                     scrollLeft.show(); | ||||
|                 } | ||||
|                 if (sl === ulWidth-scWidth) { | ||||
|                     scrollRight.hide(); | ||||
|                 } else { | ||||
|                     scrollRight.show(); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         function onTabDblClick() { | ||||
|             if (options.ondblclick) { | ||||
|                 options.ondblclick(tabs[$(this).attr('href').slice(1)]); | ||||
|             } | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         function activateTab(link) { | ||||
|             if (typeof link === "string") { | ||||
|                 link = ul.find("a[href='#"+link+"']"); | ||||
|             } | ||||
|             if (link.length === 0) { | ||||
|                 return; | ||||
|             } | ||||
|             if (!link.parent().hasClass("active")) { | ||||
|                 ul.children().removeClass("active"); | ||||
|                 ul.children().css({"transition": "width 100ms"}); | ||||
|                 link.parent().addClass("active"); | ||||
|                 if (options.scrollable) { | ||||
|                     var pos = link.parent().position().left; | ||||
|                     if (pos-21 < 0) { | ||||
|                         scrollContainer.animate( { scrollLeft: '+='+(pos-50) }, 300); | ||||
|                     } else if (pos + 120 > scrollContainer.width()) { | ||||
|                         scrollContainer.animate( { scrollLeft: '+='+(pos + 140-scrollContainer.width()) }, 300); | ||||
|                     } | ||||
|                 } | ||||
|                 if (options.onchange) { | ||||
|                     options.onchange(tabs[link.attr('href').slice(1)]); | ||||
|                 } | ||||
|                 updateTabWidths(); | ||||
|                 setTimeout(function() { | ||||
|                     ul.children().css({"transition": ""}); | ||||
|                 },100); | ||||
|             } | ||||
|         } | ||||
|         function activatePreviousTab() { | ||||
|             var previous = ul.find("li.active").prev(); | ||||
|             if (previous.length > 0) { | ||||
|                 activateTab(previous.find("a")); | ||||
|             } | ||||
|         } | ||||
|         function activateNextTab() { | ||||
|             var next = ul.find("li.active").next(); | ||||
|             if (next.length > 0) { | ||||
|                 activateTab(next.find("a")); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         function updateTabWidths() { | ||||
|             if (options.vertical) { | ||||
|                 return; | ||||
|             } | ||||
|             var tabs = ul.find("li.red-ui-tab"); | ||||
|             var width = wrapper.width(); | ||||
|             var tabCount = tabs.size(); | ||||
|             var tabWidth = (width-12-(tabCount*6))/tabCount; | ||||
|             currentTabWidth = (100*tabWidth/width)+"%"; | ||||
|             currentActiveTabWidth = currentTabWidth+"%"; | ||||
|             if (options.scrollable) { | ||||
|                 tabWidth = Math.max(tabWidth,140); | ||||
|                 currentTabWidth = tabWidth+"px"; | ||||
|                 currentActiveTabWidth = 0; | ||||
|                 var listWidth = Math.max(wrapper.width(),12+(tabWidth+6)*tabCount); | ||||
|                 ul.width(listWidth); | ||||
|                 updateScroll(); | ||||
|             } else if (options.hasOwnProperty("minimumActiveTabWidth")) { | ||||
|                 if (tabWidth < options.minimumActiveTabWidth) { | ||||
|                     tabCount -= 1; | ||||
|                     tabWidth = (width-12-options.minimumActiveTabWidth-(tabCount*6))/tabCount; | ||||
|                     currentTabWidth = (100*tabWidth/width)+"%"; | ||||
|                     currentActiveTabWidth = options.minimumActiveTabWidth+"px"; | ||||
|                 } else { | ||||
|                     currentActiveTabWidth = 0; | ||||
|                 } | ||||
|             } | ||||
|             tabs.css({width:currentTabWidth}); | ||||
|             if (tabWidth < 50) { | ||||
|                 ul.find(".red-ui-tab-close").hide(); | ||||
|                 ul.find(".red-ui-tab-icon").hide(); | ||||
|                 ul.find(".red-ui-tab-label").css({paddingLeft:Math.min(12,Math.max(0,tabWidth-38))+"px"}) | ||||
|             } else { | ||||
|                 ul.find(".red-ui-tab-close").show(); | ||||
|                 ul.find(".red-ui-tab-icon").show(); | ||||
|                 ul.find(".red-ui-tab-label").css({paddingLeft:""}) | ||||
|             } | ||||
|             if (currentActiveTabWidth !== 0) { | ||||
|                 ul.find("li.red-ui-tab.active").css({"width":options.minimumActiveTabWidth}); | ||||
|                 ul.find("li.red-ui-tab.active .red-ui-tab-close").show(); | ||||
|                 ul.find("li.red-ui-tab.active .red-ui-tab-icon").show(); | ||||
|                 ul.find("li.red-ui-tab.active .red-ui-tab-label").css({paddingLeft:""}) | ||||
|             } | ||||
|  | ||||
|         } | ||||
|  | ||||
|         ul.find("li.red-ui-tab a").on("click",onTabClick).on("dblclick",onTabDblClick); | ||||
|         setTimeout(function() { | ||||
|             updateTabWidths(); | ||||
|         },0); | ||||
|  | ||||
|  | ||||
|         function removeTab(id) { | ||||
|             var li = ul.find("a[href='#"+id+"']").parent(); | ||||
|             if (li.hasClass("active")) { | ||||
|                 var tab = li.prev(); | ||||
|                 if (tab.size() === 0) { | ||||
|                     tab = li.next(); | ||||
|                 } | ||||
|                 activateTab(tab.find("a")); | ||||
|             } | ||||
|             li.remove(); | ||||
|             if (options.onremove) { | ||||
|                 options.onremove(tabs[id]); | ||||
|             } | ||||
|             delete tabs[id]; | ||||
|             updateTabWidths(); | ||||
|         } | ||||
|  | ||||
|         return { | ||||
|             addTab: function(tab) { | ||||
|                 tabs[tab.id] = tab; | ||||
|                 var li = $("<li/>",{class:"red-ui-tab"}).appendTo(ul); | ||||
|                 li.attr('id',"red-ui-tab-"+(tab.id.replace(".","-"))); | ||||
|                 li.data("tabId",tab.id); | ||||
|                 var link = $("<a/>",{href:"#"+tab.id, class:"red-ui-tab-label"}).appendTo(li); | ||||
|                 if (tab.icon) { | ||||
|                     $('<img src="'+tab.icon+'" class="red-ui-tab-icon"/>').appendTo(link); | ||||
|                 } | ||||
|                 var span = $('<span/>',{class:"bidiAware"}).text(tab.label).appendTo(link); | ||||
|                 span.attr('dir', RED.text.bidi.resolveBaseTextDir(tab.label)); | ||||
|  | ||||
|                 link.on("click",onTabClick); | ||||
|                 link.on("dblclick",onTabDblClick); | ||||
|                 if (tab.closeable) { | ||||
|                     var closeLink = $("<a/>",{href:"#",class:"red-ui-tab-close"}).appendTo(li); | ||||
|                     closeLink.append('<i class="fa fa-times" />'); | ||||
|  | ||||
|                     closeLink.on("click",function(event) { | ||||
|                         event.preventDefault(); | ||||
|                         removeTab(tab.id); | ||||
|                     }); | ||||
|                 } | ||||
|                 updateTabWidths(); | ||||
|                 if (options.onadd) { | ||||
|                     options.onadd(tab); | ||||
|                 } | ||||
|                 link.attr("title",tab.label); | ||||
|                 if (ul.find("li.red-ui-tab").size() == 1) { | ||||
|                     activateTab(link); | ||||
|                 } | ||||
|                 if (options.onreorder) { | ||||
|                     var originalTabOrder; | ||||
|                     var tabDragIndex; | ||||
|                     var tabElements = []; | ||||
|                     var startDragIndex; | ||||
|  | ||||
|                     li.draggable({ | ||||
|                         axis:"x", | ||||
|                         distance: 20, | ||||
|                         start: function(event,ui) { | ||||
|                             originalTabOrder = []; | ||||
|                             tabElements = []; | ||||
|                             ul.children().each(function(i) { | ||||
|                                 tabElements[i] = { | ||||
|                                     el:$(this), | ||||
|                                     text: $(this).text(), | ||||
|                                     left: $(this).position().left, | ||||
|                                     width: $(this).width() | ||||
|                                 }; | ||||
|                                 if ($(this).is(li)) { | ||||
|                                     tabDragIndex = i; | ||||
|                                     startDragIndex = i; | ||||
|                                 } | ||||
|                                 originalTabOrder.push($(this).data("tabId")); | ||||
|                             }); | ||||
|                             ul.children().each(function(i) { | ||||
|                                 if (i!==tabDragIndex) { | ||||
|                                     $(this).css({ | ||||
|                                         position: 'absolute', | ||||
|                                         left: tabElements[i].left+"px", | ||||
|                                         width: tabElements[i].width+2, | ||||
|                                         transition: "left 0.3s" | ||||
|                                     }); | ||||
|                                 } | ||||
|  | ||||
|                             }) | ||||
|                             if (!li.hasClass('active')) { | ||||
|                                 li.css({'zIndex':1}); | ||||
|                             } | ||||
|                         }, | ||||
|                         drag: function(event,ui) { | ||||
|                             ui.position.left += tabElements[tabDragIndex].left+scrollContainer.scrollLeft(); | ||||
|                             var tabCenter = ui.position.left + tabElements[tabDragIndex].width/2 - scrollContainer.scrollLeft(); | ||||
|                             for (var i=0;i<tabElements.length;i++) { | ||||
|                                 if (i === tabDragIndex) { | ||||
|                                     continue; | ||||
|                                 } | ||||
|                                 if (tabCenter > tabElements[i].left && tabCenter < tabElements[i].left+tabElements[i].width) { | ||||
|                                     if (i < tabDragIndex) { | ||||
|                                         tabElements[i].left += tabElements[tabDragIndex].width+8; | ||||
|                                         tabElements[tabDragIndex].el.detach().insertBefore(tabElements[i].el); | ||||
|                                     } else { | ||||
|                                         tabElements[i].left -= tabElements[tabDragIndex].width+8; | ||||
|                                         tabElements[tabDragIndex].el.detach().insertAfter(tabElements[i].el); | ||||
|                                     } | ||||
|                                     tabElements[i].el.css({left:tabElements[i].left+"px"}); | ||||
|  | ||||
|                                     tabElements.splice(i, 0, tabElements.splice(tabDragIndex, 1)[0]); | ||||
|  | ||||
|                                     tabDragIndex = i; | ||||
|                                     break; | ||||
|                                 } | ||||
|                             } | ||||
|                         }, | ||||
|                         stop: function(event,ui) { | ||||
|                             ul.children().css({position:"relative",left:"",transition:""}); | ||||
|                             if (!li.hasClass('active')) { | ||||
|                                 li.css({zIndex:""}); | ||||
|                             } | ||||
|                             updateTabWidths(); | ||||
|                             if (startDragIndex !== tabDragIndex) { | ||||
|                                 options.onreorder(originalTabOrder, $.makeArray(ul.children().map(function() { return $(this).data('tabId');}))); | ||||
|                             } | ||||
|                             activateTab(tabElements[tabDragIndex].el.data('tabId')); | ||||
|                         } | ||||
|                     }) | ||||
|                 } | ||||
|             }, | ||||
|             removeTab: removeTab, | ||||
|             activateTab: activateTab, | ||||
|             nextTab: activateNextTab, | ||||
|             previousTab: activatePreviousTab, | ||||
|             resize: updateTabWidths, | ||||
|             count: function() { | ||||
|                 return ul.find("li.red-ui-tab").size(); | ||||
|             }, | ||||
|             contains: function(id) { | ||||
|                 return ul.find("a[href='#"+id+"']").length > 0; | ||||
|             }, | ||||
|             renameTab: function(id,label) { | ||||
|                 tabs[id].label = label; | ||||
|                 var tab = ul.find("a[href='#"+id+"']"); | ||||
|                 tab.attr("title",label); | ||||
|                 tab.find("span.bidiAware").text(label).attr('dir', RED.text.bidi.resolveBaseTextDir(label)); | ||||
|                 updateTabWidths(); | ||||
|             }, | ||||
|             order: function(order) { | ||||
|                 var existingTabOrder = $.makeArray(ul.children().map(function() { return $(this).data('tabId');})); | ||||
|                 if (existingTabOrder.length !== order.length) { | ||||
|                     return | ||||
|                 } | ||||
|                 var i; | ||||
|                 var match = true; | ||||
|                 for (i=0;i<order.length;i++) { | ||||
|                     if (order[i] !== existingTabOrder[i]) { | ||||
|                         match = false; | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|                 if (match) { | ||||
|                     return; | ||||
|                 } | ||||
|                 var existingTabMap = {}; | ||||
|                 var existingTabs = ul.children().detach().each(function() { | ||||
|                     existingTabMap[$(this).data("tabId")] = $(this); | ||||
|                 }); | ||||
|                 for (i=0;i<order.length;i++) { | ||||
|                     existingTabMap[order[i]].appendTo(ul); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         create: createTabs | ||||
|     } | ||||
| })(); | ||||