Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Delays transform modules reactivity until tab is active #1373

Merged
merged 27 commits into from
Oct 14, 2024
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
54818e6
add a note to teal_data_module constructor in its vignette about how …
m7pr Aug 30, 2024
fe4f707
move the note between files
m7pr Sep 2, 2024
fcc5d38
Merge branch 'main' into 1303_documentation@main
m7pr Sep 18, 2024
a7b3610
Merge branch 'main' into 1303_documentation@main
m7pr Oct 3, 2024
4a6bbb1
Merge branch 'main' into 1303_documentation@main
m7pr Oct 3, 2024
894910f
call_once_when
dependabot-preview[bot] Oct 7, 2024
a113824
comments
dependabot-preview[bot] Oct 7, 2024
47484e9
feat: uses reusable function to create observer that is executed once
averissimo Oct 9, 2024
ad504f8
fix: filter panel does not break with module_specific filters
averissimo Oct 9, 2024
fb986cb
Merge branch 'main' into 1303_postpone_transformer@main
averissimo Oct 9, 2024
c5a4c55
fix: module tab reactivity is only triggered after data is pulled
averissimo Oct 9, 2024
a503f90
feat: trigger filter manager in all modules at startup
averissimo Oct 10, 2024
c96d333
docs: adds warning that detects naïve case of eventReactive usage
averissimo Oct 10, 2024
6768e85
Merge remote-tracking branch 'origin/1303_documentation@main' into 13…
averissimo Oct 10, 2024
08ab298
docs: update message in warning and docs
averissimo Oct 10, 2024
ccc50a2
Update R/module_teal_data.R
averissimo Oct 11, 2024
6939d2c
docs: slight improvement on error message
averissimo Oct 11, 2024
07eb8f2
chore: correct lint errors and change argument name
averissimo Oct 11, 2024
5a60dfe
revert: argument name to condExpr as it needs to be a logical
averissimo Oct 11, 2024
126c547
chore: change argument name in line with observeEvent
averissimo Oct 11, 2024
5706d77
revert: removes protection against uninitialized filter manager modules
averissimo Oct 11, 2024
f4fdbd8
fix: typo
averissimo Oct 11, 2024
310bd07
fix: corrects test
averissimo Oct 11, 2024
69bf596
docs: add note to documentation of teal_transform_module
averissimo Oct 11, 2024
833f8f9
Update R/teal_data_module.R
averissimo Oct 14, 2024
31c5301
test: adds check if warning is thrown when teal_data_module returns e…
averissimo Oct 14, 2024
d885b10
tests: checks if modules are only called after teal_data_module is re…
averissimo Oct 14, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion R/module_filter_manager.R
Original file line number Diff line number Diff line change
Expand Up @@ -131,7 +131,10 @@ srv_filter_manager <- function(id, slices_global) {
module_labels,
simplify = FALSE,
function(module_label) {
available_slices <- slices_global$module_slices_api[[module_label]]$get_available_teal_slices()
available_slices_api <- slices_global$module_slices_api[[module_label]]
available_slices <- if (!is.null(available_slices_api)) {
averissimo marked this conversation as resolved.
Show resolved Hide resolved
available_slices_api$get_available_teal_slices()
}
global_ids <- sapply(slices_global$all_slices(), `[[`, "id", simplify = FALSE)
module_ids <- sapply(slices_global$slices_get(module_label), `[[`, "id", simplify = FALSE)
allowed_ids <- vapply(available_slices, `[[`, character(1L), "id")
Expand Down
207 changes: 125 additions & 82 deletions R/module_nested_tabs.R
Original file line number Diff line number Diff line change
Expand Up @@ -214,7 +214,11 @@ srv_teal_module.teal_modules <- function(id,
datasets = datasets,
slices_global = slices_global,
reporter = reporter,
is_active = reactive(is_active() && input$active_tab == module_id)
is_active = reactive(
is_active() &&
input$active_tab == module_id &&
identical(data_load_status(), "ok")
gogonzo marked this conversation as resolved.
Show resolved Hide resolved
)
)
},
simplify = FALSE
Expand All @@ -236,94 +240,95 @@ srv_teal_module.teal_module <- function(id,
is_active = reactive(TRUE)) {
logger::log_debug("srv_teal_module.teal_module initializing the module: { deparse1(modules$label) }.")
moduleServer(id = id, module = function(input, output, session) {
active_datanames <- reactive({
.resolve_module_datanames(data = data_rv(), modules = modules)
})
if (is.null(datasets)) {
datasets <- eventReactive(data_rv(), {
req(inherits(data_rv(), "teal_data"))
logger::log_debug("srv_teal_module@1 initializing module-specific FilteredData")
teal_data_to_filtered_data(data_rv(), datanames = active_datanames())
})
}

# manage module filters on the module level
# important:
# filter_manager_module_srv needs to be called before filter_panel_srv
# Because available_teal_slices is used in FilteredData$srv_available_slices (via srv_filter_panel)
# and if it is not set, then it won't be available in the srv_filter_panel
srv_module_filter_manager(modules$label, module_fd = datasets, slices_global = slices_global)
filtered_teal_data <- srv_filter_data(
"filter_panel",
datasets = datasets,
active_datanames = active_datanames,
data_rv = data_rv,
is_active = is_active
)
module_out <- reactiveVal()

is_transformer_failed <- reactiveValues()
transformed_teal_data <- srv_transform_data(
"data_transform",
data = filtered_teal_data,
transforms = modules$transformers,
modules = modules,
is_transformer_failed = is_transformer_failed
)
any_transformer_failed <- reactive({
any(unlist(reactiveValuesToList(is_transformer_failed)))
})
observeEvent(any_transformer_failed(), {
if (isTRUE(any_transformer_failed())) {
shinyjs::hide("teal_module_ui")
shinyjs::hide("validate_datanames")
shinyjs::show("transformer_failure_info")
} else {
shinyjs::show("teal_module_ui")
shinyjs::show("validate_datanames")
shinyjs::hide("transformer_failure_info")
call_once_when(is_active(), {
active_datanames <- reactive({
.resolve_module_datanames(data = data_rv(), modules = modules)
})
if (is.null(datasets)) {
datasets <- eventReactive(data_rv(), {
req(inherits(data_rv(), "teal_data"))
logger::log_debug("srv_teal_module@1 initializing module-specific FilteredData")
teal_data_to_filtered_data(data_rv(), datanames = active_datanames())
})
}
})

module_teal_data <- reactive({
req(inherits(transformed_teal_data(), "teal_data"))
all_teal_data <- transformed_teal_data()
module_datanames <- .resolve_module_datanames(data = all_teal_data, modules = modules)
.subset_teal_data(all_teal_data, module_datanames)
})

srv_validate_reactive_teal_data(
"validate_datanames",
data = module_teal_data,
modules = modules
)
# manage module filters on the module level
# important:
# filter_manager_module_srv needs to be called before filter_panel_srv
# Because available_teal_slices is used in FilteredData$srv_available_slices (via srv_filter_panel)
# and if it is not set, then it won't be available in the srv_filter_panel
srv_module_filter_manager(modules$label, module_fd = datasets, slices_global = slices_global)
averissimo marked this conversation as resolved.
Show resolved Hide resolved
filtered_teal_data <- srv_filter_data(
"filter_panel",
datasets = datasets,
active_datanames = active_datanames,
data_rv = data_rv,
is_active = is_active
)
is_transformer_failed <- reactiveValues()
transformed_teal_data <- srv_transform_data(
"data_transform",
data = filtered_teal_data,
transforms = modules$transformers,
modules = modules,
is_transformer_failed = is_transformer_failed
)
any_transformer_failed <- reactive({
any(unlist(reactiveValuesToList(is_transformer_failed)))
})

summary_table <- srv_data_summary("data_summary", module_teal_data)

# Call modules.
module_out <- reactiveVal(NULL)
if (!inherits(modules, "teal_module_previewer")) {
obs_module <- observeEvent(
# wait for module_teal_data() to be not NULL but only once:
ignoreNULL = TRUE,
once = TRUE,
eventExpr = module_teal_data(),
handlerExpr = {
module_out(.call_teal_module(modules, datasets, module_teal_data, reporter))
observeEvent(any_transformer_failed(), {
if (isTRUE(any_transformer_failed())) {
shinyjs::hide("teal_module_ui")
shinyjs::hide("validate_datanames")
shinyjs::show("transformer_failure_info")
} else {
shinyjs::show("teal_module_ui")
shinyjs::show("validate_datanames")
shinyjs::hide("transformer_failure_info")
}
})

module_teal_data <- reactive({
req(inherits(transformed_teal_data(), "teal_data"))
all_teal_data <- transformed_teal_data()
module_datanames <- .resolve_module_datanames(data = all_teal_data, modules = modules)
.subset_teal_data(all_teal_data, module_datanames)
})

srv_validate_reactive_teal_data(
"validate_datanames",
data = module_teal_data,
modules = modules
)
} else {
# Report previewer must be initiated on app start for report cards to be included in bookmarks.
# When previewer is delayed, cards are bookmarked only if previewer has been initiated (visited).
module_out(.call_teal_module(modules, datasets, module_teal_data, reporter))
}

# todo: (feature request) add a ReporterCard to the reporter as an output from the teal_module
# how to determine if module returns a ReporterCard so that reportPreviewer is needed?
# Should we insertUI of the ReportPreviewer then?
# What about attr(module, "reportable") - similar to attr(module, "bookmarkable")
if ("report" %in% names(module_out)) {
# (reactively) add card to the reporter
}
summary_table <- srv_data_summary("data_summary", module_teal_data)

# Call modules.
if (!inherits(modules, "teal_module_previewer")) {
obs_module <- call_once_when(
!is.null(module_teal_data()),
ignoreNULL = TRUE,
handlerExpr = {
module_out(.call_teal_module(modules, datasets, module_teal_data, reporter))
}
)
} else {
# Report previewer must be initiated on app start for report cards to be included in bookmarks.
# When previewer is delayed, cards are bookmarked only if previewer has been initiated (visited).
module_out(.call_teal_module(modules, datasets, module_teal_data, reporter))
}

# todo: (feature request) add a ReporterCard to the reporter as an output from the teal_module
# how to determine if module returns a ReporterCard so that reportPreviewer is needed?
# Should we insertUI of the ReportPreviewer then?
# What about attr(module, "reportable") - similar to attr(module, "bookmarkable")
if ("report" %in% names(module_out)) {
# (reactively) add card to the reporter
}
})

module_out
})
Expand Down Expand Up @@ -368,3 +373,41 @@ srv_teal_module.teal_module <- function(id,
)
}
}

#' Calls expression when condition is met
#'
#' Function postpones `handlerExpr` to the moment when `eventExpr` (condition) returns `TRUE`,
#' otherwise nothing happens.
#' @param condExpr logical expression determining moment when `handlerExpr` should be evaluated.
#' @param ... additional arguments passed to `observeEvent` with the exception of `eventExpr` that is not allowed.
#' @inheritParams shiny::observeEvent
#'
#' @return An observer.
#'
#' @keywords internal
call_once_when <- function(condExpr,
handlerExpr,
event.env = parent.frame(),
handler.env = parent.frame(),
...) {
if ("eventExpr" %in% names(rlang::list2(...))) {
stop("eventExpr is not allowed in call_once_when.")
}
averissimo marked this conversation as resolved.
Show resolved Hide resolved

event_quo <- rlang::new_quosure(substitute(condExpr), env = event.env)
gogonzo marked this conversation as resolved.
Show resolved Hide resolved
handler_quo <- rlang::new_quosure(substitute(handlerExpr), env = handler.env)

# When `condExpr` is TRUE, then `handlerExpr` is evaluated once.
activator <- reactive({
if (isTRUE(rlang::eval_tidy(event_quo))) {
TRUE
}
})

observeEvent(
eventExpr = activator(),
once = TRUE,
handlerExpr = rlang::eval_tidy(handler_quo),
...
)
}
19 changes: 10 additions & 9 deletions inst/WORDLIST
Original file line number Diff line number Diff line change
@@ -1,28 +1,29 @@
Biomarker
CDISC
Forkers
Hoffmann
MAEs
ORCID
Reproducibility
TLG
UI
UX
bookmarkable
CDISC
cloneable
customizable
favicon
favicons
Forkers
funder
Hoffmann
lockfile
MAEs
omics
ORCID
pre
programmatically
quosure
reactively
repo
Reproducibility
reproducibility
summarization
tabset
themer
theming
TLG
UI
uncheck
UX
42 changes: 42 additions & 0 deletions man/call_once_when.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading