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:
parent
1f0aa6908c
commit
38feb99072
@ -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 α 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 α 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();
|
||||||
|
@ -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.
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -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});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
Loading…
Reference in New Issue
Block a user