1
0
mirror of https://github.com/node-red/node-red-nodes.git synced 2023-10-10 13:36:58 +02:00

Added "reduce" option to the Smooth function node (#577)

This commit is contained in:
Ola Tuvesson 2019-09-19 09:11:23 +01:00 committed by Dave Conway-Jones
parent 1f0aa6908c
commit 38feb99072
3 changed files with 97 additions and 8 deletions

View File

@ -32,6 +32,11 @@
<option value="multi">Different msg.topic as individual streams.</option> <option value="multi">Different msg.topic as individual streams.</option>
</select> </select>
</div> </div>
<div class="form-row" id="row-input-reduce">
<label for="node-input-reduce"><i class="fa fa-compress"></i> Reduce</label>
<input type="checkbox" id="node-input-reduce" style="display:inline-block; width:20px; vertical-align:baseline;">
only emit one message per most recent N values
</div>
<br/> <br/>
<div class="form-row"> <div class="form-row">
<label for="node-input-name"><i class="fa fa-tag"></i> Name</label> <label for="node-input-name"><i class="fa fa-tag"></i> Name</label>
@ -44,9 +49,8 @@
<p>A simple node to provide various functions across several previous values, including max, min, mean, high and low pass filters.</p> <p>A simple node to provide various functions across several previous values, including max, min, mean, high and low pass filters.</p>
<p>Messages arriving with different <code>msg.topic</code> can be treated as separate streams if so configured.</p> <p>Messages arriving with different <code>msg.topic</code> can be treated as separate streams if so configured.</p>
<p>Max, Min and Mean work over a specified number of previous values.</p> <p>Max, Min and Mean work over a specified number of previous values.</p>
<p>The High and Low pass filters use a smoothing factor. The higher the number the more the smoothing. E.g. a value of 10 is <p>The High and Low pass filters use a smoothing factor. The higher the number the more the smoothing. E.g. a value of 10 is similar to an &alpha; of 0.1. It is analagous to an RC time constant - but there is no time component to this as the time is based on events arriving.</p>
similar to an &alpha; of 0.1. It is analagous to an RC time constant - but there is no time component to this as the <p>Enabling the Reduce option causes the node to only emit one message per N values (available for the Max, Min and Mean functions). E.g. if set to Mean over 10 values, there will only be one outgoing message per 10 incoming ones.</p>
time is based on events arriving.</p>
<p>If <code>msg.reset</code> is received (with any value), all the counters and intermediate values are reset to an initial state.</p> <p>If <code>msg.reset</code> is received (with any value), all the counters and intermediate values are reset to an initial state.</p>
<p><b>Note:</b> This only operates on <b>numbers</b>. Anything else will try to be made into a number and rejected if that fails.</p> <p><b>Note:</b> This only operates on <b>numbers</b>. Anything else will try to be made into a number and rejected if that fails.</p>
</script> </script>
@ -61,7 +65,8 @@
action: {value:"mean"}, action: {value:"mean"},
count: {value:"10",required:true,validate:RED.validators.number()}, count: {value:"10",required:true,validate:RED.validators.number()},
round: {value:""}, round: {value:""},
mult: {value:"single"} mult: {value:"single"},
reduce: {value:false}
}, },
inputs: 1, inputs: 1,
outputs: 1, outputs: 1,
@ -87,10 +92,12 @@
if ((a === "high") || ( a === "low" )) { if ((a === "high") || ( a === "low" )) {
$("#node-over").html("with a smoothing factor of "); $("#node-over").html("with a smoothing factor of ");
$("#node-over2").html(""); $("#node-over2").html("");
$("#row-input-reduce").hide();
} }
else { else {
$("#node-over").html("over the most recent "); $("#node-over").html("over the most recent ");
$("#node-over2").html(" values"); $("#node-over2").html(" values");
$("#row-input-reduce").show();
} }
}); });
$("#node-input-action").change(); $("#node-input-action").change();

View File

