Compare commits

...

5 Commits

Author SHA1 Message Date
Nick O'Leary
2350540a98 Introduce RED.utils.createSVGElement 2022-08-07 22:28:08 +01:00
Nick O'Leary
c021c4020e Add RED.utils.domSelection module 2022-08-07 22:27:29 +01:00
Nick O'Leary
11a6925ae4 Remove almost all d3 from palette.js 2022-08-05 23:08:09 +01:00
Nick O'Leary
24ff5262d9 Remove d3 from radialMenu 2022-08-05 23:07:39 +01:00
Nick O'Leary
c24d7fafa9 Remove d3 from keyboard.js 2022-08-05 21:19:45 +01:00
9 changed files with 384 additions and 317 deletions

View File

@@ -152,6 +152,7 @@ module.exports = function(grunt) {
"packages/node_modules/@node-red/editor-client/src/js/history.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/validators.js",
"packages/node_modules/@node-red/editor-client/src/js/ui/utils.js", "packages/node_modules/@node-red/editor-client/src/js/ui/utils.js",
"packages/node_modules/@node-red/editor-client/src/js/ui/utils-domselection.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/editableList.js",
"packages/node_modules/@node-red/editor-client/src/js/ui/common/treeList.js", "packages/node_modules/@node-red/editor-client/src/js/ui/common/treeList.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/checkboxSet.js",

View File

@@ -298,21 +298,21 @@ RED.keyboard = (function() {
return resolveKeyEvent(evt); return resolveKeyEvent(evt);
} }
} }
d3.select(window).on("keydown",function() { document.addEventListener("keydown", function(event) {
if (!handlersActive) { if (!handlersActive) {
return; return;
} }
if (metaKeyCodes[d3.event.keyCode]) { if (metaKeyCodes[event.keyCode]) {
return; return;
} }
var handler = resolveKeyEvent(d3.event); var handler = resolveKeyEvent(event);
if (handler && handler.ondown) { if (handler && handler.ondown) {
if (typeof handler.ondown === "string") { if (typeof handler.ondown === "string") {
RED.actions.invoke(handler.ondown); RED.actions.invoke(handler.ondown);
} else { } else {
handler.ondown(); handler.ondown();
} }
d3.event.preventDefault(); event.preventDefault();
} }
}); });

View File

@@ -299,7 +299,7 @@ RED.palette = (function() {
RED.view.focus(); RED.view.focus();
}, },
stop: function() { stop: function() {
d3.select('.red-ui-flow-link-splice').classed('red-ui-flow-link-splice',false); document.querySelectorAll(".red-ui-flow-link-splice").forEach(n => { n.classList.remove("red-ui-flow-link-splice") })
if (hoverGroup) { if (hoverGroup) {
document.getElementById("group_select_"+hoverGroup.id).classList.remove("red-ui-flow-group-hovered"); document.getElementById("group_select_"+hoverGroup.id).classList.remove("red-ui-flow-group-hovered");
} }
@@ -358,26 +358,26 @@ RED.palette = (function() {
var mx = mouseX / RED.view.scale(); var mx = mouseX / RED.view.scale();
var my = mouseY / RED.view.scale(); var my = mouseY / RED.view.scale();
for (var i=0;i<nodes.length;i++) { for (var i=0;i<nodes.length;i++) {
var node = d3.select(nodes[i]); var node = nodes[i];
if (node.classed('red-ui-flow-link-background') && !node.classed('red-ui-flow-link-link')) { if (node.classList.contains('red-ui-flow-link-background') && !node.classList.contains('red-ui-flow-link-link')) {
var length = nodes[i].getTotalLength(); var length = node.getTotalLength();
for (var j=0;j<length;j+=10) { for (var j=0;j<length;j+=10) {
var p = nodes[i].getPointAtLength(j); var p = node.getPointAtLength(j);
var d2 = ((p.x-mx)*(p.x-mx))+((p.y-my)*(p.y-my)); var d2 = ((p.x-mx)*(p.x-mx))+((p.y-my)*(p.y-my));
if (d2 < 200 && d2 < bestDistance) { if (d2 < 200 && d2 < bestDistance) {
bestDistance = d2; bestDistance = d2;
bestLink = nodes[i]; bestLink = node;
} }
} }
} }
} }
if (activeSpliceLink && activeSpliceLink !== bestLink) { if (activeSpliceLink && activeSpliceLink !== bestLink) {
d3.select(activeSpliceLink.parentNode).classed('red-ui-flow-link-splice',false); activeSpliceLink.parentNode.classList.remove('red-ui-flow-link-splice');
} }
if (bestLink) { if (bestLink) {
d3.select(bestLink.parentNode).classed('red-ui-flow-link-splice',true) bestLink.parentNode.classList.add('red-ui-flow-link-splice');
} else { } else {
d3.select('.red-ui-flow-link-splice').classed('red-ui-flow-link-splice',false); document.querySelectorAll(".red-ui-flow-link-splice").forEach(n => { n.classList.remove("red-ui-flow-link-splice") })
} }
if (activeSpliceLink !== bestLink) { if (activeSpliceLink !== bestLink) {
if (bestLink) { if (bestLink) {

View File

@@ -27,26 +27,24 @@ RED.touch.radialMenu = (function() {
function createRadial(obj,pos,options) { function createRadial(obj,pos,options) {
isActive = true; isActive = true;
try { try {
touchMenu = d3.select("body").append("div").classed("red-ui-editor-radial-menu",true) touchMenu = $('<div>', {class: 'red-ui-editor-radial-menu'}).appendTo('body')
.on('touchstart',function() { .on('touchstart',function(evt) {
hide(); hide();
d3.event.preventDefault(); evt.preventDefault();
}); });
var menu = touchMenu.append("div") var menu = $('<div>').appendTo(touchMenu).css({
.style({
top: (pos[1]-80)+"px", top: (pos[1]-80)+"px",
left:(pos[0]-80)+"px", left:(pos[0]-80)+"px",
}); });
var menuOpts = []; var menuOpts = [];
var createMenuOpt = function(x,y,opt) { var createMenuOpt = function(x,y,opt) {
opt.el = menu.append("div").classed("red-ui-editor-radial-menu-opt",true) opt.el = $('<div>', {class: 'red-ui-editor-radial-menu-opt'}).appendTo(menu)
.style({ .css({
top: (y+80-25)+"px", top: (y+80-25)+"px",
left:(x+80-25)+"px" left:(x+80-25)+"px"
}) })
.classed("red-ui-editor-radial-menu-opt-disabled",!!opt.disabled) .toggleClass("red-ui-editor-radial-menu-opt-disabled",!!opt.disabled)
opt.el.html(opt.name); opt.el.html(opt.name);
@@ -54,16 +52,16 @@ RED.touch.radialMenu = (function() {
opt.y = y; opt.y = y;
menuOpts.push(opt); menuOpts.push(opt);
opt.el.on('touchstart',function() { opt.el.on('touchstart',function(evt) {
opt.el.classed("red-ui-editor-radial-menu-opt-active",true) opt.el.toggleClass("red-ui-editor-radial-menu-opt-active",true)
d3.event.preventDefault(); evt.preventDefault();
d3.event.stopPropagation(); evt.stopPropagation();
}); });
opt.el.on('touchend',function() { opt.el.on('touchend',function(evt) {
hide(); hide();
opt.onselect(); opt.onselect();
d3.event.preventDefault(); evt.preventDefault();
d3.event.stopPropagation(); evt.stopPropagation();
}); });
} }
@@ -88,8 +86,8 @@ RED.touch.radialMenu = (function() {
} }
obj.on('touchend.radial',function() { obj.on('touchend.radial',function() {
obj.on('touchend.radial',null); obj.off('touchend.radial');
obj.on('touchmenu.radial',null); obj.off('touchmenu.radial');
if (activeOption) { if (activeOption) {
try { try {
@@ -103,9 +101,9 @@ RED.touch.radialMenu = (function() {
} }
}); });
obj.on('touchmove.radial',function() { obj.on('touchmove.radial',function(evt) {
try { try {
var touch0 = d3.event.touches.item(0); var touch0 = evt.touches.item(0);
var p = [touch0.pageX - pos[0],touch0.pageY-pos[1]]; var p = [touch0.pageX - pos[0],touch0.pageY-pos[1]];
for (var i=0;i<menuOpts.length;i++) { for (var i=0;i<menuOpts.length;i++) {
var opt = menuOpts[i]; var opt = menuOpts[i];
@@ -119,7 +117,7 @@ RED.touch.radialMenu = (function() {
if (opt === activeOption) { if (opt === activeOption) {
activeOption = null; activeOption = null;
} }
opt.el.classed("selected",false); opt.el.toggleClass("selected",false);
} }
} }
} }

View File

@@ -0,0 +1,61 @@
/**
* Modelled after `d3.selection` this provides a way to keep a mapping of
* DOM Nodes with a data collection.
*
* The goal here is not to reproduce d3 functionality as-is, but to provide an
* api that is specific to what we need to do
*
* const domSelection = RED.utils.domSelection(container, selector, createNode, eachNode)
*
* - container - a DOM Node that is the container of the DOM Nodes to track
* - selector - CSS selector to get the DOM nodes to track
* - createNode - function called when a DOM node must be created for a piece of data.
* `this` is the data item. Should return the DOM Node. It will
* get added to the container.
* - eachNode - function called for each DOM node/data item in the selection
*
* DomSelection.refresh(data) - called whenever the selection should be refreshed.
* Data is expected to be an array of objects that contain an 'id' property
*
*/
RED.utils.domSelection = (function() {
class DomSelection {
constructor(container, selector, createNode, eachNode) {
this.container = container
this.selector = selector
this.createNode = createNode
this.eachNode = eachNode
this.data = []
}
refresh(data) {
const domNodes = this.container.querySelectorAll(this.selector)
const domItems = new Map()
const dataLength = data.length
const domLength = domNodes.length
for (let i = 0; i < domLength; i++) {
const domNode = domNodes[i]
if (domNode.__data__) {
domItems.set(domNode.__data__.id, domNode)
}
}
for (let i = 0; i < dataLength; i++) {
const datum = data[i]
let domNode = domItems.get(datum.id)
if (!domNode) {
domNode = this.createNode.call(datum)
this.container.appendChild(domNode)
domNode.__data__ = datum
} else {
domItems.delete(datum.id)
}
this.eachNode.call(datum, domNode)
}
for (const remainingDomNodes of domItems) {
remainingDomNodes[1].remove()
}
}
}
return (container, selector, createNode, eachNode) => new DomSelection(container, selector, createNode, eachNode)
})()

View File

@@ -1397,6 +1397,17 @@ RED.utils = (function() {
return r; return r;
} }
function createSVGElement(tag, attrs = {}, parent) {
const element = document.createElementNS('http://www.w3.org/2000/svg', tag)
for (const k in attrs) {
element.setAttribute(k, attrs[k])
}
if (parent) {
parent.appendChild(element)
}
return element
}
return { return {
createObjectElement: createObjectElement, createObjectElement: createObjectElement,
getMessageProperty: getMessageProperty, getMessageProperty: getMessageProperty,
@@ -1420,6 +1431,7 @@ RED.utils = (function() {
getDarkerColor: getDarkerColor, getDarkerColor: getDarkerColor,
parseModuleList: parseModuleList, parseModuleList: parseModuleList,
checkModuleAllowed: checkModuleAllowed, checkModuleAllowed: checkModuleAllowed,
getBrowserInfo: getBrowserInfo getBrowserInfo: getBrowserInfo,
createSVGElement: createSVGElement
} }
})(); })();

View File

@@ -91,8 +91,7 @@ RED.view.annotations = (function() {
function addAnnotation(id,evt) { function addAnnotation(id,evt) {
var opts = annotations[id]; var opts = annotations[id];
evt.el.__annotations__ = evt.el.__annotations__ || []; evt.el.__annotations__ = evt.el.__annotations__ || [];
var annotationGroup = document.createElementNS("http://www.w3.org/2000/svg","g"); var annotationGroup = RED.utils.createSVGElement("g", { class: opts.class || '' })
annotationGroup.setAttribute("class",opts.class || "");
evt.el.__annotations__.push({ evt.el.__annotations__.push({
id:id, id:id,
element: annotationGroup element: annotationGroup

View File

@@ -16,7 +16,6 @@
RED.view.navigator = (function() { RED.view.navigator = (function() {
var nav_scale = 25; var nav_scale = 25;
var nav_width = 5000/nav_scale; var nav_width = 5000/nav_scale;
var nav_height = 5000/nav_scale; var nav_height = 5000/nav_scale;
@@ -24,7 +23,6 @@
var navContainer; var navContainer;
var navBox; var navBox;
var navBorder; var navBorder;
var navVis;
var scrollPos; var scrollPos;
var scaleFactor; var scaleFactor;
var chartSize; var chartSize;
@@ -32,23 +30,15 @@
var isDragging; var isDragging;
var isShowing = false; var isShowing = false;
var domSelection
function refreshNodes() { function refreshNodes() {
if (!isShowing) { if (!isShowing) {
return; return;
} }
var navNode = navVis.selectAll(".red-ui-navigator-node").data(RED.view.getActiveNodes(),function(d){return d.id}); domSelection.refresh(RED.view.getActiveNodes())
navNode.exit().remove();
navNode.enter().insert("rect")
.attr('class','red-ui-navigator-node')
.attr("pointer-events", "none");
navNode.each(function(d) {
d3.select(this).attr("x",function(d) { return (d.x-d.w/2)/nav_scale })
.attr("y",function(d) { return (d.y-d.h/2)/nav_scale })
.attr("width",function(d) { return Math.max(9,d.w/nav_scale) })
.attr("height",function(d) { return Math.max(3,d.h/nav_scale) })
.attr("fill",function(d) { return RED.utils.getNodeColor(d.type,d._def);})
});
} }
function onScroll() { function onScroll() {
if (!isDragging) { if (!isDragging) {
resizeNavBorder(); resizeNavBorder();
@@ -59,10 +49,11 @@
scaleFactor = RED.view.scale(); scaleFactor = RED.view.scale();
chartSize = [ $("#red-ui-workspace-chart").width(), $("#red-ui-workspace-chart").height()]; chartSize = [ $("#red-ui-workspace-chart").width(), $("#red-ui-workspace-chart").height()];
scrollPos = [$("#red-ui-workspace-chart").scrollLeft(),$("#red-ui-workspace-chart").scrollTop()]; scrollPos = [$("#red-ui-workspace-chart").scrollLeft(),$("#red-ui-workspace-chart").scrollTop()];
navBorder.attr('x', scrollPos[0]/nav_scale) navBorder.attr('x', scrollPos[0]/nav_scale)
.attr('y',scrollPos[1]/nav_scale) navBorder.attr('y', scrollPos[1]/nav_scale)
.attr('width',chartSize[0]/nav_scale/scaleFactor) navBorder.attr('width',chartSize[0]/nav_scale/scaleFactor)
.attr('height',chartSize[1]/nav_scale/scaleFactor) navBorder.attr('height',chartSize[1]/nav_scale/scaleFactor)
} }
} }
function toggle() { function toggle() {
@@ -87,7 +78,6 @@
$(window).on("resize", resizeNavBorder); $(window).on("resize", resizeNavBorder);
RED.events.on("sidebar:resize",resizeNavBorder); RED.events.on("sidebar:resize",resizeNavBorder);
RED.actions.add("core:toggle-navigator",toggle); RED.actions.add("core:toggle-navigator",toggle);
var hideTimeout;
navContainer = $('<div>').css({ navContainer = $('<div>').css({
"position":"absolute", "position":"absolute",
@@ -96,47 +86,52 @@
zIndex: 1 zIndex: 1
}).appendTo("#red-ui-workspace").hide(); }).appendTo("#red-ui-workspace").hide();
navBox = d3.select(navContainer[0]) navBox = $(RED.utils.createSVGElement('svg', {id: "red-ui-navigator-canvas", width: nav_width, height: nav_height, 'pointer-events': 'all'})).appendTo(navContainer)
.append("svg:svg")
.attr("width", nav_width)
.attr("height", nav_height)
.attr("pointer-events", "all")
.attr("id","red-ui-navigator-canvas")
navBox.append("rect").attr("x",0).attr("y",0).attr("width",nav_width).attr("height",nav_height).style({ const navBoxBody = $(RED.utils.createSVGElement("rect", { x: 0, y: 0, width: nav_width, height: nav_height, fill: 'none', stroke: 'none', 'pointer-events': 'all'}))
fill:"none", .css({'cursor': 'pointer'})
stroke:"none", navBoxBody.on('mousedown',function(event) {
pointerEvents:"all"
}).on("mousedown", function() {
// Update these in case they have changed // Update these in case they have changed
scaleFactor = RED.view.scale(); scaleFactor = RED.view.scale();
chartSize = [ $("#red-ui-workspace-chart").width(), $("#red-ui-workspace-chart").height()]; chartSize = [ $("#red-ui-workspace-chart").width(), $("#red-ui-workspace-chart").height()];
dimensions = [chartSize[0]/nav_scale/scaleFactor, chartSize[1]/nav_scale/scaleFactor]; dimensions = [chartSize[0]/nav_scale/scaleFactor, chartSize[1]/nav_scale/scaleFactor];
var newX = Math.max(0,Math.min(d3.event.offsetX+dimensions[0]/2,nav_width)-dimensions[0]); var newX = Math.max(0,Math.min(event.offsetX+dimensions[0]/2,nav_width)-dimensions[0]);
var newY = Math.max(0,Math.min(d3.event.offsetY+dimensions[1]/2,nav_height)-dimensions[1]); var newY = Math.max(0,Math.min(event.offsetY+dimensions[1]/2,nav_height)-dimensions[1]);
navBorder.attr('x',newX).attr('y',newY); navBorder.attr('x',newX)
navBorder.attr('y',newY);
isDragging = true; isDragging = true;
$("#red-ui-workspace-chart").scrollLeft(newX*nav_scale*scaleFactor); $("#red-ui-workspace-chart").scrollLeft(newX*nav_scale*scaleFactor);
$("#red-ui-workspace-chart").scrollTop(newY*nav_scale*scaleFactor); $("#red-ui-workspace-chart").scrollTop(newY*nav_scale*scaleFactor);
}).on("mousemove", function() { }).on('mousemove', function(event) {
if (!isDragging) { return } if (!isDragging) { return }
if (d3.event.buttons === 0) { if (event.buttons === 0) {
isDragging = false; isDragging = false;
return; return;
} }
var newX = Math.max(0,Math.min(d3.event.offsetX+dimensions[0]/2,nav_width)-dimensions[0]); var newX = Math.max(0,Math.min(event.offsetX+dimensions[0]/2,nav_width)-dimensions[0]);
var newY = Math.max(0,Math.min(d3.event.offsetY+dimensions[1]/2,nav_height)-dimensions[1]); var newY = Math.max(0,Math.min(event.offsetY+dimensions[1]/2,nav_height)-dimensions[1]);
navBorder.attr('x',newX).attr('y',newY); navBorder.attr('x',newX)
navBorder.attr('y',newY);
$("#red-ui-workspace-chart").scrollLeft(newX*nav_scale*scaleFactor); $("#red-ui-workspace-chart").scrollLeft(newX*nav_scale*scaleFactor);
$("#red-ui-workspace-chart").scrollTop(newY*nav_scale*scaleFactor); $("#red-ui-workspace-chart").scrollTop(newY*nav_scale*scaleFactor);
}).on("mouseup", function() { }).on('mouseup', function() {
isDragging = false; isDragging = false;
}).appendTo(navBox)
navBorder = $(RED.utils.createSVGElement("rect", { "class": "red-ui-navigator-border" })).appendTo(navBox)
const navVis = $(RED.utils.createSVGElement('g')).appendTo(navBox)
domSelection = RED.utils.domSelection(navVis[0], '.red-ui-navigator-node', function() {
return RED.utils.createSVGElement('rect', { class: 'red-ui-navigator-node', 'pointer-events': 'none' })
}, function(node) {
node.setAttribute('x', (this.x-this.w/2)/nav_scale)
node.setAttribute('y', (this.y-this.h/2)/nav_scale)
node.setAttribute('width', Math.max(9,this.w/nav_scale))
node.setAttribute('height', Math.max(3,this.h/nav_scale))
node.setAttribute('fill', RED.utils.getNodeColor(this.type,this._def))
}) })
navBorder = navBox.append("rect").attr("class","red-ui-navigator-border")
navVis = navBox.append("svg:g")
RED.statusBar.add({ RED.statusBar.add({
id: "view-navigator", id: "view-navigator",
align: "right", align: "right",

View File

@@ -294,25 +294,13 @@ RED.view = (function() {
] ]
startTouchDistance = Math.sqrt((a*a)+(b*b)); startTouchDistance = Math.sqrt((a*a)+(b*b));
} else { } else {
var obj = d3.select(document.body);
touch0 = d3.event.touches.item(0); touch0 = d3.event.touches.item(0);
var pos = [touch0.pageX,touch0.pageY]; var pos = [touch0.pageX,touch0.pageY];
startTouchCenter = [touch0.pageX,touch0.pageY]; startTouchCenter = [touch0.pageX,touch0.pageY];
startTouchDistance = 0; startTouchDistance = 0;
var point = d3.touches(this)[0];
touchStartTime = setTimeout(function() { touchStartTime = setTimeout(function() {
touchStartTime = null; touchStartTime = null;
showTouchMenu(obj,pos); showTouchMenu(document.body,pos);
//lasso = eventLayer.append("rect")
// .attr("ox",point[0])
// .attr("oy",point[1])
// .attr("rx",2)
// .attr("ry",2)
// .attr("x",point[0])
// .attr("y",point[1])
// .attr("width",0)
// .attr("height",0)
// .attr("class","nr-ui-view-lasso");
},touchLongPressTimeout); },touchLongPressTimeout);
} }
d3.event.preventDefault(); d3.event.preventDefault();
@@ -712,11 +700,11 @@ RED.view = (function() {
type: "badge", type: "badge",
class: "red-ui-flow-node-changed", class: "red-ui-flow-node-changed",
element: function() { element: function() {
var changeBadge = document.createElementNS("http://www.w3.org/2000/svg","circle"); return RED.utils.createSVGElement("circle", {
changeBadge.setAttribute("cx",5); cx: 5,
changeBadge.setAttribute("cy",5); cy: 5,
changeBadge.setAttribute("r",5); r: 5
return changeBadge; })
}, },
show: function(n) { return n.changed||n.moved } show: function(n) { return n.changed||n.moved }
}) })
@@ -725,9 +713,9 @@ RED.view = (function() {
type: "badge", type: "badge",
class: "red-ui-flow-node-error", class: "red-ui-flow-node-error",
element: function(d) { element: function(d) {
var errorBadge = document.createElementNS("http://www.w3.org/2000/svg","path"); return RED.utils.createSVGElement("path", {
errorBadge.setAttribute("d","M 0,9 l 10,0 -5,-8 z"); d: "M 0,9 l 10,0 -5,-8 z"
return errorBadge })
}, },
tooltip: function(d) { tooltip: function(d) {
if (d.validationErrors && d.validationErrors.length > 0) { if (d.validationErrors && d.validationErrors.length > 0) {
@@ -3703,13 +3691,12 @@ RED.view = (function() {
} }
function nodeTouchStart(d) { function nodeTouchStart(d) {
if (RED.view.DEBUG) { console.warn("nodeTouchStart", mouse_mode,d); } if (RED.view.DEBUG) { console.warn("nodeTouchStart", mouse_mode,d); }
var obj = d3.select(this);
var touch0 = d3.event.touches.item(0); var touch0 = d3.event.touches.item(0);
var pos = [touch0.pageX,touch0.pageY]; var pos = [touch0.pageX,touch0.pageY];
startTouchCenter = [touch0.pageX,touch0.pageY]; startTouchCenter = [touch0.pageX,touch0.pageY];
startTouchDistance = 0; startTouchDistance = 0;
touchStartTime = setTimeout(function() { touchStartTime = setTimeout(function() {
showTouchMenu(obj,pos); showTouchMenu(this,pos);
},touchLongPressTimeout); },touchLongPressTimeout);
nodeMouseDown.call(this,d) nodeMouseDown.call(this,d)
d3.event.preventDefault(); d3.event.preventDefault();
@@ -3868,12 +3855,11 @@ RED.view = (function() {
focusView(); focusView();
d3.event.stopPropagation(); d3.event.stopPropagation();
var obj = d3.select(document.body);
var touch0 = d3.event.touches.item(0); var touch0 = d3.event.touches.item(0);
var pos = [touch0.pageX,touch0.pageY]; var pos = [touch0.pageX,touch0.pageY];
touchStartTime = setTimeout(function() { touchStartTime = setTimeout(function() {
touchStartTime = null; touchStartTime = null;
showTouchMenu(obj,pos); showTouchMenu(document.body,pos);
},touchLongPressTimeout); },touchLongPressTimeout);
d3.event.preventDefault(); d3.event.preventDefault();
} }
@@ -4247,13 +4233,14 @@ RED.view = (function() {
d.resize = true; d.resize = true;
d.dirty = true; d.dirty = true;
var mainRect = document.createElementNS("http://www.w3.org/2000/svg","rect"); var mainRect = RED.utils.createSVGElement("rect", {
class: "red-ui-flow-subflow-port",
rx: 8,
ry: 8,
width: 40,
height: 40,
})
mainRect.__data__ = d; mainRect.__data__ = d;
mainRect.setAttribute("class", "red-ui-flow-subflow-port");
mainRect.setAttribute("rx", 8);
mainRect.setAttribute("ry", 8);
mainRect.setAttribute("width", 40);
mainRect.setAttribute("height", 40);
node[0][0].__mainRect__ = mainRect; node[0][0].__mainRect__ = mainRect;
d3.select(mainRect) d3.select(mainRect)
.on("mouseup",nodeMouseUp) .on("mouseup",nodeMouseUp)
@@ -4262,50 +4249,50 @@ RED.view = (function() {
.on("touchend",nodeTouchEnd) .on("touchend",nodeTouchEnd)
nodeContents.appendChild(mainRect); nodeContents.appendChild(mainRect);
var output_groupEl = document.createElementNS("http://www.w3.org/2000/svg","g"); var output_groupEl = RED.utils.createSVGElement("g", { x: 0, y: 0 })
output_groupEl.setAttribute("x",0);
output_groupEl.setAttribute("y",0);
node[0][0].__outputLabelGroup__ = output_groupEl; node[0][0].__outputLabelGroup__ = output_groupEl;
var output_output = document.createElementNS("http://www.w3.org/2000/svg","text"); var output_output = RED.utils.createSVGElement("text", { class: "red-ui-flow-port-label" })
output_output.setAttribute("class","red-ui-flow-port-label");
output_output.style["font-size"] = "10px"; output_output.style["font-size"] = "10px";
output_output.textContent = "output"; output_output.textContent = "output";
output_groupEl.appendChild(output_output); output_groupEl.appendChild(output_output);
node[0][0].__outputOutput__ = output_output; node[0][0].__outputOutput__ = output_output;
var output_number = document.createElementNS("http://www.w3.org/2000/svg","text"); var output_number = RED.utils.createSVGElement("text", {
output_number.setAttribute("class","red-ui-flow-port-label red-ui-flow-port-index"); class: "red-ui-flow-port-label red-ui-flow-port-index",
output_number.setAttribute("x",0); x: 0,
output_number.setAttribute("y",0); y: 0
})
output_number.textContent = d.i+1; output_number.textContent = d.i+1;
output_groupEl.appendChild(output_number); output_groupEl.appendChild(output_number);
node[0][0].__outputNumber__ = output_number; node[0][0].__outputNumber__ = output_number;
var output_border = document.createElementNS("http://www.w3.org/2000/svg","path"); var output_border = RED.utils.createSVGElement("path", {
output_border.setAttribute("d","M 40 1 l 0 38") d: "M 40 1 l 0 38",
output_border.setAttribute("class", "red-ui-flow-node-icon-shade-border") class: "red-ui-flow-node-icon-shade-border"
})
output_groupEl.appendChild(output_border); output_groupEl.appendChild(output_border);
node[0][0].__outputBorder__ = output_border; node[0][0].__outputBorder__ = output_border;
nodeContents.appendChild(output_groupEl); nodeContents.appendChild(output_groupEl);
var text = document.createElementNS("http://www.w3.org/2000/svg","g"); var text = RED.utils.createSVGElement("g", {
text.setAttribute("class","red-ui-flow-port-label"); class: "red-ui-flow-port-label",
text.setAttribute("transform","translate(38,0)"); transform: "translate(38,0)",
text.setAttribute('style', 'fill : #888'); // hard coded here! style: 'fill : #888' // hard coded here!
})
node[0][0].__textGroup__ = text; node[0][0].__textGroup__ = text;
nodeContents.append(text); nodeContents.append(text);
var portEl = document.createElementNS("http://www.w3.org/2000/svg","g"); var portEl = RED.utils.createSVGElement("g", { transform: 'translate(-5,15)'})
portEl.setAttribute('transform','translate(-5,15)')
var port = document.createElementNS("http://www.w3.org/2000/svg","rect"); var port = RED.utils.createSVGElement("rect", {
port.setAttribute("class","red-ui-flow-port"); class: "red-ui-flow-port",
port.setAttribute("rx",3); rx: 3,
port.setAttribute("ry",3); ry: 3,
port.setAttribute("width",10); width: 10,
port.setAttribute("height",10); height: 10
})
portEl.appendChild(port); portEl.appendChild(port);
port.__data__ = d; port.__data__ = d;
@@ -4434,10 +4421,11 @@ RED.view = (function() {
} }
for (var i=0; i<sn; i++) { for (var i=0; i<sn; i++) {
if (i===textLines.length) { if (i===textLines.length) {
var line = document.createElementNS("http://www.w3.org/2000/svg","text"); var line = RED.utils.createSVGElement("text", {
line.setAttribute("class","red-ui-flow-node-label-text"); class: "red-ui-flow-node-label-text",
line.setAttribute("x",0); x: 0,
line.setAttribute("y",i*24); y: i*24
});
this.__textGroup__.appendChild(line); this.__textGroup__.appendChild(line);
} }
textLines[i].textContent = sa[i]; textLines[i].textContent = sa[i];
@@ -4507,32 +4495,35 @@ RED.view = (function() {
d.resize = true; d.resize = true;
if (d._def.button) { if (d._def.button) {
var buttonGroup = document.createElementNS("http://www.w3.org/2000/svg","g"); var buttonGroup = RED.utils.createSVGElement("g", {
class: "red-ui-flow-node-button",
transform: "translate("+((d._def.align == "right") ? 94 : -25)+",2)"
})
buttonGroup.__data__ = d; buttonGroup.__data__ = d;
buttonGroup.setAttribute("transform", "translate("+((d._def.align == "right") ? 94 : -25)+",2)");
buttonGroup.setAttribute("class","red-ui-flow-node-button");
node[0][0].__buttonGroup__ = buttonGroup; node[0][0].__buttonGroup__ = buttonGroup;
var bgBackground = document.createElementNS("http://www.w3.org/2000/svg","rect"); var bgBackground = RED.utils.createSVGElement("rect", {
class: "red-ui-flow-node-button-background",
rx: 5,
ry: 5,
width: 32,
height: node_height-4
})
bgBackground.__data__ = d; bgBackground.__data__ = d;
bgBackground.setAttribute("class","red-ui-flow-node-button-background");
bgBackground.setAttribute("rx",5);
bgBackground.setAttribute("ry",5);
bgBackground.setAttribute("width",32);
bgBackground.setAttribute("height",node_height-4);
buttonGroup.appendChild(bgBackground); buttonGroup.appendChild(bgBackground);
node[0][0].__buttonGroupBackground__ = bgBackground; node[0][0].__buttonGroupBackground__ = bgBackground;
var bgButton = document.createElementNS("http://www.w3.org/2000/svg","rect"); var bgButton = RED.utils.createSVGElement("rect", {
class: "red-ui-flow-node-button-button",
x: d._def.align == "right"? 11:5,
y: 4,
rx: 4,
ry: 4,
width: 16,
height: node_height-12,
fill: RED.utils.getNodeColor(d.type,d._def)
})
bgButton.__data__ = d; bgButton.__data__ = d;
bgButton.setAttribute("class","red-ui-flow-node-button-button");
bgButton.setAttribute("x", d._def.align == "right"? 11:5);
bgButton.setAttribute("y",4);
bgButton.setAttribute("rx",4);
bgButton.setAttribute("ry",4);
bgButton.setAttribute("width",16);
bgButton.setAttribute("height",node_height-12);
bgButton.setAttribute("fill", RED.utils.getNodeColor(d.type,d._def));
d3.select(bgButton) d3.select(bgButton)
.on("mousedown",function(d) {if (!lasso && isButtonEnabled(d)) {focusView();d3.select(this).attr("fill-opacity",0.2);d3.event.preventDefault(); d3.event.stopPropagation();}}) .on("mousedown",function(d) {if (!lasso && isButtonEnabled(d)) {focusView();d3.select(this).attr("fill-opacity",0.2);d3.event.preventDefault(); d3.event.stopPropagation();}})
.on("mouseup",function(d) {if (!lasso && isButtonEnabled(d)) { d3.select(this).attr("fill-opacity",0.4);d3.event.preventDefault();d3.event.stopPropagation();}}) .on("mouseup",function(d) {if (!lasso && isButtonEnabled(d)) { d3.select(this).attr("fill-opacity",0.4);d3.event.preventDefault();d3.event.stopPropagation();}})
@@ -4553,12 +4544,13 @@ RED.view = (function() {
} }
var mainRect = document.createElementNS("http://www.w3.org/2000/svg","rect"); var mainRect = RED.utils.createSVGElement("rect", {
class: "red-ui-flow-node "+(d.type == "unknown"?"red-ui-flow-node-unknown":""),
rx: 5,
ry: 5,
fill: RED.utils.getNodeColor(d.type,d._def)
})
mainRect.__data__ = d; mainRect.__data__ = d;
mainRect.setAttribute("class", "red-ui-flow-node "+(d.type == "unknown"?"red-ui-flow-node-unknown":""));
mainRect.setAttribute("rx", 5);
mainRect.setAttribute("ry", 5);
mainRect.setAttribute("fill", RED.utils.getNodeColor(d.type,d._def));
node[0][0].__mainRect__ = mainRect; node[0][0].__mainRect__ = mainRect;
d3.select(mainRect) d3.select(mainRect)
.on("mouseup",nodeMouseUp) .on("mouseup",nodeMouseUp)
@@ -4573,61 +4565,65 @@ RED.view = (function() {
if (d._def.icon) { if (d._def.icon) {
var icon_url = RED.utils.getNodeIcon(d._def,d); var icon_url = RED.utils.getNodeIcon(d._def,d);
var icon_groupEl = document.createElementNS("http://www.w3.org/2000/svg","g"); var icon_groupEl = RED.utils.createSVGElement("g", {
icon_groupEl.__data__ = d; class: "red-ui-flow-node-icon-group"+("right" == d._def.align?" red-ui-flow-node-icon-group-right":""),
icon_groupEl.setAttribute("class","red-ui-flow-node-icon-group"+("right" == d._def.align?" red-ui-flow-node-icon-group-right":"")); x: 0,
icon_groupEl.setAttribute("x",0); y: 0
icon_groupEl.setAttribute("y",0); })
icon_groupEl.style["pointer-events"] = "none"; icon_groupEl.style["pointer-events"] = "none";
icon_groupEl.__data__ = d;
node[0][0].__iconGroup__ = icon_groupEl; node[0][0].__iconGroup__ = icon_groupEl;
var icon_shade = document.createElementNS("http://www.w3.org/2000/svg","path"); var icon_shade = RED.utils.createSVGElement("path", {
icon_shade.setAttribute("x",0); x: 0,
icon_shade.setAttribute("y",0); y: 0,
icon_shade.setAttribute("class","red-ui-flow-node-icon-shade") class: "red-ui-flow-node-icon-shade"
})
icon_groupEl.appendChild(icon_shade); icon_groupEl.appendChild(icon_shade);
node[0][0].__iconShade__ = icon_shade; node[0][0].__iconShade__ = icon_shade;
var icon_group = d3.select(icon_groupEl) var icon_group = d3.select(icon_groupEl)
createIconAttributes(icon_url, icon_group, d); createIconAttributes(icon_url, icon_group, d);
var icon_shade_border = document.createElementNS("http://www.w3.org/2000/svg","path"); var icon_shade_border = RED.utils.createSVGElement("path", {
icon_shade_border.setAttribute("d","right" != d._def.align ? "M 30 1 l 0 "+(d.h-2) : "M 0 1 l 0 "+(d.h-2) ) d: "right" != d._def.align ? "M 30 1 l 0 "+(d.h-2) : "M 0 1 l 0 "+(d.h-2),
icon_shade_border.setAttribute("class", "red-ui-flow-node-icon-shade-border") class: "red-ui-flow-node-icon-shade-border"
})
icon_groupEl.appendChild(icon_shade_border); icon_groupEl.appendChild(icon_shade_border);
node[0][0].__iconShadeBorder__ = icon_shade_border; node[0][0].__iconShadeBorder__ = icon_shade_border;
nodeContents.appendChild(icon_groupEl); nodeContents.appendChild(icon_groupEl);
} }
var text = document.createElementNS("http://www.w3.org/2000/svg","g"); var text = RED.utils.createSVGElement("g", {
text.setAttribute("class","red-ui-flow-node-label"+(hideLabel?" hide":"")+(d._def.align?" red-ui-flow-node-label-"+d._def.align:"")); class: "red-ui-flow-node-label"+(hideLabel?" hide":"")+(d._def.align?" red-ui-flow-node-label-"+d._def.align:""),
text.setAttribute("transform","translate(38,0)"); transform: "translate(38,0)"
// text.setAttribute("dy", ".3px"); // dy: ".3px",
// text.setAttribute("text-anchor",d._def.align !== "right" ? "start":"end"); // "text-anchor": d._def.align !== "right" ? "start":"end"
})
nodeContents.appendChild(text); nodeContents.appendChild(text);
node[0][0].__textGroup__ = text; node[0][0].__textGroup__ = text;
var statusEl = document.createElementNS("http://www.w3.org/2000/svg","g"); var statusEl = RED.utils.createSVGElement("g", { class: "red-ui-flow-node-status-group" })
// statusEl.__data__ = d;
statusEl.setAttribute("class","red-ui-flow-node-status-group");
statusEl.style.display = "none"; statusEl.style.display = "none";
node[0][0].__statusGroup__ = statusEl; node[0][0].__statusGroup__ = statusEl;
var statusRect = document.createElementNS("http://www.w3.org/2000/svg","rect"); var statusRect = RED.utils.createSVGElement("rect", {
statusRect.setAttribute("class","red-ui-flow-node-status"); class: "red-ui-flow-node-status",
statusRect.setAttribute("x",6); x: 6,
statusRect.setAttribute("y",1); y: 1,
statusRect.setAttribute("width",9); width: 9,
statusRect.setAttribute("height",9); height: 9,
statusRect.setAttribute("rx",2); rx: 2,
statusRect.setAttribute("ry",2); ry: 2,
statusRect.setAttribute("stroke-width","3"); "stroke-width": 3
})
statusEl.appendChild(statusRect); statusEl.appendChild(statusRect);
node[0][0].__statusShape__ = statusRect; node[0][0].__statusShape__ = statusRect;
var statusLabel = document.createElementNS("http://www.w3.org/2000/svg","text"); var statusLabel = RED.utils.createSVGElement("text", {
statusLabel.setAttribute("class","red-ui-flow-node-status-label"); class: "red-ui-flow-node-status-label",
statusLabel.setAttribute("x",20); x: 20,
statusLabel.setAttribute("y",10); y: 10
})
statusEl.appendChild(statusLabel); statusEl.appendChild(statusLabel);
node[0][0].__statusLabel__ = statusLabel; node[0][0].__statusLabel__ = statusLabel;
@@ -4712,10 +4708,11 @@ RED.view = (function() {
} }
for (var i=0; i<sn; i++) { for (var i=0; i<sn; i++) {
if (i===textLines.length) { if (i===textLines.length) {
var line = document.createElementNS("http://www.w3.org/2000/svg","text"); var line = RED.utils.createSVGElement("text", {
line.setAttribute("class","red-ui-flow-node-label-text"); class: "red-ui-flow-node-label-text",
line.setAttribute("x",0); x: 0,
line.setAttribute("y",i*24); y: i*24
})
this.__textGroup__.appendChild(line); this.__textGroup__.appendChild(line);
} }
textLines[i].textContent = sa[i]; textLines[i].textContent = sa[i];
@@ -4814,22 +4811,23 @@ RED.view = (function() {
for(var portIndex = 0; portIndex < numOutputs; portIndex++ ) { for(var portIndex = 0; portIndex < numOutputs; portIndex++ ) {
var portGroup; var portGroup;
if (portIndex === this.__outputs__.length) { if (portIndex === this.__outputs__.length) {
portGroup = document.createElementNS("http://www.w3.org/2000/svg","g"); portGroup = RED.utils.createSVGElement("g", { class: "red-ui-flow-port-output" })
portGroup.setAttribute("class","red-ui-flow-port-output");
var portPort; var portPort;
if (d.type === "link out") { if (d.type === "link out") {
portPort = document.createElementNS("http://www.w3.org/2000/svg","circle"); portPort = RED.utils.createSVGElement("circle", {
portPort.setAttribute("cx",11); cx: 11,
portPort.setAttribute("cy",5); cy: 5,
portPort.setAttribute("r",5); r: 5,
portPort.setAttribute("class","red-ui-flow-port red-ui-flow-link-port"); class: "red-ui-flow-port red-ui-flow-link-port"
})
} else { } else {
portPort = document.createElementNS("http://www.w3.org/2000/svg","rect"); portPort = RED.utils.createSVGElement("rect", {
portPort.setAttribute("rx",3); rx: 3,
portPort.setAttribute("ry",3); ry: 3,
portPort.setAttribute("width",10); width: 10,
portPort.setAttribute("height",10); height: 10,
portPort.setAttribute("class","red-ui-flow-port"); class: "red-ui-flow-port"
})
} }
portGroup.appendChild(portPort); portGroup.appendChild(portPort);
portGroup.__port__ = portPort; portGroup.__port__ = portPort;
@@ -4981,26 +4979,28 @@ RED.view = (function() {
var junction = d3.select(this); var junction = d3.select(this);
var contents = document.createDocumentFragment(); var contents = document.createDocumentFragment();
// d.added = true; // d.added = true;
var junctionBack = document.createElementNS("http://www.w3.org/2000/svg","rect"); var junctionBack = RED.utils.createSVGElement("rect", {
junctionBack.setAttribute("class","red-ui-flow-junction-background"); class: "red-ui-flow-junction-background",
junctionBack.setAttribute("x",-5); x: -5,
junctionBack.setAttribute("y",-5); y: -5,
junctionBack.setAttribute("width",10); width: 10,
junctionBack.setAttribute("height",10); height: 10,
junctionBack.setAttribute("rx",3); rx: 3,
junctionBack.setAttribute("ry",3); ry: 3
})
junctionBack.__data__ = d; junctionBack.__data__ = d;
this.__junctionBack__ = junctionBack; this.__junctionBack__ = junctionBack;
contents.appendChild(junctionBack); contents.appendChild(junctionBack);
var junctionInput = document.createElementNS("http://www.w3.org/2000/svg","rect"); var junctionInput = RED.utils.createSVGElement("rect", {
junctionInput.setAttribute("class","red-ui-flow-junction-port red-ui-flow-junction-port-input"); class: "red-ui-flow-junction-port red-ui-flow-junction-port-input",
junctionInput.setAttribute("x",-5); x: -5,
junctionInput.setAttribute("y",-5); y: -5,
junctionInput.setAttribute("width",10); width: 10,
junctionInput.setAttribute("height",10); height: 10,
junctionInput.setAttribute("rx",3); rx: 3,
junctionInput.setAttribute("ry",3); ry: 3
})
junctionInput.__data__ = d; junctionInput.__data__ = d;
junctionInput.__portType__ = PORT_TYPE_INPUT; junctionInput.__portType__ = PORT_TYPE_INPUT;
junctionInput.__portIndex__ = 0; junctionInput.__portIndex__ = 0;
@@ -5012,14 +5012,15 @@ RED.view = (function() {
this.__junctionInput__ = junctionInput; this.__junctionInput__ = junctionInput;
contents.appendChild(junctionInput); contents.appendChild(junctionInput);
var junctionOutput = document.createElementNS("http://www.w3.org/2000/svg","rect"); var junctionOutput = RED.utils.createSVGElement("rect", {
junctionOutput.setAttribute("class","red-ui-flow-junction-port red-ui-flow-junction-port-output"); class: "red-ui-flow-junction-port red-ui-flow-junction-port-output",
junctionOutput.setAttribute("x",-5); x: -5,
junctionOutput.setAttribute("y",-5); y: -5,
junctionOutput.setAttribute("width",10); width: 10,
junctionOutput.setAttribute("height",10); height: 10,
junctionOutput.setAttribute("rx",3); rx: 3,
junctionOutput.setAttribute("ry",3); ry: 3,
})
junctionOutput.__data__ = d; junctionOutput.__data__ = d;
junctionOutput.__portType__ = PORT_TYPE_OUTPUT; junctionOutput.__portType__ = PORT_TYPE_OUTPUT;
junctionOutput.__portIndex__ = 0; junctionOutput.__portIndex__ = 0;
@@ -5076,9 +5077,8 @@ RED.view = (function() {
var pathContents = document.createDocumentFragment(); var pathContents = document.createDocumentFragment();
d.added = true; d.added = true;
var pathBack = document.createElementNS("http://www.w3.org/2000/svg","path"); var pathBack = RED.utils.createSVGElement("path", { class: "red-ui-flow-link-background red-ui-flow-link-path"+(d.link?" red-ui-flow-link-link":"")})
pathBack.__data__ = d; pathBack.__data__ = d;
pathBack.setAttribute("class","red-ui-flow-link-background red-ui-flow-link-path"+(d.link?" red-ui-flow-link-link":""));
this.__pathBack__ = pathBack; this.__pathBack__ = pathBack;
pathContents.appendChild(pathBack); pathContents.appendChild(pathBack);
d3.select(pathBack) d3.select(pathBack)
@@ -5114,16 +5114,17 @@ RED.view = (function() {
} }
}) })
var pathOutline = document.createElementNS("http://www.w3.org/2000/svg","path"); var pathOutline = RED.utils.createSVGElement("path", {
class: "red-ui-flow-link-outline red-ui-flow-link-path"
})
pathOutline.__data__ = d; pathOutline.__data__ = d;
pathOutline.setAttribute("class","red-ui-flow-link-outline red-ui-flow-link-path");
this.__pathOutline__ = pathOutline; this.__pathOutline__ = pathOutline;
pathContents.appendChild(pathOutline); pathContents.appendChild(pathOutline);
var pathLine = document.createElementNS("http://www.w3.org/2000/svg","path"); var pathLine = RED.utils.createSVGElement("path", {
class: "red-ui-flow-link-line red-ui-flow-link-path"+(d.link?" red-ui-flow-link-link":(activeSubflow?" red-ui-flow-subflow-link":""))
})
pathLine.__data__ = d; pathLine.__data__ = d;
pathLine.setAttribute("class","red-ui-flow-link-line red-ui-flow-link-path"+
(d.link?" red-ui-flow-link-link":(activeSubflow?" red-ui-flow-subflow-link":"")));
this.__pathLine__ = pathLine; this.__pathLine__ = pathLine;
pathContents.appendChild(pathLine); pathContents.appendChild(pathLine);