Skip to contents
library(animovement)
#> Error in get(paste0(generic, ".", class), envir = get_method_env()) : 
#>   object 'type_sum.accel' not found
library(tibble)
library(dplyr, warn.conflicts = FALSE)
library(readxl)
library(here)
#> here() starts at /home/runner/work/animovement/animovement
here::i_am("vignettes/articles/clean-tracks.Rmd")
#> here() starts at /home/runner/work/animovement/animovement

The next step in our workflow is to clean the tracks. This step commonly covers three separate components:

Although these are not always completely separate steps in practice, we will treat them as such to ensure the integrity of our tracks.

Outlier detection

If you are confident that your sensors give correct readings, you can skip this step.. Even in a typical trackball setup with optical flow sensors (e.g. computer mice), there is a chance that we will get a few spurious readings from the sensors. The aim of this step is to root out those readings.

We’ll use the data we read in the Read data article, so if you’ve missed that step you need to revisit how to read data. Let us first visualise our dx and dy values.

library(ggplot2)
df <- df |>
  mutate(displacement = sqrt(dx^2 + dy^2))
full_trace <- df |>
  ggplot(aes(time, displacement)) +
  geom_line() +
  xlab("Time (s)") +
  ylab("dx, dy (dots)")

partial_trace <- df |>
  filter(time < 20) |>
  ggplot(aes(time, displacement)) +
  geom_line() +
  xlab("Time (s)") +
  ylab("dx, dy (dots)")

hist_dist <- df |>
  ggplot(aes(displacement)) +
  geom_histogram()

library(patchwork)
full_trace / partial_trace / hist_dist
#> `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.

Some animals spend a significant amount of time sitting still (as was the case here). To ensure that these zero-readings do not skew our outlier detection, let’s filter them out and have another look at the histograms.

hist_dist2 <- df |>
  filter(displacement != 0) |>
  ggplot(aes(displacement)) +
  geom_histogram()

hist_dist2
#> `stat_bin()` using `bins = 30`. Pick better value with `binwidth`.

Now we can see that there might be a few observations with extreme observations. It is now up to each researcher to decide whether to attempt filtering out outliers, or whether we are happy with what we have.

we may want to filter out all the 0 readings At first sight, it all looks fine. We could stop here and say we’re confident there are no spurious outliers. We could also apply a variety of automated outlier detection techniques. There are several ways of achieving this, but luckily the performance package has got us covered with their check_outliers() function. Since most of these methods rely on histograms/probability density functions,

library(performance)
library(tidyr)
#> 
#> Attaching package: 'tidyr'
#> The following object is masked from 'package:animovement':
#> 
#>     replace_na
df |>
  mutate(
    ax = dx - lag(dx),
    ay = dy - lag(dy)
  ) |>
  tidyr::drop_na(ax, ay) |> 
  filter(time < 200) |>
  filter(dx != 0) |>
  select(ax, ay) |> 
  check_outliers() |>
  plot()

Interpolation

In case we removed any outliers, we can now interpolate across the gaps created.

TO BE CONTINUED…

Smoothing

All there is left to do is smooth our tracks, which is done using the smooth_tracks() function. The smoothing itself is super simple. smooth_tracks() provides a few different options:

  • roll_mean
  • roll_median
  • SOON savitsky_golay

For the rolling filters you can provide the window_width, i.e. how many observations to use in the rolling filter. The filters result in some NA values at the beginning and end of your data.

An important point about smooth_tracks() is that, instead of using our x and y values, it first back-transforms into the raw values obtained from your sensors (which are effectively “differences” between coordinates, so dx and dy) and performs the smoothing on them, before finally converting back to x and y. This may seem strange if you have previously worked with tracking data from computer vision or GPS loggers. However, whereas those modalities would return to the “true” coordinates after an outlier, mouse sensors do not. So the only way we can identify rogue values is by filtering those raw values.

Let’s try smoothing our data with a rolling_mean filter with 0.5 second (30 observations at at a sampling rate of 60Hz) window width. In case we work with multiple keypoints and/or individuals we can use group_by with our metadata for a tidyverse-friendly workflow.

df_rollmedian <- df |>
  filter_movement(method = "rollmedian", 
                  window_width = 3, 
                  use_derivatives = TRUE)
df_kalman <- df |>
  filter_movement(method = "kalman", 
                  sampling_rate = 60, 
                  use_derivatives = TRUE)
df_sgolay <- df |>
  filter_movement(method = "sgolay", 
                  sampling_rate = 60, 
                  use_derivatives = TRUE)
df_lowpass <- df |>
  filter_movement(method = "lowpass", 
                  cutoff_freq = 0.1, 
                  sampling_rate = 60, 
                  use_derivatives = TRUE)

Let’s visualise how they compare. Note that although the difference may seem negligible when plotting paths, they may become important when computing derivatives such as velocity and acceleration.

library(ggplot2)
ggplot() +
  geom_path(data = df, aes(x, y), colour = "red") +
  geom_path(data = df_rollmedian, aes(x, y), colour = "blue") +
  geom_path(data = df_kalman, aes(x, y), colour = "orange") +
  geom_path(data = df_sgolay, aes(x, y), colour = "purple") +
  geom_path(data = df_lowpass, aes(x, y), colour = "green")
#> Warning: Removed 1 row containing missing values or values outside the scale range
#> (`geom_path()`).

Not that different as the sensors are doing a good job! But we can see that the smoothed track end a bit further to the left than the raw version.