155 lines
6.3 KiB
JavaScript

module.exports = function(RED) {
var ui = require('../ui')(RED);
function validateSwitchValue(node,property,type,payload) {
if (payloadType === 'flow' || payloadType === 'global') {
try {
var parts = RED.util.normalisePropertyExpression(payload);
if (parts.length === '') {
throw new Error();
}
} catch(err) {
node.warn("Invalid payload property expression - defaulting to node id")
payload = node.id;
payloadType = 'str';
}
}
else {
payload = payload || node.id;
}
}
function SwitchNode(config) {
RED.nodes.createNode(this, config);
this.pt = config.passthru;
this.state = ["off"," "];
this.decouple = (config.decouple === "true") ? false : true;
var node = this;
node.status({});
var group = RED.nodes.getNode(config.group);
if (!group) { return; }
var tab = RED.nodes.getNode(group.config.tab);
if (!tab) { return; }
var parts;
var onvalue = config.onvalue;
var onvalueType = config.onvalueType;
if (onvalueType === 'flow' || onvalueType === 'global') {
try {
parts = RED.util.normalisePropertyExpression(onvalue);
if (parts.length === 0) {
throw new Error();
}
} catch(err) {
node.warn("Invalid onvalue property expression - defaulting to true")
onvalue = true;
onvalueType = 'bool';
}
}
var offvalue = config.offvalue;
var offvalueType = config.offvalueType;
if (offvalueType === 'flow' || offvalueType === 'global') {
try {
parts = RED.util.normalisePropertyExpression(offvalue);
if (parts.length === 0) {
throw new Error();
}
} catch(err) {
node.warn("Invalid offvalue property expression - defaulting to false")
offvalue = false;
offvalueType = 'bool';
}
}
node.on("input", function(msg) {
node.topi = msg.topic;
});
var done = ui.add({
node: node,
tab: tab,
group: group,
emitOnlyNewValues: false,
forwardInputMessages: config.passthru,
storeFrontEndInputAsState: (config.decouple === "true") ? false : true, //config.passthru,
state: false,
control: {
type: 'switch' + (config.style ? '-' + config.style : ''),
label: config.label,
tooltip: config.tooltip,
order: config.order,
value: false,
onicon: config.onicon,
officon: config.officon,
oncolor: config.oncolor,
offcolor: config.offcolor,
animate: config.animate?"flip-icon":"",
width: config.width || group.config.width || 6,
height: config.height || 1,
className: config.className || '',
},
convert: function (payload, oldval, msg) {
var myOnValue,myOffValue;
if (onvalueType === "date") { myOnValue = Date.now(); }
else { myOnValue = RED.util.evaluateNodeProperty(onvalue,onvalueType,node); }
if (offvalueType === "date") { myOffValue = Date.now(); }
else { myOffValue = RED.util.evaluateNodeProperty(offvalue,offvalueType,node); }
if (!this.forwardInputMessages && this.storeFrontEndInputAsState) {
if (myOnValue === oldval) { return true; }
if (oldval === true) { return true; }
else { return false; }
}
if (RED.util.compareObjects(myOnValue,msg.payload)) { node.state[0] = "on"; return true; }
else if (RED.util.compareObjects(myOffValue,msg.payload)) { node.state[0] = "off"; return false; }
else { return oldval; }
},
convertBack: function (value) {
node.state[1] = value?"on":"off";
if (node.pt) {
node.status({fill:(value?"green":"red"),shape:(value?"dot":"ring"),text:value?"on":"off"});
}
else {
var col = (node.decouple) ? ((node.state[1]=="on")?"green":"red") : ((node.state[0]=="on")?"green":"red");
var shp = (node.decouple) ? ((node.state[1]=="on")?"dot":"ring") : ((node.state[0]=="on")?"dot":"ring");
var txt = (node.decouple) ? (node.state[0] +" | "+node.state[1].toUpperCase()) : (node.state[0].toUpperCase() +" | "+node.state[1])
node.status({fill:col, shape:shp, text:txt});
}
var payload = value ? onvalue : offvalue;
var payloadType = value ? onvalueType : offvalueType;
if (payloadType === "date") { value = Date.now(); }
else {
try {
value = RED.util.evaluateNodeProperty(payload,payloadType,node);
}
catch(e) {
if (payloadType === "bin") { node.error("Badly formatted buffer"); }
else { node.error(e,payload); }
}
}
return value;
},
beforeSend: function (msg) {
var t = RED.util.evaluateNodeProperty(config.topic,config.topicType || "str",node,msg) || node.topi;
if (t) { msg.topic = t; }
}
});
if (!node.pt) {
node.on("input", function() {
var col = (node.state[0]=="on") ? "green" : "red";
var shp = (node.state[0]=="on") ? "dot" : "ring";
var txt = (node.decouple) ? (node.state[0] +" | "+node.state[1].toUpperCase()) : (node.state[0].toUpperCase() +" | "+node.state[1])
node.status({fill:col, shape:shp, text:txt});
});
}
node.on("close", done);
}
RED.nodes.registerType("ui_switch", SwitchNode);
};