2016-01-11 21:22:30 +00:00
|
|
|
|
|
|
|
module.exports = function(RED) {
|
|
|
|
"use strict";
|
|
|
|
var spawn = require('child_process').spawn;
|
2017-09-01 15:22:35 +01:00
|
|
|
var execSync = require('child_process').execSync;
|
2016-01-29 20:08:30 +00:00
|
|
|
var fs = require('fs');
|
2016-01-11 21:22:30 +00:00
|
|
|
var colors = require('./colours.js');
|
|
|
|
var piCommand = __dirname+'/neopix';
|
2018-04-15 11:24:56 +01:00
|
|
|
var allOK = true;
|
2016-01-11 21:22:30 +00:00
|
|
|
|
2016-11-02 10:02:07 +00:00
|
|
|
try {
|
|
|
|
var cpuinfo = fs.readFileSync("/proc/cpuinfo").toString();
|
2018-04-15 11:24:56 +01:00
|
|
|
if (cpuinfo.indexOf(": BCM") === -1) {
|
|
|
|
RED.log.warn("rpi-neopixels : "+RED._("node-red:rpi-gpio.errors.ignorenode"));
|
|
|
|
allOK = false;
|
|
|
|
}
|
2021-12-12 18:05:13 +01:00
|
|
|
else if (execSync(piCommand+" check").toString() !== "") {
|
2018-04-15 11:24:56 +01:00
|
|
|
RED.log.warn("rpi-neopixels : Can't find neopixel python library");
|
|
|
|
allOK = false;
|
|
|
|
}
|
|
|
|
else if (!(1 & parseInt ((fs.statSync(piCommand).mode & parseInt ("777", 8)).toString (8)[0]))) {
|
|
|
|
RED.log.warn("rpi-neopixels : "+RED._("node-red:rpi-gpio.errors.needtobeexecutable",{command:piCommand}));
|
|
|
|
allOK = false;
|
|
|
|
}
|
2017-01-29 17:45:44 +00:00
|
|
|
}
|
|
|
|
catch(err) {
|
2018-04-15 11:24:56 +01:00
|
|
|
RED.log.warn("rpi-neopixels : "+RED._("node-red:rpi-gpio.errors.ignorenode"));
|
|
|
|
allOK = false;
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// the magic to make python print stuff immediately
|
|
|
|
process.env.PYTHONUNBUFFERED = 1;
|
|
|
|
|
2016-03-01 21:10:58 +00:00
|
|
|
function PiNeopixelNode(n) {
|
2016-01-11 21:22:30 +00:00
|
|
|
RED.nodes.createNode(this,n);
|
|
|
|
this.pixels = n.pixels || 1;
|
|
|
|
this.bgnd = n.bgnd || "0,0,0";
|
|
|
|
this.fgnd = n.fgnd || "128,128,128";
|
|
|
|
this.mode = n.mode || "pcent";
|
2016-01-29 20:08:30 +00:00
|
|
|
this.rgb = n.rgb || "rgb";
|
2018-04-14 23:17:12 +01:00
|
|
|
this.gamma = n.gamma;
|
|
|
|
if (this.gamma === undefined) { this.gamma = true; }
|
2020-12-12 14:26:31 +00:00
|
|
|
this.gpio = n.gpio || 18;
|
|
|
|
this.channel = 0;
|
|
|
|
if (this.gpio == 13 || this.gpio == 19) { this.channel = 1; }
|
2018-04-14 23:17:12 +01:00
|
|
|
this.brightness = Number(n.brightness || 100);
|
2016-01-11 21:22:30 +00:00
|
|
|
this.wipe = Number(n.wipe || 40);
|
|
|
|
if (this.wipe < 0) { this.wipe = 0; }
|
2018-04-14 23:17:12 +01:00
|
|
|
if (this.brightness < 0) { this.brightness = 0; }
|
|
|
|
if (this.brightness > 100) { this.brightness = 100; }
|
2016-01-11 21:22:30 +00:00
|
|
|
var node = this;
|
2016-01-13 19:59:30 +00:00
|
|
|
var needle = "255,255,255";
|
2016-04-09 18:00:14 +01:00
|
|
|
//var p1 = /^\#[A-Fa-f0-9]{6}$/
|
2016-01-11 21:22:30 +00:00
|
|
|
var p2 = /^[0-9]+,[0-9]+,[0-9]+$/
|
|
|
|
var p3 = /^[0-9]+,[0-9]+,[0-9]+,[0-9]+$/
|
|
|
|
var p4 = /^[0-9]+,[0-9]+,[0-9]+,[0-9]+,[0-9]+$/
|
|
|
|
|
|
|
|
function inputlistener(msg) {
|
2018-04-14 23:17:12 +01:00
|
|
|
if (msg.hasOwnProperty("brightness")) {
|
2018-04-15 17:59:30 +01:00
|
|
|
node.child.stdin.write("brightness,"+msg.brightness.toString()+"\n");
|
2018-04-14 23:17:12 +01:00
|
|
|
}
|
2016-01-11 21:22:30 +00:00
|
|
|
if (msg.hasOwnProperty("payload")) {
|
|
|
|
var pay = msg.payload.toString().toUpperCase();
|
|
|
|
var parts = pay.split(",");
|
|
|
|
if (parts.length <= 2) {
|
|
|
|
if (parts.length === 2) { // it's a colour and length
|
|
|
|
if (isNaN(parseInt(parts[1]))) { parts = parts.reverse(); }
|
2016-01-29 20:08:30 +00:00
|
|
|
if (colors.getRGB(parts[0],node.rgb)) {
|
2016-01-11 21:22:30 +00:00
|
|
|
var l = parts[1];
|
2016-01-13 19:59:30 +00:00
|
|
|
if (node.mode.indexOf("pcent") >= 0) { l = parseInt(l / 100 * node.pixels + 0.5); }
|
2016-01-11 21:22:30 +00:00
|
|
|
l = l - 1;
|
2016-01-13 19:59:30 +00:00
|
|
|
if (node.mode.indexOf("need") >= 0) {
|
2016-01-29 20:08:30 +00:00
|
|
|
needle = colors.getRGB(parts[0],node.rgb);
|
2016-01-13 19:59:30 +00:00
|
|
|
pay = "0,"+(l-1)+","+node.fgnd+"\n"+l+","+needle+"\n"+(l+1)+","+(node.pixels-1)+","+node.bgnd;
|
2017-01-29 17:45:44 +00:00
|
|
|
}
|
|
|
|
else {
|
2016-01-29 20:08:30 +00:00
|
|
|
node.fgnd = colors.getRGB(parts[0],node.rgb);
|
2016-01-13 19:59:30 +00:00
|
|
|
pay = "0,"+l+","+node.fgnd+"\n"+(l+1)+","+(node.pixels-1)+","+node.bgnd;
|
|
|
|
}
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
2016-01-29 20:08:30 +00:00
|
|
|
else { node.warn("Invalid colour : "+pay); return; }
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (isNaN(pay)) { // it's a single colour word so set background
|
2016-01-29 20:08:30 +00:00
|
|
|
if (colors.getRGB(pay,node.rgb)) {
|
|
|
|
node.bgnd = colors.getRGB(pay,node.rgb);
|
2016-01-11 21:22:30 +00:00
|
|
|
pay = node.bgnd;
|
|
|
|
}
|
|
|
|
else { node.warn("Invalid payload : "+pay); return; }
|
|
|
|
}
|
|
|
|
else { // it's a single number so just draw bar
|
2016-03-01 21:10:58 +00:00
|
|
|
var ll = pay;
|
|
|
|
if (node.mode.indexOf("pcent") >= 0) { ll = parseInt(ll / 100 * node.pixels + 0.5); }
|
|
|
|
ll = ll - 1;
|
2016-01-13 19:59:30 +00:00
|
|
|
if (node.mode.indexOf("need") >= 0) {
|
2016-03-01 21:10:58 +00:00
|
|
|
pay = "0,"+(ll-1)+","+node.fgnd+"\n"+ll+","+needle+"\n"+(ll+1)+","+(node.pixels-1)+","+node.bgnd;
|
2017-01-29 17:45:44 +00:00
|
|
|
}
|
|
|
|
else {
|
2016-03-01 21:10:58 +00:00
|
|
|
pay = "0,"+ll+","+node.fgnd+"\n"+(ll+1)+","+(node.pixels-1)+","+node.bgnd;
|
2016-01-13 19:59:30 +00:00
|
|
|
}
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
|
|
|
}
|
2016-01-29 20:08:30 +00:00
|
|
|
node.child.stdin.write(pay+"\n");
|
|
|
|
return;
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
2016-01-29 20:08:30 +00:00
|
|
|
if ( p2.test(pay) || p3.test(pay) || p4.test(pay) ) {
|
|
|
|
if ((parts.length > 2) && (node.rgb === "grb")) { // swap r and g values
|
|
|
|
var tmp = parts[parts.length-3];
|
|
|
|
parts[parts.length-3] = parts[parts.length-2];
|
|
|
|
parts[parts.length-2] = tmp;
|
|
|
|
}
|
|
|
|
if (parts.length === 3) { node.bgnd = parts.join(","); }
|
|
|
|
node.child.stdin.write(parts.join(",")+"\n"); // handle 3 parts, 4 part and 5 parts in the python
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
|
|
|
else { node.warn("Invalid payload : "+pay); }
|
|
|
|
}
|
|
|
|
}
|
2018-04-15 11:09:32 +01:00
|
|
|
|
2018-04-15 11:24:56 +01:00
|
|
|
if (allOK === true) {
|
2020-12-12 14:26:31 +00:00
|
|
|
node.child = spawn(piCommand, [node.pixels, node.wipe, node.mode, node.brightness, node.gamma, node.channel, node.gpio]);
|
2018-04-15 11:24:56 +01:00
|
|
|
node.status({fill:"green",shape:"dot",text:"ok"});
|
|
|
|
|
|
|
|
node.on("input", inputlistener);
|
|
|
|
|
|
|
|
node.child.stdout.on('data', function (data) {
|
|
|
|
if (RED.settings.verbose) { node.log("out: "+data+" :"); }
|
|
|
|
});
|
|
|
|
|
|
|
|
node.child.stderr.on('data', function (data) {
|
|
|
|
if (RED.settings.verbose) { node.log("err: "+data+" :"); }
|
|
|
|
});
|
|
|
|
|
|
|
|
node.child.on('close', function () {
|
|
|
|
node.child = null;
|
|
|
|
if (RED.settings.verbose) { node.log(RED._("rpi-gpio.status.closed")); }
|
2019-06-27 22:26:50 +01:00
|
|
|
if (node.finished) {
|
2018-04-15 11:24:56 +01:00
|
|
|
node.status({fill:"grey",shape:"ring",text:"closed"});
|
2019-06-27 22:26:50 +01:00
|
|
|
node.finished();
|
2018-04-15 11:24:56 +01:00
|
|
|
}
|
|
|
|
else { node.status({fill:"red",shape:"ring",text:"stopped"}); }
|
|
|
|
});
|
2016-01-11 21:22:30 +00:00
|
|
|
|
2018-04-15 11:24:56 +01:00
|
|
|
node.child.on('error', function (err) {
|
|
|
|
if (err.errno === "ENOENT") { node.error(RED._("rpi-gpio.errors.commandnotfound")); }
|
|
|
|
else if (err.errno === "EACCES") { node.error(RED._("rpi-gpio.errors.commandnotexecutable")); }
|
|
|
|
else { node.error(RED._("rpi-gpio.errors.error")+': ' + err.errno); }
|
|
|
|
});
|
2016-01-11 21:22:30 +00:00
|
|
|
|
2018-04-15 11:24:56 +01:00
|
|
|
node.on("close", function(done) {
|
2016-01-11 21:22:30 +00:00
|
|
|
node.status({fill:"grey",shape:"ring",text:"closed"});
|
2018-04-15 11:24:56 +01:00
|
|
|
if (node.child != null) {
|
2019-06-27 22:26:50 +01:00
|
|
|
node.finished = done;
|
2018-04-15 11:24:56 +01:00
|
|
|
node.child.kill('SIGKILL');
|
|
|
|
}
|
|
|
|
else { done(); }
|
|
|
|
});
|
2016-01-11 21:22:30 +00:00
|
|
|
|
2018-04-15 11:24:56 +01:00
|
|
|
if (node.bgnd) {
|
|
|
|
if (node.bgnd.split(',').length === 1) {
|
|
|
|
node.bgnd = colors.getRGB(node.bgnd,node.rgb);
|
|
|
|
}
|
|
|
|
if (node.mode.indexOf("shift") === -1) {
|
|
|
|
node.child.stdin.write(node.bgnd+"\n");
|
|
|
|
}
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
|
|
|
|
2018-04-15 11:24:56 +01:00
|
|
|
if (node.fgnd) {
|
|
|
|
if (node.fgnd.split(',').length === 1) {
|
|
|
|
node.fgnd = colors.getRGB(node.fgnd,node.rgb);
|
|
|
|
}
|
2016-01-13 19:59:30 +00:00
|
|
|
}
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
2018-04-15 11:24:56 +01:00
|
|
|
else {
|
|
|
|
node.status({fill:"grey",shape:"dot",text:"node-red:rpi-gpio.status.not-available"});
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|
|
|
|
}
|
2016-03-01 21:10:58 +00:00
|
|
|
RED.nodes.registerType("rpi-neopixels",PiNeopixelNode);
|
2016-01-11 21:22:30 +00:00
|
|
|
}
|