@ -9,6 +9,7 @@ module.exports = function(RED) {
if (this.round == "true") { this.round = 0; } if (this.round == "true") { this.round = 0; }
this.count = Number(n.count); this.count = Number(n.count);
this.mult = n.mult || "single"; this.mult = n.mult || "single";
this.reduce = n.reduce || false;
this.property = n.property || "payload"; this.property = n.property || "payload";
var node = this; var node = this;
var v = {}; var v = {};
@ -16,6 +17,7 @@ module.exports = function(RED) {
this.on('input', function (msg) { this.on('input', function (msg) {
var value = RED.util.getMessageProperty(msg,node.property); var value = RED.util.getMessageProperty(msg,node.property);
var top = msg.topic || "_my_default_topic"; var top = msg.topic || "_my_default_topic";
var reduce = node.reduce;
if (this.mult === "single") { top = "a"; } if (this.mult === "single") { top = "a"; }
if ((v.hasOwnProperty(top) !== true) || msg.hasOwnProperty("reset")) { if ((v.hasOwnProperty(top) !== true) || msg.hasOwnProperty("reset")) {
@ -26,15 +28,18 @@ module.exports = function(RED) {
v[top].pop = 0; v[top].pop = 0;
v[top].old = null; v[top].old = null;
v[top].count = this.count; v[top].count = this.count;
v[top].iter = 0;
} }
if (value !== undefined) { if (value !== undefined) {
var n = Number(value); var n = Number(value);
if (!isNaN(n)) { if (!isNaN(n)) {
v[top].iter++;
if ((node.action === "low") || (node.action === "high")) { if ((node.action === "low") || (node.action === "high")) {
if (v[top].old == null) { v[top].old = n; } if (v[top].old == null) { v[top].old = n; }
v[top].old = v[top].old + (n - v[top].old) / v[top].count; v[top].old = v[top].old + (n - v[top].old) / v[top].count;
if (node.action === "low") { value = v[top].old; } if (node.action === "low") { value = v[top].old; }
else { value = n - v[top].old; } else { value = n - v[top].old; }
reduce = false;
} }
else { else {
v[top].a.push(n); v[top].a.push(n);
@ -61,10 +66,13 @@ module.exports = function(RED) {
if (node.round !== false) { if (node.round !== false) {
value = Math.round(value * Math.pow(10, node.round)) / Math.pow(10, node.round); value = Math.round(value * Math.pow(10, node.round)) / Math.pow(10, node.round);
} }
RED.util.setMessageProperty(msg,node.property,value); if (reduce == false || v[top].iter == v[top].count) {
node.send(msg); v[top].iter = 0;
RED.util.setMessageProperty(msg,node.property,value);
node.send(msg);
}
} }
else { node.log("Not a number: "+value); } else { node.log("Not a number: " + value); }
} // ignore msg with no payload property. } // ignore msg with no payload property.
}); });
} }

View File

@ -316,5 +316,79 @@ describe('smooth node', function() {
n1.emit("input", {payload:9, topic:"B"}); n1.emit("input", {payload:9, topic:"B"});
}); });
}); });
it("should reduce the number of messages by averaging if asked", function(done) {
var flow = [{"id":"n1", "type":"smooth", action:"mean", count:"5", reduce:"true", wires:[["n2"]] },
{id:"n2", type:"helper"} ];
helper.load(testNode, flow, function() {
var n1 = helper.getNode("n1");
var n2 = helper.getNode("n2");
var c = 0;
n2.on("input", function(msg) {
c += 1;
if (c === 1) { msg.should.have.a.property("payload", 3); }
else if (c === 2) { msg.should.have.a.property("payload", 6); done(); }
else if (c > 2) { done(new Error("should not emit more than two messages.")); }
});
n1.emit("input", {payload:1});
n1.emit("input", {payload:2});
n1.emit("input", {payload:3});
n1.emit("input", {payload:4});
n1.emit("input", {payload:5});
n1.emit("input", {payload:6});
n1.emit("input", {payload:7});
n1.emit("input", {payload:8});
n1.emit("input", {payload:9})
; n1.emit("input", {payload:0});
});
});
it("should reduce the number of messages by max value, if asked", function(done) {
var flow = [{"id":"n1", "type":"smooth", action:"max", count:"5", reduce:"true", wires:[["n2"]] },
{id:"n2", type:"helper"} ];
helper.load(testNode, flow, function() {
var n1 = helper.getNode("n1");
var n2 = helper.getNode("n2");
var c = 0;
n2.on("input", function(msg) {
c += 1;
if (c === 1) { msg.should.have.a.property("payload", 5); }
else if (c === 2) { msg.should.have.a.property("payload", 9); done(); }
else if (c > 2) { done(new Error("should not emit more than two messages.")); }
});
n1.emit("input", {payload:1});
n1.emit("input", {payload:2});
n1.emit("input", {payload:3});
n1.emit("input", {payload:4});
n1.emit("input", {payload:5});
n1.emit("input", {payload:6});
n1.emit("input", {payload:7});
n1.emit("input", {payload:8});
n1.emit("input", {payload:9});
n1.emit("input", {payload:0});
});
});
it("should reduce the number of messages by min value, if asked", function(done) {
var flow = [{"id":"n1", "type":"smooth", action:"min", count:"5", reduce:"true", wires:[["n2"]] },
{id:"n2", type:"helper"} ];
helper.load(testNode, flow, function() {
var n1 = helper.getNode("n1");
var n2 = helper.getNode("n2");
var c = 0;
n2.on("input", function(msg) {
c += 1;
if (c === 1) { msg.should.have.a.property("payload", 1); }
else if (c === 2) { msg.should.have.a.property("payload", 0); done(); }
else if (c > 2) { done(new Error("should not emit more than two messages.")); }
});
n1.emit("input", {payload:1});
n1.emit("input", {payload:2});
n1.emit("input", {payload:3});
n1.emit("input", {payload:4});
n1.emit("input", {payload:5});
n1.emit("input", {payload:6});
n1.emit("input", {payload:7});
n1.emit("input", {payload:8});
n1.emit("input", {payload:9});
n1.emit("input", {payload:0});
});
});
}); });