2013-09-05 15:02:48 +01:00
/ * *
* Copyright 2013 IBM Corp .
*
* 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 .
* * /
2014-08-08 00:01:35 +01:00
RED . nodes = ( function ( ) {
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
var node _defs = { } ;
var nodes = [ ] ;
var configNodes = { } ;
var links = [ ] ;
2013-10-25 21:34:00 +01:00
var defaultWorkspace ;
var workspaces = { } ;
2014-02-24 23:35:11 +00:00
var subflows = { } ;
2014-08-28 00:35:07 +01:00
var registry = ( function ( ) {
var nodeList = [ ] ;
var nodeSets = { } ;
var typeToId = { } ;
var nodeDefinitions = { } ;
var exports = {
getNodeList : function ( ) {
return nodeList ;
} ,
setNodeList : function ( list ) {
nodeList = [ ] ;
for ( var i = 0 ; i < list . length ; i ++ ) {
var ns = list [ i ] ;
exports . addNodeSet ( ns ) ;
}
} ,
addNodeSet : function ( ns ) {
ns . added = false ;
nodeSets [ ns . id ] = ns ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
typeToId [ ns . types [ j ] ] = ns . id ;
}
nodeList . push ( ns ) ;
} ,
removeNodeSet : function ( id ) {
var ns = nodeSets [ id ] ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
if ( ns . added ) {
// TODO: too tightly coupled into palette UI
RED . palette . remove ( ns . types [ j ] ) ;
var def = nodeDefinitions [ ns . types [ j ] ] ;
if ( def . onpaletteremove && typeof def . onpaletteremove === "function" ) {
def . onpaletteremove . call ( def ) ;
}
}
delete typeToId [ ns . types [ j ] ] ;
}
delete nodeSets [ id ] ;
for ( var i = 0 ; i < nodeList . length ; i ++ ) {
if ( nodeList [ i ] . id == id ) {
nodeList . splice ( i , 1 ) ;
break ;
}
}
return ns ;
} ,
getNodeSet : function ( id ) {
return nodeSets [ id ] ;
} ,
enableNodeSet : function ( id ) {
var ns = nodeSets [ id ] ;
ns . enabled = true ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
// TODO: too tightly coupled into palette UI
RED . palette . show ( ns . types [ j ] ) ;
var def = nodeDefinitions [ ns . types [ j ] ] ;
if ( def . onpaletteadd && typeof def . onpaletteadd === "function" ) {
def . onpaletteadd . call ( def ) ;
}
}
} ,
disableNodeSet : function ( id ) {
var ns = nodeSets [ id ] ;
ns . enabled = false ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
// TODO: too tightly coupled into palette UI
RED . palette . hide ( ns . types [ j ] ) ;
var def = nodeDefinitions [ ns . types [ j ] ] ;
if ( def . onpaletteremove && typeof def . onpaletteremove === "function" ) {
def . onpaletteremove . call ( def ) ;
}
}
} ,
registerNodeType : function ( nt , def ) {
nodeDefinitions [ nt ] = def ;
2014-02-24 23:35:11 +00:00
if ( def . category != "subflows" ) {
nodeSets [ typeToId [ nt ] ] . added = true ;
// TODO: too tightly coupled into palette UI
}
2014-08-28 00:35:07 +01:00
RED . palette . add ( nt , def ) ;
if ( def . onpaletteadd && typeof def . onpaletteadd === "function" ) {
def . onpaletteadd . call ( def ) ;
}
} ,
2014-02-24 23:35:11 +00:00
removeNodeType : function ( nt ) {
if ( nt . substring ( 0 , 8 ) != "subflow:" ) {
throw new Error ( "this api is subflow only. called with:" , nt ) ;
}
delete nodeDefinitions [ nt ] ;
RED . palette . remove ( nt ) ;
} ,
2014-08-28 00:35:07 +01:00
getNodeType : function ( nt ) {
return nodeDefinitions [ nt ] ;
}
2014-09-24 09:57:45 +01:00
} ;
2014-08-28 00:35:07 +01:00
return exports ;
} ) ( ) ;
2013-10-25 21:34:00 +01:00
function getID ( ) {
return ( 1 + Math . random ( ) * 4294967295 ) . toString ( 16 ) ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function addNode ( n ) {
if ( n . _def . category == "config" ) {
configNodes [ n . id ] = n ;
2014-02-26 22:58:44 +00:00
RED . sidebar . config . refresh ( ) ;
2013-09-05 15:02:48 +01:00
} else {
n . dirty = true ;
2014-01-25 22:31:43 +00:00
var updatedConfigNode = false ;
for ( var d in n . _def . defaults ) {
2014-08-08 00:01:35 +01:00
if ( n . _def . defaults . hasOwnProperty ( d ) ) {
var property = n . _def . defaults [ d ] ;
if ( property . type ) {
2014-09-24 09:57:45 +01:00
var type = registry . getNodeType ( property . type ) ;
2014-08-08 00:01:35 +01:00
if ( type && type . category == "config" ) {
var configNode = configNodes [ n [ d ] ] ;
if ( configNode ) {
updatedConfigNode = true ;
configNode . users . push ( n ) ;
}
2014-01-25 22:31:43 +00:00
}
}
}
}
if ( updatedConfigNode ) {
2014-02-26 22:58:44 +00:00
RED . sidebar . config . refresh ( ) ;
2014-01-25 22:31:43 +00:00
}
2014-02-24 23:35:11 +00:00
if ( n . _def . category == "subflows" && typeof n . i === "undefined" ) {
var nextId = 0 ;
RED . nodes . eachNode ( function ( node ) {
nextId = Math . max ( nextId , node . i || 0 ) ;
} ) ;
n . i = nextId + 1 ;
}
nodes . push ( n ) ;
2013-09-05 15:02:48 +01:00
}
}
function addLink ( l ) {
links . push ( l ) ;
}
function addConfig ( c ) {
configNodes [ c . id ] = c ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function getNode ( id ) {
if ( id in configNodes ) {
return configNodes [ id ] ;
} else {
for ( var n in nodes ) {
if ( nodes [ n ] . id == id ) {
return nodes [ n ] ;
}
}
}
return null ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function removeNode ( id ) {
var removedLinks = [ ] ;
if ( id in configNodes ) {
delete configNodes [ id ] ;
2014-02-27 16:47:28 +00:00
RED . sidebar . config . refresh ( ) ;
2013-09-05 15:02:48 +01:00
} else {
var node = getNode ( id ) ;
if ( node ) {
nodes . splice ( nodes . indexOf ( node ) , 1 ) ;
removedLinks = links . filter ( function ( l ) { return ( l . source === node ) || ( l . target === node ) ; } ) ;
2014-02-24 23:35:11 +00:00
removedLinks . forEach ( function ( l ) { links . splice ( links . indexOf ( l ) , 1 ) ; } ) ;
var updatedConfigNode = false ;
for ( var d in node . _def . defaults ) {
if ( node . _def . defaults . hasOwnProperty ( d ) ) {
var property = node . _def . defaults [ d ] ;
if ( property . type ) {
var type = registry . getNodeType ( property . type ) ;
if ( type && type . category == "config" ) {
var configNode = configNodes [ node [ d ] ] ;
if ( configNode ) {
updatedConfigNode = true ;
var users = configNode . users ;
users . splice ( users . indexOf ( node ) , 1 ) ;
}
2014-08-08 00:01:35 +01:00
}
2014-01-25 22:31:43 +00:00
}
}
}
2014-02-24 23:35:11 +00:00
if ( updatedConfigNode ) {
RED . sidebar . config . refresh ( ) ;
}
2014-01-25 22:31:43 +00:00
}
2013-09-05 15:02:48 +01:00
}
return removedLinks ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function removeLink ( l ) {
var index = links . indexOf ( l ) ;
if ( index != - 1 ) {
links . splice ( index , 1 ) ;
}
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function refreshValidation ( ) {
2014-08-08 00:01:35 +01:00
for ( var n = 0 ; n < nodes . length ; n ++ ) {
2013-09-05 15:02:48 +01:00
RED . editor . validateNode ( nodes [ n ] ) ;
}
}
2013-12-08 21:55:34 +00:00
2013-10-25 21:34:00 +01:00
function addWorkspace ( ws ) {
workspaces [ ws . id ] = ws ;
}
2013-10-26 22:29:24 +01:00
function getWorkspace ( id ) {
return workspaces [ id ] ;
}
function removeWorkspace ( id ) {
delete workspaces [ id ] ;
var removedNodes = [ ] ;
var removedLinks = [ ] ;
2014-08-08 00:01:35 +01:00
var n ;
for ( n = 0 ; n < nodes . length ; n ++ ) {
2013-10-26 22:29:24 +01:00
var node = nodes [ n ] ;
if ( node . z == id ) {
removedNodes . push ( node ) ;
}
}
2014-08-08 00:01:35 +01:00
for ( n = 0 ; n < removedNodes . length ; n ++ ) {
2013-10-26 22:29:24 +01:00
var rmlinks = removeNode ( removedNodes [ n ] . id ) ;
removedLinks = removedLinks . concat ( rmlinks ) ;
}
return { nodes : removedNodes , links : removedLinks } ;
}
2013-12-11 22:22:33 +00:00
2014-02-24 23:35:11 +00:00
function addSubflow ( sf ) {
subflows [ sf . id ] = sf ;
RED . nodes . registerType ( "subflow:" + sf . id , {
defaults : { name : { value : "" } } ,
icon : "subflow.png" ,
category : "subflows" ,
inputs : sf . in . length ,
outputs : sf . out . length ,
color : "#da9" ,
label : function ( ) { return this . name || RED . nodes . subflow ( sf . id ) . name } ,
labelStyle : function ( ) { return this . name ? "node_label_italic" : "" ; } ,
paletteLabel : function ( ) { return RED . nodes . subflow ( sf . id ) . name }
} ) ;
}
function getSubflow ( id ) {
return subflows [ id ] ;
}
function removeSubflow ( sf ) {
delete subflows [ sf . id ] ;
registry . removeNodeType ( "subflow:" + sf . id ) ;
}
function subflowContains ( sfid , nodeid ) {
for ( var i = 0 ; i < nodes . length ; i ++ ) {
var node = nodes [ i ] ;
if ( node . z === sfid ) {
var m = /^subflow:(.+)$/ . exec ( node . type ) ;
if ( m ) {
if ( m [ 1 ] === nodeid ) {
return true ;
} else {
var result = subflowContains ( m [ 1 ] , nodeid ) ;
if ( result ) {
return true ;
}
}
}
}
}
return false ;
}
2013-09-05 15:02:48 +01:00
function getAllFlowNodes ( node ) {
var visited = { } ;
visited [ node . id ] = true ;
var nns = [ node ] ;
var stack = [ node ] ;
2014-08-08 00:01:35 +01:00
while ( stack . length !== 0 ) {
2013-09-05 15:02:48 +01:00
var n = stack . shift ( ) ;
var childLinks = links . filter ( function ( d ) { return ( d . source === n ) || ( d . target === n ) ; } ) ;
2014-08-08 00:01:35 +01:00
for ( var i = 0 ; i < childLinks . length ; i ++ ) {
2013-09-05 15:02:48 +01:00
var child = ( childLinks [ i ] . source === n ) ? childLinks [ i ] . target : childLinks [ i ] . source ;
2014-02-24 23:35:11 +00:00
var id = child . id ;
if ( ! id ) {
id = child . direction + ":" + child . i ;
}
if ( ! visited [ id ] ) {
visited [ id ] = true ;
2013-09-05 15:02:48 +01:00
nns . push ( child ) ;
stack . push ( child ) ;
}
}
}
return nns ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
/ * *
2013-12-08 21:55:34 +00:00
* Converts a node to an exportable JSON Object
2013-09-05 15:02:48 +01:00
* * /
2014-06-24 13:13:19 +03:00
function convertNode ( n , exportCreds ) {
exportCreds = exportCreds || false ;
2013-09-05 15:02:48 +01:00
var node = { } ;
node . id = n . id ;
node . type = n . type ;
2014-10-30 21:41:42 +00:00
if ( node . type == "unknown" ) {
for ( var p in n . _orig ) {
if ( n . _orig . hasOwnProperty ( p ) ) {
node [ p ] = n . _orig [ p ] ;
}
2014-08-08 00:01:35 +01:00
}
2014-10-30 21:41:42 +00:00
} else {
for ( var d in n . _def . defaults ) {
if ( n . _def . defaults . hasOwnProperty ( d ) ) {
node [ d ] = n [ d ] ;
}
}
if ( exportCreds && n . credentials ) {
node . credentials = { } ;
for ( var cred in n . _def . credentials ) {
if ( n . _def . credentials . hasOwnProperty ( cred ) ) {
if ( n . credentials [ cred ] != null ) {
node . credentials [ cred ] = n . credentials [ cred ] ;
}
2014-07-19 00:16:21 +01:00
}
}
}
2014-06-24 13:13:19 +03:00
}
2013-09-05 15:02:48 +01:00
if ( n . _def . category != "config" ) {
node . x = n . x ;
node . y = n . y ;
2013-10-20 23:11:38 +01:00
node . z = n . z ;
2013-09-05 15:02:48 +01:00
node . wires = [ ] ;
for ( var i = 0 ; i < n . outputs ; i ++ ) {
node . wires . push ( [ ] ) ;
}
var wires = links . filter ( function ( d ) { return d . source === n ; } ) ;
2014-08-08 00:01:35 +01:00
for ( var j = 0 ; j < wires . length ; j ++ ) {
var w = wires [ j ] ;
2014-02-24 23:35:11 +00:00
if ( w . target . type != "subflow" ) {
node . wires [ w . sourcePort ] . push ( w . target . id ) ;
}
2013-09-05 15:02:48 +01:00
}
}
return node ;
}
2013-12-08 21:55:34 +00:00
2014-02-24 23:35:11 +00:00
function convertSubflow ( n ) {
var node = { } ;
node . id = n . id ;
node . type = n . type ;
node . name = n . name ;
node . in = [ ] ;
node . out = [ ] ;
n . in . forEach ( function ( p ) {
var nIn = { x : p . x , y : p . y , wires : [ ] } ;
var wires = links . filter ( function ( d ) { return d . source === p } ) ;
for ( var i = 0 ; i < wires . length ; i ++ ) {
var w = wires [ i ] ;
if ( w . target . id != p . id ) {
nIn . wires . push ( { id : w . target . id } )
}
}
node . in . push ( nIn ) ;
} ) ;
n . out . forEach ( function ( p , c ) {
var nOut = { x : p . x , y : p . y , wires : [ ] } ;
var wires = links . filter ( function ( d ) { return d . target === p } ) ;
for ( i = 0 ; i < wires . length ; i ++ ) {
if ( wires [ i ] . source . type != "subflow" ) {
nOut . wires . push ( { id : wires [ i ] . source . id , port : wires [ i ] . sourcePort } )
} else {
nOut . wires . push ( { id : n . id , port : 0 } )
}
}
node . out . push ( nOut ) ;
} ) ;
return node ;
}
2013-09-05 15:02:48 +01:00
/ * *
* Converts the current node selection to an exportable JSON Object
* * /
function createExportableNodeSet ( set ) {
var nns = [ ] ;
var exportedConfigNodes = { } ;
2014-02-24 23:35:11 +00:00
var exportedSubflows = { } ;
2014-08-08 00:01:35 +01:00
for ( var n = 0 ; n < set . length ; n ++ ) {
2013-09-05 15:02:48 +01:00
var node = set [ n ] . n ;
2014-02-24 23:35:11 +00:00
if ( node . type . substring ( 0 , 8 ) == "subflow:" ) {
var subflowId = node . type . substring ( 8 ) ;
if ( ! exportedSubflows [ subflowId ] ) {
exportedSubflows [ subflowId ] = true ;
var subflow = getSubflow ( subflowId ) ;
var subflowSet = [ { n : subflow } ] ;
RED . nodes . eachNode ( function ( n ) {
if ( n . z == subflowId ) {
subflowSet . push ( { n : n } ) ;
}
} ) ;
var exportableSubflow = createExportableNodeSet ( subflowSet ) ;
nns = exportableSubflow . concat ( nns ) ;
}
}
if ( node . type != "subflow" ) {
var convertedNode = RED . nodes . convertNode ( node ) ;
for ( var d in node . _def . defaults ) {
if ( node . _def . defaults [ d ] . type && node [ d ] in configNodes ) {
var confNode = configNodes [ node [ d ] ] ;
var exportable = registry . getNodeType ( node . _def . defaults [ d ] . type ) . exportable ;
if ( ( exportable == null || exportable ) ) {
if ( ! ( node [ d ] in exportedConfigNodes ) ) {
exportedConfigNodes [ node [ d ] ] = true ;
nns . unshift ( RED . nodes . convertNode ( confNode ) ) ;
}
} else {
convertedNode [ d ] = "" ;
2013-09-05 15:02:48 +01:00
}
}
}
2014-02-24 23:35:11 +00:00
nns . push ( convertedNode ) ;
} else {
var convertedSubflow = convertSubflow ( node ) ;
nns . push ( convertedSubflow ) ;
2013-09-05 15:02:48 +01:00
}
}
return nns ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
//TODO: rename this (createCompleteNodeSet)
function createCompleteNodeSet ( ) {
var nns = [ ] ;
2014-08-08 00:01:35 +01:00
var i ;
for ( i in workspaces ) {
if ( workspaces . hasOwnProperty ( i ) ) {
2014-02-24 23:35:11 +00:00
if ( workspaces [ i ] . type == "tab" ) {
nns . push ( workspaces [ i ] ) ;
}
}
}
for ( i in subflows ) {
if ( subflows . hasOwnProperty ( i ) ) {
nns . push ( convertSubflow ( subflows [ i ] ) ) ;
2014-08-08 00:01:35 +01:00
}
2013-10-25 21:34:00 +01:00
}
2014-08-08 00:01:35 +01:00
for ( i in configNodes ) {
if ( configNodes . hasOwnProperty ( i ) ) {
nns . push ( convertNode ( configNodes [ i ] , true ) ) ;
}
2013-09-05 15:02:48 +01:00
}
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < nodes . length ; i ++ ) {
2013-09-05 15:02:48 +01:00
var node = nodes [ i ] ;
2014-06-24 13:13:19 +03:00
nns . push ( convertNode ( node , true ) ) ;
2013-09-05 15:02:48 +01:00
}
return nns ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function importNodes ( newNodesObj , createNewIds ) {
try {
2014-08-08 00:01:35 +01:00
var i ;
var n ;
2013-09-05 15:02:48 +01:00
var newNodes ;
if ( typeof newNodesObj === "string" ) {
2014-08-08 00:01:35 +01:00
if ( newNodesObj === "" ) {
2013-09-05 15:02:48 +01:00
return ;
}
newNodes = JSON . parse ( newNodesObj ) ;
} else {
newNodes = newNodesObj ;
}
if ( ! $ . isArray ( newNodes ) ) {
newNodes = [ newNodes ] ;
}
2013-12-12 15:51:15 +00:00
var unknownTypes = [ ] ;
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < newNodes . length ; i ++ ) {
n = newNodes [ i ] ;
2013-10-30 21:45:45 +00:00
// TODO: remove workspace in next release+1
2014-02-24 23:35:11 +00:00
if ( n . type != "workspace" &&
n . type != "tab" &&
n . type != "subflow" &&
! registry . getNodeType ( n . type ) &&
n . type . substring ( 0 , 8 ) != "subflow:" ) {
2013-09-05 15:02:48 +01:00
// TODO: get this UI thing out of here! (see below as well)
2014-10-30 21:41:42 +00:00
if ( unknownTypes . indexOf ( n . type ) == - 1 ) {
unknownTypes . push ( n . type ) ;
2014-04-28 21:40:32 +01:00
}
2014-10-30 21:41:42 +00:00
//if (n.x == null && n.y == null) {
// // config node - remove it
// newNodes.splice(i,1);
// i--;
//}
2013-12-12 15:51:15 +00:00
}
2013-09-05 15:02:48 +01:00
}
2013-12-12 15:51:15 +00:00
if ( unknownTypes . length > 0 ) {
var typeList = "<ul><li>" + unknownTypes . join ( "</li><li>" ) + "</li></ul>" ;
var type = "type" + ( unknownTypes . length > 1 ? "s" : "" ) ;
RED . notify ( "<strong>Imported unrecognised " + type + ":</strong>" + typeList , "error" , false , 10000 ) ;
//"DO NOT DEPLOY while in this state.<br/>Either, add missing types to Node-RED, restart and then reload page,<br/>or delete unknown "+n.name+", rewire as required, and then deploy.","error");
}
2014-02-24 23:35:11 +00:00
var activeWorkspace = RED . view . getWorkspace ( ) ;
var activeSubflow = getSubflow ( activeWorkspace ) ;
if ( activeSubflow ) {
for ( i = 0 ; i < newNodes . length ; i ++ ) {
var m = /^subflow:(.+)$/ . exec ( newNodes [ i ] . type ) ;
if ( m ) {
var subflowId = m [ 1 ] ;
var err ;
if ( subflowId === activeSubflow . id ) {
err = new Error ( "Cannot add subflow to itself" ) ;
}
if ( subflowContains ( m [ 1 ] , activeSubflow . id ) ) {
err = new Error ( "Cannot add subflow - circular reference detected" ) ;
}
if ( err ) {
// TODO: standardise error codes
err . code = "NODE_RED" ;
throw err ;
}
}
}
}
2014-09-08 10:53:18 +01:00
var new _workspaces = [ ] ;
var workspace _map = { } ;
2014-02-24 23:35:11 +00:00
var new _subflows = [ ] ;
var subflow _map = { } ;
var nid ;
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < newNodes . length ; i ++ ) {
n = newNodes [ i ] ;
2013-10-30 21:45:45 +00:00
// TODO: remove workspace in next release+1
if ( n . type === "workspace" || n . type === "tab" ) {
if ( n . type === "workspace" ) {
n . type = "tab" ;
}
2013-10-25 21:34:00 +01:00
if ( defaultWorkspace == null ) {
defaultWorkspace = n ;
}
2014-09-08 10:53:18 +01:00
if ( createNewIds ) {
2014-02-24 23:35:11 +00:00
nid = getID ( ) ;
2014-09-08 10:53:18 +01:00
workspace _map [ n . id ] = nid ;
n . id = nid ;
}
2013-10-25 21:34:00 +01:00
addWorkspace ( n ) ;
RED . view . addWorkspace ( n ) ;
2014-09-08 10:53:18 +01:00
new _workspaces . push ( n ) ;
2014-02-24 23:35:11 +00:00
} else if ( n . type === "subflow" ) {
subflow _map [ n . id ] = n ;
if ( createNewIds ) {
nid = getID ( ) ;
n . id = nid ;
}
// TODO: handle createNewIds - map old to new subflow ids
n . in . forEach ( function ( input , i ) {
input . type = "subflow" ;
input . direction = "in" ;
input . z = n . id ;
input . i = i ;
} ) ;
n . out . forEach ( function ( output , i ) {
output . type = "subflow" ;
output . direction = "out" ;
output . z = n . id ;
output . i = i ;
} ) ;
new _subflows . push ( n ) ;
addSubflow ( n ) ;
2013-10-25 21:34:00 +01:00
}
}
if ( defaultWorkspace == null ) {
2013-10-30 21:45:45 +00:00
defaultWorkspace = { type : "tab" , id : getID ( ) , label : "Sheet 1" } ;
2013-10-25 21:34:00 +01:00
addWorkspace ( defaultWorkspace ) ;
RED . view . addWorkspace ( defaultWorkspace ) ;
2014-09-08 10:53:18 +01:00
new _workspaces . push ( defaultWorkspace ) ;
2013-10-25 21:34:00 +01:00
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
var node _map = { } ;
var new _nodes = [ ] ;
var new _links = [ ] ;
2013-12-08 21:55:34 +00:00
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < newNodes . length ; i ++ ) {
n = newNodes [ i ] ;
2013-10-30 21:45:45 +00:00
// TODO: remove workspace in next release+1
2014-02-24 23:35:11 +00:00
if ( n . type !== "workspace" && n . type !== "tab" && n . type !== "subflow" ) {
2014-08-28 00:35:07 +01:00
var def = registry . getNodeType ( n . type ) ;
2013-10-25 21:34:00 +01:00
if ( def && def . category == "config" ) {
if ( ! RED . nodes . node ( n . id ) ) {
var configNode = { id : n . id , type : n . type , users : [ ] } ;
for ( var d in def . defaults ) {
2014-08-08 00:01:35 +01:00
if ( def . defaults . hasOwnProperty ( d ) ) {
configNode [ d ] = n [ d ] ;
}
2013-10-25 21:34:00 +01:00
}
configNode . label = def . label ;
configNode . _def = def ;
RED . nodes . add ( configNode ) ;
2013-09-05 15:02:48 +01:00
}
} else {
2013-11-15 23:40:36 +00:00
var node = { x : n . x , y : n . y , z : n . z , type : 0 , wires : n . wires , changed : false } ;
2013-10-25 21:34:00 +01:00
if ( createNewIds ) {
2014-02-24 23:35:11 +00:00
if ( subflow _map [ node . z ] ) {
node . z = subflow _map [ node . z ] . id ;
} else {
node . z = workspace _map [ node . z ] ;
if ( ! workspaces [ node . z ] ) {
node . z = activeWorkspace ;
}
2014-09-08 10:53:18 +01:00
}
2013-10-25 21:34:00 +01:00
node . id = getID ( ) ;
} else {
node . id = n . id ;
2014-02-24 23:35:11 +00:00
if ( node . z == null || ( ! workspaces [ node . z ] && ! subflow _map [ node . z ] ) ) {
node . z = activeWorkspace ;
2013-10-25 21:34:00 +01:00
}
2013-09-05 15:02:48 +01:00
}
2013-10-25 21:34:00 +01:00
node . type = n . type ;
node . _def = def ;
2014-02-24 23:35:11 +00:00
if ( n . type . substring ( 0 , 7 ) === "subflow" ) {
var parentId = n . type . split ( ":" ) [ 1 ] ;
var subflow = subflow _map [ parentId ] || getSubflow ( parentId ) ;
if ( createNewIds ) {
parentId = subflow . id ;
node . type = "subflow:" + parentId ;
node . _def = registry . getNodeType ( node . type ) ;
delete node . i ;
2013-10-25 21:34:00 +01:00
}
2014-10-31 22:44:58 +00:00
node . name = n . name ;
2014-02-24 23:35:11 +00:00
node . outputs = subflow . out . length ;
node . inputs = subflow . in . length ;
} else {
if ( ! node . _def ) {
2014-10-30 21:41:42 +00:00
if ( node . x && node . y ) {
node . _def = {
color : "#fee" ,
defaults : { } ,
label : "unknown: " + n . type ,
labelStyle : "node_label_italic" ,
outputs : n . outputs || n . wires . length
}
} else {
node . _def = {
category : "config"
} ;
node . users = [ ] ;
2014-02-24 23:35:11 +00:00
}
2014-10-30 21:41:42 +00:00
var orig = { } ;
for ( var p in n ) {
if ( n . hasOwnProperty ( p ) && p != "x" && p != "y" && p != "z" && p != "id" && p != "wires" ) {
orig [ p ] = n [ p ] ;
}
}
node . _orig = orig ;
node . name = n . type ;
node . type = "unknown" ;
2014-02-24 23:35:11 +00:00
}
2014-10-30 21:41:42 +00:00
if ( node . _def . category != "config" ) {
node . inputs = n . inputs || node . _def . inputs ;
node . outputs = n . outputs || node . _def . outputs ;
for ( var d2 in node . _def . defaults ) {
if ( node . _def . defaults . hasOwnProperty ( d2 ) ) {
node [ d2 ] = n [ d2 ] ;
}
2014-02-24 23:35:11 +00:00
}
2014-08-08 00:01:35 +01:00
}
2013-09-05 15:02:48 +01:00
}
2013-10-25 21:34:00 +01:00
addNode ( node ) ;
RED . editor . validateNode ( node ) ;
node _map [ n . id ] = node ;
2014-10-30 21:41:42 +00:00
if ( node . _def . category != "config" ) {
new _nodes . push ( node ) ;
}
2013-09-05 15:02:48 +01:00
}
}
}
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < new _nodes . length ; i ++ ) {
n = new _nodes [ i ] ;
for ( var w1 = 0 ; w1 < n . wires . length ; w1 ++ ) {
2013-09-05 15:02:48 +01:00
var wires = ( n . wires [ w1 ] instanceof Array ) ? n . wires [ w1 ] : [ n . wires [ w1 ] ] ;
2014-08-08 00:01:35 +01:00
for ( var w2 = 0 ; w2 < wires . length ; w2 ++ ) {
2013-09-05 15:02:48 +01:00
if ( wires [ w2 ] in node _map ) {
var link = { source : n , sourcePort : w1 , target : node _map [ wires [ w2 ] ] } ;
addLink ( link ) ;
new _links . push ( link ) ;
}
}
}
delete n . wires ;
}
2014-02-24 23:35:11 +00:00
for ( i = 0 ; i < new _subflows . length ; i ++ ) {
n = new _subflows [ i ] ;
n . in . forEach ( function ( input ) {
input . wires . forEach ( function ( wire ) {
var link = { source : input , sourcePort : 0 , target : node _map [ wire . id ] } ;
addLink ( link ) ;
new _links . push ( link ) ;
} ) ;
delete input . wires ;
} ) ;
n . out . forEach ( function ( output ) {
output . wires . forEach ( function ( wire ) {
var link ;
if ( wire . id == n . id ) {
link = { source : n . in [ wire . port ] , sourcePort : wire . port , target : output } ;
} else {
link = { source : node _map [ wire . id ] , sourcePort : wire . port , target : output } ;
}
addLink ( link ) ;
new _links . push ( link ) ;
} ) ;
delete output . wires ;
} ) ;
}
return [ new _nodes , new _links , new _workspaces , new _subflows ] ;
2013-09-05 15:02:48 +01:00
} catch ( error ) {
2014-02-24 23:35:11 +00:00
if ( error . code != "NODE_RED" ) {
console . log ( error . stack ) ;
RED . notify ( "<strong>Error</strong>: " + error , "error" ) ;
} else {
RED . notify ( "<strong>Error</strong>: " + error . message , "error" ) ;
}
2013-09-05 15:02:48 +01:00
return null ;
}
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
return {
2014-08-28 00:35:07 +01:00
registry : registry ,
setNodeList : registry . setNodeList ,
getNodeSet : registry . getNodeSet ,
addNodeSet : registry . addNodeSet ,
removeNodeSet : registry . removeNodeSet ,
enableNodeSet : registry . enableNodeSet ,
disableNodeSet : registry . disableNodeSet ,
registerType : registry . registerNodeType ,
getType : registry . getNodeType ,
2013-09-05 15:02:48 +01:00
convertNode : convertNode ,
2014-02-24 23:35:11 +00:00
2013-09-05 15:02:48 +01:00
add : addNode ,
remove : removeNode ,
2014-02-24 23:35:11 +00:00
addLink : addLink ,
2013-09-05 15:02:48 +01:00
removeLink : removeLink ,
2014-02-24 23:35:11 +00:00
2013-10-25 21:34:00 +01:00
addWorkspace : addWorkspace ,
2013-10-26 22:29:24 +01:00
removeWorkspace : removeWorkspace ,
workspace : getWorkspace ,
2014-02-24 23:35:11 +00:00
addSubflow : addSubflow ,
removeSubflow : removeSubflow ,
subflow : getSubflow ,
subflowContains : subflowContains ,
2013-09-05 15:02:48 +01:00
eachNode : function ( cb ) {
2014-08-08 00:01:35 +01:00
for ( var n = 0 ; n < nodes . length ; n ++ ) {
2013-09-05 15:02:48 +01:00
cb ( nodes [ n ] ) ;
}
} ,
eachLink : function ( cb ) {
2014-08-08 00:01:35 +01:00
for ( var l = 0 ; l < links . length ; l ++ ) {
2013-09-05 15:02:48 +01:00
cb ( links [ l ] ) ;
}
} ,
eachConfig : function ( cb ) {
for ( var id in configNodes ) {
2014-08-08 00:01:35 +01:00
if ( configNodes . hasOwnProperty ( id ) ) {
cb ( configNodes [ id ] ) ;
}
2013-09-05 15:02:48 +01:00
}
} ,
2014-02-24 23:35:11 +00:00
eachSubflow : function ( cb ) {
for ( var id in subflows ) {
if ( subflows . hasOwnProperty ( id ) ) {
cb ( subflows [ id ] ) ;
}
}
} ,
2013-09-05 15:02:48 +01:00
node : getNode ,
import : importNodes ,
refreshValidation : refreshValidation ,
getAllFlowNodes : getAllFlowNodes ,
createExportableNodeSet : createExportableNodeSet ,
createCompleteNodeSet : createCompleteNodeSet ,
2013-10-25 21:34:00 +01:00
id : getID ,
2013-09-05 15:02:48 +01:00
nodes : nodes , // TODO: exposed for d3 vis
links : links // TODO: exposed for d3 vis
} ;
2014-08-08 00:01:35 +01:00
} ) ( ) ;