Nothing
#' Plot pre-processed pupil data from `eyeris`
#'
#' S3 plotting method for objects of class `eyeris`. Plots a single-panel
#' timeseries for a subset of the pupil timeseries at each preprocessing step.
#' The intended use of this function is to provide a simple method for
#' qualitatively assessing the consequences of the preprocessing recipe and
#' parameters on the raw pupillary signal.
#'
#' @param x An object of class `eyeris` dervived from [eyeris::load_asc()].
#' @param ... Additional arguments to be passed to `plot`.
#' @param steps Which steps to plot; defaults to `all` (i.e., plot all steps).
#' Otherwise, pass in a vector containing the index of the step(s) you want to
#' plot, with index `1` being the original raw pupil timeseries.
#' @param preview_n Number of random example "epochs" to generate for
#' previewing the effect of each preprocessing step on the pupil timeseries.
#' @param preview_duration Time in seconds of each randomly selected preview.
#' @param preview_window The start and stop raw timestamps used to subset the
#' preprocessed data from each step of the `eyeris` workflow for visualization.
#' Defaults to NULL, meaning random epochs as defined by `preview_n` and
#' `preview_duration` will be plotted. To override the random epochs, set
#' `preview_window` here to a vector with relative start and stop times (in
#' seconds), for example -- `c(5,6)` -- to indicate the raw data from 5-6 secs
#' on data that were recorded at 1000 Hz). Note, the start/stop time values
#' indicated here are in seconds because `eyeris` automatically computes the
#' indices for the supplied range of seconds using the `$info$sample.rate`
#' metadata in the `eyeris` S3 class object.
#' @param seed Random seed for current plotting session. Leave NULL to select
#' `preview_n` number of random preview "epochs" (of `preview_duration`) each
#' time. Otherwise, choose any seed-integer as you would normally select for
#' [base::set.seed()], and you will be able to continue re-plotting the same
#' random example pupil epochs each time -- which is helpful when adjusting
#' parameters within and across `eyeris` workflow steps.
#' @param block For multi-block recordings, specifies which block to plot.
#' Defaults to 1. When a single `.asc` data file contains multiple
#' recording blocks, this parameter determines which block's timeseries to
#' visualize. Must be a positive integer not exceeding the total number of
#' blocks in the recording.
#' @param plot_distributions Logical flag to indicate whether to plot both
#' diagnostic pupil timeseries *and* accompanying histograms of the pupil
#' samples at each processing step. Defaults to `FALSE`.
#' @param suppress_prompt Logical flag to disable interactive confirmation
#' prompts during plotting. Defaults to `TRUE`, which avoids hanging behavior in
#' non-interactive or automated contexts (e.g., RMarkdown, scripts).
#' Set to `FALSE` only when running inside `glassbox()` with
#' `interactive_preview = TRUE`, where prompting after each step is desired, as
#' well as in the generation of interactive HTML reports with [eyeris::bidsify].
#' @param num_previews **(Deprecated)** Use `preview_n` instead.
#'
#' @return No return value; iteratively plots a subset of the pupil timeseries
#' from each preprocessing step run.
#'
#' @seealso [lifecycle::deprecate_warn()]
#'
#' @examples
#' # first, generate the preprocessed pupil data
#' my_eyeris_data <- system.file("extdata", "memory.asc", package = "eyeris") |>
#' eyeris::load_asc() |>
#' eyeris::deblink(extend = 50) |>
#' eyeris::detransient() |>
#' eyeris::interpolate() |>
#' eyeris::lpfilt(plot_freqz = TRUE) |>
#' eyeris::zscore()
#'
#' # controlling the timeseries range (i.e., preview window) in your plots:
#'
#' ## example 1: using the default 10000 to 20000 ms time subset
#' plot(my_eyeris_data, seed = 0)
#'
#' ## example 2: using a custom time subset (i.e., 1 to 500 ms)
#' plot(my_eyeris_data, preview_window = c(0.01, 0.5), seed = 0)
#'
#' # controlling which block of data you would like to plot:
#'
#' ## example 1: plots first block (default)
#' plot(my_eyeris_data, seed = 0)
#'
#' ## example 2: plots a specific block
#' plot(my_eyeris_data, block = 1, seed = 0)
#'
#' ## example 3: plots a specific block along with a custom preview window
#' ## (i.e., 1000 to 2000 ms)
#' plot(
#' my_eyeris_data,
#' block = 1,
#' preview_window = c(1, 2),
#' seed = 0
#' )
#'
#' @rdname plot.eyeris
#'
#' @export
plot.eyeris <- function(x, ..., steps = NULL, preview_n = NULL,
preview_duration = NULL, preview_window = NULL,
seed = NULL, block = 1, plot_distributions = FALSE,
suppress_prompt = TRUE,
num_previews = deprecated()) {
# handle deprecated parameters
if (is_present(num_previews)) {
deprecate_warn(
"1.2.0",
"plot(num_previews)",
"plot(preview_n)"
)
preview_n <- num_previews
}
# safely handle user's current options
oldpar <- par(no.readonly = TRUE)
on.exit(par(oldpar))
# tests
tryCatch(
{
check_data(x, "plot")
},
error = function(e) {
error_handler(e, "input_data_type_error")
}
)
tryCatch(
{
check_pupil_cols(x, "plot")
},
error = function(e) {
error_handler(e, "missing_pupil_raw_error")
}
)
params <- list(...)
only_liner_trend <- if ("only_linear_trend" %in% names(params)) {
params$only_linear_trend <- params$only_linear_trend
} else {
params$only_linear_trend <- FALSE
}
non_plot_params <- c(
"preview_window", "seed", "steps", "num_previews",
"preview_n", "preview_duration", "block",
"suppress_prompt", "plot_distributions",
"only_linear_trend", "next_step"
)
plot_params <- params[!(names(params) %in% non_plot_params)]
# set param defaults outside of function declaration
if (!is.null(preview_window)) {
if (!is.null(preview_n) || !is.null(preview_duration)) {
cli::cli_alert_warning(
paste(
"preview_n and/or preview_duration will be ignored,",
"since preview_window was specified here."
)
)
}
}
if (is.null(steps)) {
steps <- "all"
}
if (is.null(preview_n)) {
preview_n <- 3
}
if (is.null(preview_duration)) {
preview_duration <- 5
}
# handle random seed for this plotting session
if (is.null(seed)) {
seed <- sample.int(.Machine$integer.max, 1)
}
# blocks handler
if (is.list(x$timeseries) && !is.data.frame(x$timeseries)) {
available_blocks <- get_block_numbers(x)
if (block %in% available_blocks) {
pupil_data <- x$timeseries[[paste0("block_", block)]]
cli::cli_alert_warning(sprintf(
"Plotting block %d from possible blocks: %s",
block,
toString(available_blocks)
))
} else {
cli::cli_abort(sprintf(
"Block %d does not exist. Available blocks: %d",
block, toString(available_blocks)
))
}
} else {
pupil_data <- x$timeseries$block_1
}
pupil_steps <- grep("^pupil_", names(pupil_data), value = TRUE)
# modified from `RColorBrewer`: Set1
colorpal <- c(
"#E41A1C", "#377EB8", "#4DAF4A", "#984EA3", "#FF7F00", "#F781BF", "#A65628"
)
colors <- c("black", colorpal)
transparent_colors <- sapply(colors, function(x) {
grDevices::adjustcolor(x, alpha.f = 0.5)
})
if (length(steps) == 1) {
if (steps[1] == "all") {
pupil_steps <- pupil_steps
colors <- colors
} else {
pupil_steps <- pupil_steps[steps]
colors <- colors[steps]
}
} else if (length(steps) > 1 && !is.null(preview_window)) {
pupil_steps <- pupil_steps[steps]
colors <- colors[steps]
} else {
pupil_steps <- pupil_steps
colors <- colors
}
if (is.null(preview_window)) {
hz <- x$info$sample.rate
withr::with_seed(seed, {
random_epochs <- draw_random_epochs(
pupil_data, preview_n,
preview_duration, hz
)
})
par(mfrow = c(1, preview_n), oma = c(0, 0, 3, 0))
detrend_plotted <- FALSE
for (i in seq_along(pupil_steps)) {
for (n in 1:preview_n) {
st <- min(random_epochs[[n]]$time_orig)
et <- max(random_epochs[[n]]$time_orig)
title <- paste0("\n[", st, " - ", et, "]")
header <- paste0(
gsub("_", " > ", gsub("pupil_", "", pupil_steps[i])),
if (is.list(x$timeseries) && !is.data.frame(x$timeseries)) {
sprintf(" (Run %d)", block)
} else {
""
}
)
if (grepl("z", pupil_steps[i])) {
y_units <- "(z)"
} else {
y_units <- "(a.u.)"
}
y_label <- paste("pupil size", y_units)
# used when running `plot()` by itself (and thus plotting all steps)
if (!only_liner_trend) {
if (grepl("_detrend$", pupil_steps[i]) && !detrend_plotted) {
plot_detrend_overlay(
pupil_data,
pupil_steps = pupil_steps,
preview_n = preview_n,
suppress_prompt = suppress_prompt
)
detrend_plotted <- TRUE
}
} else {
if (!detrend_plotted) {
plot_detrend_overlay(
pupil_data,
pupil_steps = pupil_steps,
preview_n = preview_n,
suppress_prompt = suppress_prompt
)
detrend_plotted <- TRUE
}
}
if (!is.null(params$next_step)) {
plot_data <- random_epochs[[n]][[
params$next_step[length(params$next_step)]
]]
do.call(robust_plot, c(
list(y = plot_data),
plot_params,
list(
type = "l", col = colors[i], lwd = 2,
main = title, xlab = "time (ms)", ylab = y_label
)
))
} else {
plot_data <- random_epochs[[n]][[pupil_steps[i]]]
do.call(robust_plot, c(
list(y = plot_data),
plot_params,
list(
type = "l", col = colors[i], lwd = 2,
main = title, xlab = "time (ms)", ylab = y_label
)
))
}
}
graphics::mtext(header,
outer = TRUE, cex = 1.25, font = 2
)
if (plot_distributions) {
plot_pupil_distribution(
data = pupil_data[[pupil_steps[i]]],
color = colors[i],
main = header,
xlab = y_label,
backuplab = "pupil size"
)
par(mfrow = c(1, preview_n), oma = c(0, 0, 3, 0))
}
}
par(mfrow = c(1, preview_n), oma = c(0, 0, 3, 0))
} else {
preview_window_indices <- round(preview_window * x$info$sample.rate) + 1
start_index <- preview_window_indices[1]
end_index <- preview_window_indices[2]
if (start_index < 1 || start_index > nrow(pupil_data) ||
end_index < 1 || end_index > nrow(pupil_data) ||
start_index >= end_index) {
cli::cli_abort(
"Invalid preview_window: start/end index out of range or invalid."
)
}
sliced_pupil_data <- pupil_data[start_index:end_index, ]
for (i in seq_along(pupil_steps)) {
st <- min(sliced_pupil_data$time_secs)
et <- max(sliced_pupil_data$time_secs)
if (grepl("z", pupil_steps[i])) {
y_units <- "(z)"
} else {
y_units <- "(a.u.)"
}
y_label <- paste("pupil size", y_units)
do.call(robust_plot, c(
list(y = sliced_pupil_data[[pupil_steps[i]]]),
plot_params,
list(
type = "l",
col = colors[i],
lwd = 2,
main = paste0(
gsub("_", " > ", gsub("pupil_", "", pupil_steps[i])),
if (is.list(x$timeseries) && !is.data.frame(x$timeseries)) {
sprintf(" (Run %d)", block)
} else {
""
},
"\n[", st, " - ", et, " seconds] | ",
"[index: ", preview_window_indices[1], " - ",
preview_window_indices[2], "]"
),
xlab = "time (ms)",
ylab = y_label
)
))
if (plot_distributions) {
plot_pupil_distribution(
data = pupil_data[[pupil_steps[i]]],
color = colors[i],
main = paste(paste0(
gsub("_", " > ", gsub("pupil_", "", pupil_steps[i])),
if (is.list(x$timeseries) && !is.data.frame(x$timeseries)) {
sprintf(" (Run %d)", block)
} else {
""
}
)),
xlab = y_label,
backuplab = "pupil size"
)
par(mfrow = c(1, 1), oma = c(0, 0, 0, 0))
}
}
par(mfrow = c(1, 1), oma = c(0, 0, 0, 0))
}
par(mfrow = c(1, 1), oma = c(0, 0, 0, 0))
}
draw_random_epochs <- function(x, n, d, hz) {
n_samples <- d * hz
min_timestamp <- min(x$time_orig)
max_timestamp <- max(x$time_orig)
if ((max_timestamp - min_timestamp) < d) {
cli::cli_abort("Example duration is longer than the duration of data.")
}
drawn_epochs <- list()
for (i in 1:n) {
rand_start <- sample(min_timestamp:(max_timestamp - n_samples), 1)
rand_end <- rand_start + n_samples
drawn_epochs[[i]] <- x |>
dplyr::filter(time_orig >= rand_start & time_orig < rand_end)
}
drawn_epochs
}
robust_plot <- function(y, x = NULL, ...) {
tryCatch(
{
if (length(y) == 0 || all(is.na(y))) {
cli::cli_alert_warning("No finite data to plot.")
return(invisible(NULL))
}
dots <- list(...)
col_user <- if ("col" %in% names(dots)) dots$col else "blue"
# store original y for getting NA positions
y_orig <- y
# if x is NULL, use 1:length(y)
if (is.null(x)) {
x_seq <- seq_along(y_orig)
} else {
x_seq <- x
}
# init placeholder line
plot(x_seq, ifelse(is.na(y_orig), NA, y_orig),
xlim = range(x_seq, na.rm = TRUE),
...
)
# add vertical lines where there are NAs
na_idx <- which(is.na(y_orig))
if (length(na_idx) > 0) {
abline(v = na_idx, col = "black", lty = 2)
}
# replace NA with -1 after drawing NA lines for continuity
y_clean <- y_orig
y_clean[is.na(y_clean)] <- -1
lines(x_seq, y_clean, col = col_user)
},
error = function(e) {
cli::cli_alert_info(
paste("An error occurred during plotting:", e$message)
)
},
warning = function(w) {
cli::cli_alert_warning(
paste("A warning occurred during plotting:", w$message)
)
}
)
}
plot_pupil_distribution <- function(data, color, main, xlab, backuplab = NULL) {
# safely handle user's current options
oldpar <- par(no.readonly = TRUE)
on.exit(par(oldpar))
par(mfrow = c(1, 1), oma = c(0, 0, 0, 0))
new_xlab <- if (!is.null(xlab)) {
xlab
} else if (!is.null(backuplab)) {
backuplab
} else {
"pupil size"
}
hist(
data,
main = main,
xlab = new_xlab,
ylab = "frequency (count)",
col = color,
border = "white",
breaks = "FD"
)
}
draw_na_lines <- function(x, y, ...) {
na_idx <- which(is.na(y))
abline(v = x[na_idx], col = "black", lty = 2, ...)
}
#' Internal helper to plot detrending overlay
#'
#' This function replicates the exact detrending visualization from the
#' `glassbox()` interactive preview mode. It uses `robust_plot()` to show the
#' most recent detrended pupil signal overlaid with the fitted linear trend.
#'
#' @param pupil_data A single block of pupil timeseries data
#' (e.g. `eyeris$timeseries$block_1`)
#' @param preview_n Number of columns for `par(mfrow)`. Default = 3.
#' @param plot_params A named list of additional parameters to forward to
#' `robust_plot()`
#' @param suppress_prompt Logical. Whether to skip prompting. Default = TRUE.
#'
#' @keywords internal
plot_detrend_overlay <- function(pupil_data,
pupil_steps,
preview_n = preview_n,
plot_params = list(),
suppress_prompt = TRUE) {
par(mfrow = c(1, 1), oma = c(0, 0, 0, 0))
detrend_step <- grep("_detrend$", pupil_steps, value = TRUE)
all_cols <- colnames(pupil_data)
detrend_fitted_index <- which(all_cols == "detrend_fitted_values")
if (length(detrend_fitted_index) == 0 || detrend_fitted_index == 1) {
cli::cli_alert_danger(
"detrend_fitted_values not found in eyeris S3 object."
)
prev_col <- NULL
} else {
prev_col <- all_cols[detrend_fitted_index - 1]
}
ydat <- pupil_data[[prev_col]]
xdat <- pupil_data$time_secs
do.call(robust_plot, c(
list(y = ydat, x = xdat),
plot_params,
list(
type = "l",
col = "black",
lwd = 2,
main = paste0(
"detrend:\n",
gsub("_", " > ", gsub("pupil_", "", detrend_step))
),
xlab = "tracker time (s)",
ylab = "pupil size (a.u.)"
)
))
lines(pupil_data$time_secs,
pupil_data$detrend_fitted_values,
type = "l", col = "blue", lwd = 2, lty = 1
)
legend("topleft",
legend = c("pupil timeseries", "linear trend"),
col = c("black", "blue"), lwd = 2, lty = c(1, 1)
)
par(mfrow = c(1, preview_n), oma = c(0, 0, 3, 0))
if (!suppress_prompt) prompt_user()
invisible(NULL)
}
Any scripts or data that you put into this service are public.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.