Skip to content

Commit 39b680f

Browse files
authored
Merge pull request #2 from DavZim/runningmean
Adding a running mean option to ar_plotter
2 parents d91c9c9 + 8483f8c commit 39b680f

6 files changed

+141
-4
lines changed

DESCRIPTION

+3
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,11 @@ Imports:
1919
glue,
2020
shiny,
2121
plotly
22+
Depends:
23+
Rcpp (>= 0.12.15)
2224
Remotes: ropenscilabs/webrockets
2325
LinkingTo: Rcpp
26+
RcppModules: RunningMeanModule
2427
VignetteBuilder: knitr
2528
Encoding: UTF-8
2629
RoxygenNote: 6.1.1

R/ar_plotter.R

+26-3
Original file line numberDiff line numberDiff line change
@@ -9,16 +9,21 @@
99
#' process frequency > 25 Hz (40ms delay time) or it might be the case that
1010
#' my computer doesn't have enough power. Anyway, I set this option here to
1111
#' add 40ms delay time to reduce the sampling frequency.
12+
#' @param running_mean If an integer larger than 0 is chosen, the running mean
13+
#' of each series is computed, effectively smoothing the signals. Default value
14+
#' is set to 0 (no running mean is calculated)
1215
#' @inheritParams ar_monitor
1316
#'
1417
#' @export
1518
ar_plotter <- function(fd, names = NULL, sep_fun = ar_sep_comma,
1619
reduce_freq = TRUE, flush_time = 0.05,
17-
eolchar = "\n", buf_max = 256, timeout = 5000) {
20+
eolchar = "\n", buf_max = 256, timeout = 5000,
21+
running_mean = 0) {
1822
shiny::runApp(
1923
ar_app(con = fd, names = names, sep_fun = sep_fun,
2024
flush_time = flush_time, reduce_freq = reduce_freq,
21-
eolchar = eolchar, buf_max = buf_max, timeout = timeout),
25+
eolchar = eolchar, buf_max = buf_max, timeout = timeout,
26+
running_mean = running_mean),
2227
launch.browser = rstudioapi::viewer
2328
)
2429
}
@@ -27,7 +32,9 @@ ar_plotter <- function(fd, names = NULL, sep_fun = ar_sep_comma,
2732

2833
ar_app <- function(con, names = NULL, sep_fun = ar_sep_comma,
2934
flush_time = 0.05, reduce_freq = TRUE,
30-
eolchar = "\n", buf_max = 256, timeout = 5000) {
35+
eolchar = "\n", buf_max = 256, timeout = 5000,
36+
running_mean = 0) {
37+
3138
message("Flushing Port...")
3239
ar_flush_hard(con, flush_time)
3340
first_dot <- ar_read(con, eolchar, buf_max, timeout)
@@ -38,6 +45,14 @@ ar_app <- function(con, names = NULL, sep_fun = ar_sep_comma,
3845
first_dot <- sep_fun(first_dot)
3946
signal_vars <- seq(length(first_dot))
4047

48+
if (running_mean > 0) {
49+
rmeans <- vector("list", length(first_dot))
50+
for (i in signal_vars) {
51+
rmeans[[i]] <- arduinor:::RunningMean$new(running_mean)
52+
rmeans[[i]]$insert(first_dot[i])
53+
}
54+
}
55+
4156
if (is.null(names)) {
4257
names(signal_vars) <- paste("Var", signal_vars)
4358
names(first_dot) <- paste("Var", signal_vars)
@@ -113,6 +128,14 @@ ar_app <- function(con, names = NULL, sep_fun = ar_sep_comma,
113128
if (rv$state) {
114129
ar_flush_hard(con, 0.04, FALSE)
115130
realtime <- sep_fun(ar_read(con, eolchar, buf_max, timeout))
131+
132+
if (running_mean > 0) {
133+
for (i in seq(length(realtime))) {
134+
rmeans[[i]]$insert(realtime[[i]])
135+
realtime[[i]] <- rmeans[[i]]$get_mean()
136+
}
137+
}
138+
116139
if (input$save) {
117140
cat(csv_newline(realtime), file = input$file, append = TRUE)
118141
}

R/zzz.R

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
11
.onUnload <- function(libpath) {
22
library.dynam.unload("arduinor", libpath)
3-
}
3+
}
4+
5+
Rcpp::loadModule("RunningMeanModule", TRUE)
6+

src/RcppExports.cpp

+3
Original file line numberDiff line numberDiff line change
@@ -65,12 +65,15 @@ BEGIN_RCPP
6565
END_RCPP
6666
}
6767

68+
RcppExport SEXP _rcpp_module_boot_RunningMeanModule();
69+
6870
static const R_CallMethodDef CallEntries[] = {
6971
{"_arduinor_ar_init", (DL_FUNC) &_arduinor_ar_init, 2},
7072
{"_arduinor_ar_is_open", (DL_FUNC) &_arduinor_ar_is_open, 1},
7173
{"_arduinor_ar_close", (DL_FUNC) &_arduinor_ar_close, 1},
7274
{"_arduinor_ar_read", (DL_FUNC) &_arduinor_ar_read, 4},
7375
{"_arduinor_ar_flush", (DL_FUNC) &_arduinor_ar_flush, 1},
76+
{"_rcpp_module_boot_RunningMeanModule", (DL_FUNC) &_rcpp_module_boot_RunningMeanModule, 0},
7477
{NULL, NULL, 0}
7578
};
7679

src/RunningMean.cpp

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
#include "RunningMean.h"
2+
3+
4+
RunningMean::RunningMean() : n(0) {
5+
stop("n (size of window) must be supplied");
6+
}
7+
8+
// Initiates a new Running Mean class of size n (constant)
9+
RunningMean::RunningMean(int n) : n(n) {
10+
if (n <= 0) stop("n (size of window) must be supplied");
11+
deq.resize(n);
12+
n_values = 0;
13+
sum = 0;
14+
}
15+
16+
// inserts a new value into the internal deque. Deletes the oldest value
17+
void RunningMean::insert(double v) {
18+
if (n == 0) stop("n (size of window) must be larger than zero");
19+
20+
// TODO: What happens if we insert a missing value?
21+
if (!R_IsNA(v) && !R_IsNaN(v)) {
22+
n_values++;
23+
sum += v - deq.back();
24+
deq.push_front(v);
25+
deq.pop_back();
26+
}
27+
}
28+
29+
// Returns the running mean of the values in the deque
30+
// if the size is 0, return NA, if the number of already inserted values (n_values)
31+
// is smaller than the size of the deque, use n_values. This allows the running mean
32+
// to return values even if the number of values is smaller than window lengths
33+
// could also be set to NA if needed...
34+
double RunningMean::get_mean() {
35+
if (n == 0) return NA_REAL;
36+
if (n_values < n) return(sum / n_values);
37+
return sum / n;
38+
}
39+
40+
// prints the values of the deque
41+
void RunningMean::show() {
42+
Rcout << "RunningMean object <" << static_cast<const void*>(this) <<
43+
"> window size " << n << ", number of inserts " << n_values << "\n" <<
44+
deq[0];
45+
for (int i = 1; i < n; ++i) {
46+
Rcout << " " << deq[i];
47+
}
48+
Rcout << "\n";
49+
}
50+

src/RunningMean.h

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
#include <Rcpp.h>
2+
using namespace Rcpp;
3+
4+
/*
5+
* Example R Code
6+
*
7+
* # Initiate the Class with a window-length of size 3
8+
* obj <- arduinor:::RunningMean$new(3)
9+
*
10+
* # insert a new value
11+
* obj$insert(1.03)
12+
* obj$insert(1.05)
13+
* obj$insert(1.04)
14+
*
15+
* # inspect the object
16+
* obj
17+
*
18+
* # get the running mean
19+
* obj$get_mean()
20+
*
21+
* # add another value, effectively removing the first insert (1.03)
22+
* obj$insert(1.06)
23+
* obj
24+
* obj$get_mean()
25+
*
26+
*/
27+
28+
// defines the RunningMean class
29+
class RunningMean{
30+
public:
31+
RunningMean();
32+
RunningMean(int n);
33+
34+
void insert(double val);
35+
double get_mean();
36+
void show();
37+
38+
private:
39+
std::deque<double> deq;
40+
const int n;
41+
int n_values;
42+
double sum;
43+
};
44+
45+
// Exposes the class
46+
RCPP_MODULE(RunningMeanModule) {
47+
class_<RunningMean>("RunningMean")
48+
.default_constructor("Default constructor")
49+
.constructor<int>("Constructor with an argument")
50+
.method("insert", &RunningMean::insert)
51+
.method("get_mean", &RunningMean::get_mean)
52+
.method("show", &RunningMean::show)
53+
;
54+
}
55+

0 commit comments

Comments
 (0)